UNPKG

wix-style-react

Version:
328 lines (266 loc) • 12.8 kB
"use strict"; var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); var _typeof = require("@babel/runtime/helpers/typeof"); Object.defineProperty(exports, "__esModule", { value: true }); exports["default"] = void 0; var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty")); var _slicedToArray2 = _interopRequireDefault(require("@babel/runtime/helpers/slicedToArray")); var _react = _interopRequireWildcard(require("react")); var _propTypes = _interopRequireDefault(require("prop-types")); var _AudioPlayerSt = require("./AudioPlayer.st.css"); var _Tooltip = _interopRequireDefault(require("../Tooltip")); var _IconButton = _interopRequireDefault(require("../IconButton")); var _Loader = _interopRequireDefault(require("../Loader")); var _Heading = _interopRequireDefault(require("../Heading")); var _PlayFilled = _interopRequireDefault(require("wix-ui-icons-common/PlayFilled")); var _PauseFilled = _interopRequireDefault(require("wix-ui-icons-common/PauseFilled")); var _constants = require("./constants"); var _AudioManager = require("./AudioManager/AudioManager"); var _utils = require("./utils"); function _getRequireWildcardCache(nodeInterop) { if (typeof WeakMap !== "function") return null; var cacheBabelInterop = new WeakMap(); var cacheNodeInterop = new WeakMap(); return (_getRequireWildcardCache = function _getRequireWildcardCache(nodeInterop) { return nodeInterop ? cacheNodeInterop : cacheBabelInterop; })(nodeInterop); } function _interopRequireWildcard(obj, nodeInterop) { if (!nodeInterop && obj && obj.__esModule) { return obj; } if (obj === null || _typeof(obj) !== "object" && typeof obj !== "function") { return { "default": obj }; } var cache = _getRequireWildcardCache(nodeInterop); if (cache && cache.has(obj)) { return cache.get(obj); } var newObj = {}; var hasPropertyDescriptor = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var key in obj) { if (key !== "default" && Object.prototype.hasOwnProperty.call(obj, key)) { var desc = hasPropertyDescriptor ? Object.getOwnPropertyDescriptor(obj, key) : null; if (desc && (desc.get || desc.set)) { Object.defineProperty(newObj, key, desc); } else { newObj[key] = obj[key]; } } } newObj["default"] = obj; if (cache) { cache.set(obj, newObj); } return newObj; } /** AudioPlayer */ var AudioPlayer = /*#__PURE__*/(0, _react.memo)( /*#__PURE__*/(0, _react.forwardRef)(function (_ref, ref) { var dataHook = _ref.dataHook, className = _ref.className, src = _ref.src, format = _ref.format, preload = _ref.preload, webAudioAPI = _ref.webAudioAPI, onLoad = _ref.onLoad, onLoadError = _ref.onLoadError, onPlay = _ref.onPlay, onPause = _ref.onPause, onEnd = _ref.onEnd, onSeek = _ref.onSeek; var _useState = (0, _react.useState)(true), _useState2 = (0, _slicedToArray2["default"])(_useState, 2), isSliderLocked = _useState2[0], setIsSliderLocked = _useState2[1]; var _useState3 = (0, _react.useState)(0), _useState4 = (0, _slicedToArray2["default"])(_useState3, 2), hoverPosition = _useState4[0], setHoverPosition = _useState4[1]; var _useState5 = (0, _react.useState)(0), _useState6 = (0, _slicedToArray2["default"])(_useState5, 2), handleSizeInPercentage = _useState6[0], setHandleSizeInPercentage = _useState6[1]; var _useState7 = (0, _react.useState)(false), _useState8 = (0, _slicedToArray2["default"])(_useState7, 2), playing = _useState8[0], setPlaying = _useState8[1]; var _useState9 = (0, _react.useState)(true), _useState10 = (0, _slicedToArray2["default"])(_useState9, 2), showDuration = _useState10[0], setShowDuration = _useState10[1]; var playPauseButtonRef = (0, _react.useRef)(); (0, _react.useImperativeHandle)(ref, function () { return { focus: function focus() { var _playPauseButtonRef$c; return playPauseButtonRef === null || playPauseButtonRef === void 0 ? void 0 : (_playPauseButtonRef$c = playPauseButtonRef.current) === null || _playPauseButtonRef$c === void 0 ? void 0 : _playPauseButtonRef$c.focus(); } }; }); var _onDestroy = (0, _react.useCallback)(function () { setPlaying(false); setShowDuration(true); }, [setPlaying, setShowDuration]); var _onEnd = (0, _react.useCallback)(function () { setShowDuration(true); setPlaying(false); onEnd && onEnd(); }, [onEnd, setShowDuration, setPlaying]); var _useAudioManager = (0, _AudioManager.useAudioManager)({ src: src, format: format, preload: preload, webAudioAPI: webAudioAPI, onLoad: onLoad, onLoadError: onLoadError, onPlay: onPlay, onSeek: onSeek, onPause: onPause, playing: playing, onDestroy: _onDestroy, onEnd: _onEnd, allowSeekLoop: isSliderLocked }), loadingState = _useAudioManager.loadingState, duration = _useAudioManager.duration, seek = _useAudioManager.seek, setSeek = _useAudioManager.setSeek; var isLoaded = loadingState === 'loaded'; var _hoverISO = (0, _react.useMemo)(function () { if (!isLoaded) { return (0, _utils.secondsToISO)(0, false, duration); } return (0, _utils.secondsToISO)((0, _utils.positionToSeconds)(hoverPosition, duration), true, duration); }, [isLoaded, hoverPosition, duration]); // takes the current seek (in seconds) and converts it to slider position. var _seekPercentage = (0, _react.useMemo)(function () { if (!isLoaded) { return 0; } return (0, _utils.secondsToPosition)(seek, duration); }, [isLoaded, seek, duration]); var _togglePlayPause = (0, _react.useCallback)(function () { setShowDuration(false); if (playing) { setPlaying(false); } else { setPlaying(true); } }, [playing, setPlaying, setShowDuration]); var _playPauseButtonContent = (0, _react.useMemo)(function () { if (loadingState === 'loading') { return /*#__PURE__*/_react["default"].createElement("span", { "data-hook": _constants.dataHooks.audioPlayerLoad }, /*#__PURE__*/_react["default"].createElement(_Loader["default"], { size: "tiny" })); } return playing ? /*#__PURE__*/_react["default"].createElement(_PauseFilled["default"], { "data-hook": _constants.dataHooks.audioPlayerPause }) : /*#__PURE__*/_react["default"].createElement(_PlayFilled["default"], { "data-hook": _constants.dataHooks.audioPlayerPlay }); }, [loadingState, playing]); var _setSliderPositions = (0, _react.useCallback)(function (x, width, clickX) { var positionInPixels = (clickX - x) / width * 100; var position = Math.min(Math.max(positionInPixels, 0), 100); setHandleSizeInPercentage(12 / width * 100); setHoverPosition(position); }, [setHoverPosition]); var _handleSliderMouseDown = (0, _react.useCallback)(function (event) { var clientX = event.clientX, currentTarget = event.currentTarget; var _currentTarget$getBou = currentTarget.getBoundingClientRect(), x = _currentTarget$getBou.x, width = _currentTarget$getBou.width; setIsSliderLocked(false); _setSliderPositions(x, width, clientX); }, [_setSliderPositions, setIsSliderLocked]); var _handleSliderMouseMove = (0, _react.useCallback)(function (event) { var clientX = event.clientX, currentTarget = event.currentTarget; var _currentTarget$getBou2 = currentTarget.getBoundingClientRect(), x = _currentTarget$getBou2.x, width = _currentTarget$getBou2.width; _setSliderPositions(x, width, clientX); }, [_setSliderPositions]); var _handleSliderMouseUp = (0, _react.useCallback)(function () { setIsSliderLocked(true); }, [setIsSliderLocked]); (0, _react.useEffect)(function () { window.addEventListener('mouseup', _handleSliderMouseUp); return function () { return window.removeEventListener('mouseup', _handleSliderMouseUp); }; }, [_handleSliderMouseUp]); // seek audio file to the slider location when dragged. (0, _react.useEffect)(function () { if (!isSliderLocked) { setSeek((0, _utils.positionToSeconds)(hoverPosition, duration)); setShowDuration(false); } }, [duration, hoverPosition, isSliderLocked, setSeek]); return /*#__PURE__*/_react["default"].createElement("div", { className: (0, _AudioPlayerSt.st)(_AudioPlayerSt.classes.root, className), "data-hook": dataHook }, /*#__PURE__*/_react["default"].createElement(_IconButton["default"], { ref: playPauseButtonRef, size: "small", onClick: _togglePlayPause, dataHook: _constants.dataHooks.audioPlayerPlayPause, className: _AudioPlayerSt.classes.playPauseButton }, _playPauseButtonContent), /*#__PURE__*/_react["default"].createElement("div", { "data-hook": _constants.dataHooks.audioPlayerSlider, className: _AudioPlayerSt.classes.slider, style: (0, _defineProperty2["default"])({}, _AudioPlayerSt.vars['audio-player-position'], "".concat(_seekPercentage, "%")), onMouseDown: _handleSliderMouseDown, onMouseMove: _handleSliderMouseMove }, /*#__PURE__*/_react["default"].createElement("div", { className: _AudioPlayerSt.classes.track }), /*#__PURE__*/_react["default"].createElement("div", { className: _AudioPlayerSt.classes.tooltip, style: { left: "".concat(hoverPosition, "%") } }, /*#__PURE__*/_react["default"].createElement(_Tooltip["default"], { content: "".concat(_hoverISO) }, /*#__PURE__*/_react["default"].createElement("div", { className: _AudioPlayerSt.classes.tooltipTarget }))), /*#__PURE__*/_react["default"].createElement("div", { "data-hook": _constants.dataHooks.audioPlayerSliderHandle, className: (0, _AudioPlayerSt.st)(_AudioPlayerSt.classes.handle, { grow: isLoaded && (!isSliderLocked || Math.abs(_seekPercentage - hoverPosition) < handleSizeInPercentage) }), style: { left: "".concat(_seekPercentage, "%") } })), /*#__PURE__*/_react["default"].createElement(_Heading["default"], { appearance: "H5", className: _AudioPlayerSt.classes.timer, dataHook: _constants.dataHooks.audioTimeIndicator }, showDuration ? (0, _utils.secondsToISO)(duration, isLoaded, duration) : (0, _utils.secondsToISO)(seek, isLoaded, duration))); })); AudioPlayer.displayName = 'AudioPlayer'; AudioPlayer.propTypes = { /** Applies a data-hook HTML attribute that can be used in the tests. */ dataHook: _propTypes["default"].string, /** Specifies a CSS class name to be appended to the component’s root element. */ className: _propTypes["default"].string, /** * Specifies a link to the source of the track to be loaded for the sound (URL or base64 data URI). * If a file has no extension, you will need to specify the extension using the format property. */ src: _propTypes["default"].string.isRequired, /** * Specifies a file format in situations where extraction does not work (such as a SoundCloud stream).<br/> * By default, AudioPlayer detects your file format from the extension. */ format: _propTypes["default"].string, /** * Determines what to download when the component is rendered: full file, its metadata or nothing at all. * When webAudioAPI = true you can only set it to either 'auto' or 'none'. * When webAudioAPI = false you can set it to 'auto', 'metadata' or 'none'. */ preload: _propTypes["default"].oneOf(['auto', 'metadata', 'none']), /** * Specifies whether to force web audio API. Use it for relatively small audio files only because you have to wait for the full file * to be downloaded and decoded before playing. Web Audio API allows advanced capabilities as described in * [Web Audio API](https://developer.mozilla.org/en-US/docs/Web/API/Web_Audio_API). */ webAudioAPI: _propTypes["default"].bool, /** * Defines a callback function which is called when audio is loaded. */ onLoad: _propTypes["default"].func, /** * Defines a callback function which is called every time audio fails to load. */ onLoadError: _propTypes["default"].func, /** * Defines a callback function which is called when audio is played. */ onPlay: _propTypes["default"].func, /** * Defines a callback function which is called when audio is paused. */ onPause: _propTypes["default"].func, /** Will be called when audio is ended. */ onEnd: _propTypes["default"].func, /** * Defines a callback function which is called when audio is seeked explicitly (i.e. when user drags the slider). */ onSeek: _propTypes["default"].func }; AudioPlayer.defaultProps = { preload: 'metadata', webAudioAPI: false }; var _default = AudioPlayer; exports["default"] = _default;