@ndla/ui 38.0.0 → 39.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/es/AudioPlayer/Controls.js +52 -40
- package/es/BlogPost/BlogPost.js +4 -4
- package/es/Grid/Grid.js +41 -0
- package/es/Grid/index.js +9 -0
- package/es/index.js +1 -0
- package/lib/AudioPlayer/Controls.js +52 -40
- package/lib/BlogPost/BlogPost.d.ts +1 -1
- package/lib/BlogPost/BlogPost.js +4 -4
- package/lib/Grid/Grid.d.ts +15 -0
- package/lib/Grid/Grid.js +48 -0
- package/lib/Grid/index.d.ts +9 -0
- package/lib/Grid/index.js +13 -0
- package/lib/index.d.ts +2 -0
- package/lib/index.js +7 -0
- package/package.json +17 -17
- package/src/AudioPlayer/Controls.tsx +8 -5
- package/src/BlogPost/BlogPost.stories.tsx +15 -12
- package/src/BlogPost/BlogPost.tsx +1 -1
- package/src/Grid/Grid.stories.tsx +68 -0
- package/src/Grid/Grid.tsx +63 -0
- package/src/Grid/index.ts +10 -0
- package/src/KeyFigure/KeyFigure.stories.tsx +10 -8
- package/src/index.ts +2 -0
|
@@ -26,45 +26,45 @@ import { ButtonV2, IconButtonV2 } from '@ndla/button';
|
|
|
26
26
|
import { jsx as _jsx } from "@emotion/react/jsx-runtime";
|
|
27
27
|
import { jsxs as _jsxs } from "@emotion/react/jsx-runtime";
|
|
28
28
|
var ControlsWrapper = /*#__PURE__*/_styled("div", {
|
|
29
|
-
target: "
|
|
29
|
+
target: "e1gaeajv20",
|
|
30
30
|
label: "ControlsWrapper"
|
|
31
31
|
})("border:1px solid ", colors.brand.lighter, ";display:flex;align-items:center;justify-content:center;background:", colors.white, ";font-family:", fonts.sans, ";gap:", spacing.xsmall, ";padding:", spacing.small, " ", spacing.normal, ";", mq.range({
|
|
32
32
|
until: breakpoints.tabletWide
|
|
33
|
-
}), "{display:grid;padding:", spacing.small, ";grid-template-columns:1fr repeat(5, auto) 1fr;grid-template-areas:'track track track track track track track' '. speed backwards play forwards volume .';}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAmBkC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <IconButtonV2 variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </IconButtonV2>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
33
|
+
}), "{display:grid;padding:", spacing.small, ";grid-template-columns:1fr repeat(5, auto) 1fr;grid-template-areas:'track track track track track track track' '. speed backwards play forwards volume .';}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAmBkC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
34
34
|
var PlayButton = /*#__PURE__*/_styled(IconButtonV2, {
|
|
35
|
-
target: "
|
|
35
|
+
target: "e1gaeajv19",
|
|
36
36
|
label: "PlayButton"
|
|
37
37
|
})(mq.range({
|
|
38
38
|
until: breakpoints.tabletWide
|
|
39
|
-
}), "{grid-area:play;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAsCuC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <IconButtonV2 variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </IconButtonV2>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
39
|
+
}), "{grid-area:play;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAsCuC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
40
40
|
var Forward15SecButton = /*#__PURE__*/_styled(IconButtonV2, {
|
|
41
|
-
target: "
|
|
41
|
+
target: "e1gaeajv18",
|
|
42
42
|
label: "Forward15SecButton"
|
|
43
43
|
})(mq.range({
|
|
44
44
|
until: breakpoints.tabletWide
|
|
45
|
-
}), "{grid-area:forwards;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA4C+C","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <IconButtonV2 variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </IconButtonV2>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
45
|
+
}), "{grid-area:forwards;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA4C+C","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
46
46
|
var Back15SecButton = /*#__PURE__*/_styled(IconButtonV2, {
|
|
47
|
-
target: "
|
|
47
|
+
target: "e1gaeajv17",
|
|
48
48
|
label: "Back15SecButton"
|
|
49
49
|
})(mq.range({
|
|
50
50
|
until: breakpoints.tabletWide
|
|
51
|
-
}), "{grid-area:backwards;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAiD4C","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <IconButtonV2 variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </IconButtonV2>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
51
|
+
}), "{grid-area:backwards;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAiD4C","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
52
52
|
var SpeedButton = /*#__PURE__*/_styled(ButtonV2, {
|
|
53
|
-
target: "
|
|
53
|
+
target: "e1gaeajv16",
|
|
54
54
|
label: "SpeedButton"
|
|
55
55
|
})(mq.range({
|
|
56
56
|
until: breakpoints.tabletWide
|
|
57
|
-
}), "{grid-area:speed;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAuDoC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <IconButtonV2 variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </IconButtonV2>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
57
|
+
}), "{grid-area:speed;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAuDoC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
58
58
|
var SpeedList = /*#__PURE__*/_styled(Content, {
|
|
59
|
-
target: "
|
|
59
|
+
target: "e1gaeajv15",
|
|
60
60
|
label: "SpeedList"
|
|
61
|
-
})("background:", colors.white, ";border:1px solid ", colors.brand.lighter, ";border-radius:", misc.borderRadius, ";padding:5px 10px;display:flex;flex-direction:column;justify-content:center;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA6DiC","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 { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <IconButtonV2 variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </IconButtonV2>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
61
|
+
})("background:", colors.white, ";border:1px solid ", colors.brand.lighter, ";border-radius:", misc.borderRadius, ";padding:5px 10px;display:flex;flex-direction:column;justify-content:center;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA6DiC","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 { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
62
62
|
var SpeedValueButton = /*#__PURE__*/_styled(Item, {
|
|
63
|
-
target: "
|
|
63
|
+
target: "e1gaeajv14",
|
|
64
64
|
label: "SpeedValueButton"
|
|
65
|
-
})("height:28px;padding:0 14px;cursor:pointer;font-weight:", fonts.weight.semibold, ";", fonts.sizes('14px'), ";color:", colors.text.light, ";display:flex;justify-content:center;&:hover,&:active,&:focus,&[data-highlighted]{background:", colors.brand.greyLighter, ";border-radius:5px;outline:none;color:", colors.text.primary, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAuEqC","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 { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <IconButtonV2 variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </IconButtonV2>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
65
|
+
})("height:28px;padding:0 14px;cursor:pointer;font-weight:", fonts.weight.semibold, ";", fonts.sizes('14px'), ";color:", colors.text.light, ";display:flex;justify-content:center;&:hover,&:active,&:focus,&[data-highlighted]{background:", colors.brand.greyLighter, ";border-radius:5px;outline:none;color:", colors.text.primary, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAuEqC","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 { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
66
66
|
var SpeedSelectedMark = /*#__PURE__*/_styled("span", {
|
|
67
|
-
target: "
|
|
67
|
+
target: "e1gaeajv13",
|
|
68
68
|
label: "SpeedSelectedMark"
|
|
69
69
|
})(process.env.NODE_ENV === "production" ? {
|
|
70
70
|
name: "as7yir",
|
|
@@ -72,21 +72,21 @@ var SpeedSelectedMark = /*#__PURE__*/_styled("span", {
|
|
|
72
72
|
} : {
|
|
73
73
|
name: "as7yir",
|
|
74
74
|
styles: "border-radius:50%;background:#d1372e;width:6px;height:6px;margin:6px 0 0 2px",
|
|
75
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA2FqC","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 { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <IconButtonV2 variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </IconButtonV2>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
|
|
75
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA2FqC","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 { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
|
|
76
76
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
77
77
|
});
|
|
78
78
|
var Time = /*#__PURE__*/_styled("div", {
|
|
79
|
-
target: "
|
|
79
|
+
target: "e1gaeajv12",
|
|
80
80
|
label: "Time"
|
|
81
|
-
})(fonts.sizes('16px'), ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAmGuB","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 { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <IconButtonV2 variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </IconButtonV2>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
81
|
+
})(fonts.sizes('16px'), ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAmGuB","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 { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
82
82
|
var ProgressWrapper = /*#__PURE__*/_styled("div", {
|
|
83
|
-
target: "
|
|
83
|
+
target: "e1gaeajv11",
|
|
84
84
|
label: "ProgressWrapper"
|
|
85
85
|
})("flex:1;display:flex;align-items:center;gap:", spacing.small, ";", mq.range({
|
|
86
86
|
until: breakpoints.tabletWide
|
|
87
|
-
}), "{grid-area:track;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAuGkC","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 { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <IconButtonV2 variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </IconButtonV2>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
87
|
+
}), "{grid-area:track;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAuGkC","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 { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
88
88
|
var SliderWrapper = /*#__PURE__*/_styled(SliderRoot, {
|
|
89
|
-
target: "
|
|
89
|
+
target: "e1gaeajv10",
|
|
90
90
|
label: "SliderWrapper"
|
|
91
91
|
})(process.env.NODE_ENV === "production" ? {
|
|
92
92
|
name: "5l00y9",
|
|
@@ -94,15 +94,15 @@ var SliderWrapper = /*#__PURE__*/_styled(SliderRoot, {
|
|
|
94
94
|
} : {
|
|
95
95
|
name: "5l00y9",
|
|
96
96
|
styles: "cursor:pointer;flex:1;position:relative;display:flex;align-items:center;user-select:none;touch-action:none",
|
|
97
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAiHwC","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 { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <IconButtonV2 variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </IconButtonV2>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
|
|
97
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAiHwC","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 { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
|
|
98
98
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
99
99
|
});
|
|
100
100
|
var StyledTrack = /*#__PURE__*/_styled(Track, {
|
|
101
|
-
target: "
|
|
101
|
+
target: "e1gaeajv9",
|
|
102
102
|
label: "StyledTrack"
|
|
103
|
-
})("height:4px;width:100%;background:", colors.brand.lighter, ";border-radius:7px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA2HiC","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 { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <IconButtonV2 variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </IconButtonV2>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
103
|
+
})("height:4px;width:100%;background:", colors.brand.lighter, ";border-radius:7px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA2HiC","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 { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
104
104
|
var StyledRange = /*#__PURE__*/_styled(Range, {
|
|
105
|
-
target: "
|
|
105
|
+
target: "e1gaeajv8",
|
|
106
106
|
label: "StyledRange"
|
|
107
107
|
})(process.env.NODE_ENV === "production" ? {
|
|
108
108
|
name: "ufqu4g",
|
|
@@ -110,11 +110,11 @@ var StyledRange = /*#__PURE__*/_styled(Range, {
|
|
|
110
110
|
} : {
|
|
111
111
|
name: "ufqu4g",
|
|
112
112
|
styles: "position:absolute;height:4px;background:#5cbc80;border-radius:7px",
|
|
113
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAkIiC","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 { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <IconButtonV2 variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </IconButtonV2>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
|
|
113
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAkIiC","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 { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
|
|
114
114
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
115
115
|
});
|
|
116
116
|
var StyledThumb = /*#__PURE__*/_styled(SliderThumb, {
|
|
117
|
-
target: "
|
|
117
|
+
target: "e1gaeajv7",
|
|
118
118
|
label: "StyledThumb"
|
|
119
119
|
})(process.env.NODE_ENV === "production" ? {
|
|
120
120
|
name: "183frwh",
|
|
@@ -122,21 +122,27 @@ var StyledThumb = /*#__PURE__*/_styled(SliderThumb, {
|
|
|
122
122
|
} : {
|
|
123
123
|
name: "183frwh",
|
|
124
124
|
styles: "display:block;width:20px;height:20px;background:#5cbc80;border-radius:50%;outline:none",
|
|
125
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAyIuC","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 { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <IconButtonV2 variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </IconButtonV2>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
|
|
125
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAyIuC","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 { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
|
|
126
126
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
127
127
|
});
|
|
128
128
|
var VolumeWrapper = /*#__PURE__*/_styled(PopoverRoot, {
|
|
129
|
-
target: "
|
|
129
|
+
target: "e1gaeajv6",
|
|
130
130
|
label: "VolumeWrapper"
|
|
131
|
-
})("
|
|
132
|
-
|
|
133
|
-
}), "{grid-area:volume;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAkJyC","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 { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <IconButtonV2 variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </IconButtonV2>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
131
|
+
})(process.env.NODE_ENV === "production" ? {
|
|
132
|
+
name: "16evi40",
|
|
133
|
+
styles: "position:relative;display:flex;justify-content:center"
|
|
134
|
+
} : {
|
|
135
|
+
name: "16evi40",
|
|
136
|
+
styles: "position:relative;display:flex;justify-content:center",
|
|
137
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAkJyC","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 { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
|
|
138
|
+
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
139
|
+
});
|
|
134
140
|
var VolumeList = /*#__PURE__*/_styled(PopoverContent, {
|
|
135
|
-
target: "
|
|
141
|
+
target: "e1gaeajv5",
|
|
136
142
|
label: "VolumeList"
|
|
137
|
-
})("box-shadow:0 14px 20px -5px rgba(32, 88, 143, 0.17);border-radius:60px;background:", colors.white, ";padding:", spacing.small, ";border:1px solid ", colors.brand.lighter, ";height:128px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA2JyC","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 { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <IconButtonV2 variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </IconButtonV2>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
143
|
+
})("box-shadow:0 14px 20px -5px rgba(32, 88, 143, 0.17);border-radius:60px;background:", colors.white, ";padding:", spacing.small, ";border:1px solid ", colors.brand.lighter, ";height:128px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAwJyC","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 { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
138
144
|
var VolumeSliderWrapper = /*#__PURE__*/_styled(SliderRoot, {
|
|
139
|
-
target: "
|
|
145
|
+
target: "e1gaeajv4",
|
|
140
146
|
label: "VolumeSliderWrapper"
|
|
141
147
|
})(process.env.NODE_ENV === "production" ? {
|
|
142
148
|
name: "1naj2yc",
|
|
@@ -144,21 +150,27 @@ var VolumeSliderWrapper = /*#__PURE__*/_styled(SliderRoot, {
|
|
|
144
150
|
} : {
|
|
145
151
|
name: "1naj2yc",
|
|
146
152
|
styles: "cursor:pointer;height:100%;position:relative;display:flex;flex-direction:column;align-items:center;user-select:none;touch-action:none",
|
|
147
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAoK8C","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 { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <IconButtonV2 variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </IconButtonV2>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
|
|
153
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAiK8C","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 { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
|
|
148
154
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
149
155
|
});
|
|
156
|
+
var VolumeButton = /*#__PURE__*/_styled(IconButtonV2, {
|
|
157
|
+
target: "e1gaeajv3",
|
|
158
|
+
label: "VolumeButton"
|
|
159
|
+
})(mq.range({
|
|
160
|
+
until: breakpoints.tabletWide
|
|
161
|
+
}), "{grid-area:volume;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA4KyC","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 { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
150
162
|
var VolumeSliderBackground = /*#__PURE__*/_styled(Track, {
|
|
151
163
|
target: "e1gaeajv2",
|
|
152
164
|
label: "VolumeSliderBackground"
|
|
153
|
-
})("height:100%;width:5px;background:", colors.brand.lighter, ";border-radius:7px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA+K4C","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 { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <IconButtonV2 variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </IconButtonV2>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
165
|
+
})("height:100%;width:5px;background:", colors.brand.lighter, ";border-radius:7px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAkL4C","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 { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
154
166
|
var VolumeSliderSelected = /*#__PURE__*/_styled(Range, {
|
|
155
167
|
target: "e1gaeajv1",
|
|
156
168
|
label: "VolumeSliderSelected"
|
|
157
|
-
})("position:absolute;width:5px;background:", colors.brand.secondary, ";border-radius:7px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAsL0C","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 { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <IconButtonV2 variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </IconButtonV2>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
169
|
+
})("position:absolute;width:5px;background:", colors.brand.secondary, ";border-radius:7px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAyL0C","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 { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
158
170
|
var VolumeSliderHandle = /*#__PURE__*/_styled(SliderThumb, {
|
|
159
171
|
target: "e1gaeajv0",
|
|
160
172
|
label: "VolumeSliderHandle"
|
|
161
|
-
})("display:block;width:20px;height:20px;background:", colors.brand.primary, ";border-radius:50%;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA6L8C","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 { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <IconButtonV2 variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </IconButtonV2>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
173
|
+
})("display:block;width:20px;height:20px;background:", colors.brand.primary, ";border-radius:50%;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAgM8C","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 { Root, Trigger, Item, Content, DropdownMenuPortal } from '@radix-ui/react-dropdown-menu';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(Content)`\n  background: ${colors.white};\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: ${misc.borderRadius};\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(Item)`\n  height: 28px;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('14px')};\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n\n        <Root>\n          <Trigger asChild>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </Trigger>\n          <DropdownMenuPortal>\n            <SpeedList side=\"top\">\n              {speedValues.map((speed) => (\n                <SpeedValueButton key={speed} onSelect={() => setSpeedValue(speed)}>\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              ))}\n            </SpeedList>\n          </DropdownMenuPortal>\n        </Root>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
162
174
|
var formatTime = function formatTime(seconds) {
|
|
163
175
|
var minutes = Math.floor(seconds / 60);
|
|
164
176
|
var currentSeconds = seconds % 60;
|
|
@@ -327,7 +339,7 @@ var Controls = function Controls(_ref) {
|
|
|
327
339
|
}), _jsxs(VolumeWrapper, {
|
|
328
340
|
children: [_jsx(PopoverTrigger, {
|
|
329
341
|
asChild: true,
|
|
330
|
-
children: _jsx(
|
|
342
|
+
children: _jsx(VolumeButton, {
|
|
331
343
|
variant: "ghost",
|
|
332
344
|
colorTheme: "greyLighter",
|
|
333
345
|
"aria-label": t('audio.controls.adjustVolume'),
|