bigbuffet-rw/app/soapbox/features/audio/index.js

536 lines
16 KiB
JavaScript
Raw Normal View History

2021-05-17 17:39:08 -07:00
import classNames from 'classnames';
import debounce from 'lodash/debounce';
import throttle from 'lodash/throttle';
import PropTypes from 'prop-types';
import React from 'react';
import { defineMessages, injectIntl } from 'react-intl';
2022-01-10 14:01:24 -08:00
import Icon from 'soapbox/components/icon';
import { formatTime, getPointerPosition, fileNameFromURL } from 'soapbox/features/video';
2021-05-17 17:39:08 -07:00
import Visualizer from './visualizer';
2020-06-18 13:10:57 -07:00
const messages = defineMessages({
2021-05-17 17:39:08 -07:00
play: { id: 'video.play', defaultMessage: 'Play' },
pause: { id: 'video.pause', defaultMessage: 'Pause' },
mute: { id: 'video.mute', defaultMessage: 'Mute sound' },
unmute: { id: 'video.unmute', defaultMessage: 'Unmute sound' },
download: { id: 'video.download', defaultMessage: 'Download file' },
2020-06-18 13:10:57 -07:00
});
2021-05-17 17:39:08 -07:00
const TICK_SIZE = 10;
const PADDING = 180;
2020-06-18 13:10:57 -07:00
2021-05-17 17:39:08 -07:00
export default @injectIntl
2020-06-18 13:24:49 -07:00
class Audio extends React.PureComponent {
2020-06-18 13:10:57 -07:00
static propTypes = {
src: PropTypes.string.isRequired,
alt: PropTypes.string,
2021-05-17 17:39:08 -07:00
poster: PropTypes.string,
duration: PropTypes.number,
width: PropTypes.number,
height: PropTypes.number,
editable: PropTypes.bool,
fullscreen: PropTypes.bool,
2020-06-18 13:10:57 -07:00
intl: PropTypes.object.isRequired,
2021-05-17 17:39:08 -07:00
cacheWidth: PropTypes.func,
backgroundColor: PropTypes.string,
foregroundColor: PropTypes.string,
accentColor: PropTypes.string,
currentTime: PropTypes.number,
autoPlay: PropTypes.bool,
volume: PropTypes.number,
muted: PropTypes.bool,
deployPictureInPicture: PropTypes.func,
2020-06-18 13:10:57 -07:00
};
state = {
2021-05-17 17:39:08 -07:00
width: this.props.width,
2020-06-18 13:10:57 -07:00
currentTime: 0,
2021-05-17 17:39:08 -07:00
buffer: 0,
duration: null,
2020-06-18 13:10:57 -07:00
paused: true,
muted: false,
2021-05-17 17:39:08 -07:00
volume: 0.5,
dragging: false,
2020-06-18 13:10:57 -07:00
};
2021-05-17 17:39:08 -07:00
constructor(props) {
super(props);
this.visualizer = new Visualizer(TICK_SIZE);
2020-06-18 13:10:57 -07:00
}
setPlayerRef = c => {
this.player = c;
2021-05-17 17:39:08 -07:00
if (this.player) {
this._setDimensions();
2020-06-18 13:10:57 -07:00
}
}
2021-05-17 17:39:08 -07:00
_pack() {
return {
src: this.props.src,
volume: this.audio.volume,
muted: this.audio.muted,
currentTime: this.audio.currentTime,
poster: this.props.poster,
backgroundColor: this.props.backgroundColor,
foregroundColor: this.props.foregroundColor,
accentColor: this.props.accentColor,
};
}
2020-06-18 13:10:57 -07:00
2021-05-17 17:39:08 -07:00
_setDimensions() {
const width = this.player.offsetWidth;
const height = this.props.fullscreen ? this.player.offsetHeight : (width / (16 / 9));
2021-05-17 17:39:08 -07:00
if (this.props.cacheWidth) {
this.props.cacheWidth(width);
2020-06-18 13:10:57 -07:00
}
2021-05-17 17:39:08 -07:00
this.setState({ width, height });
2020-06-18 13:10:57 -07:00
}
setSeekRef = c => {
this.seek = c;
}
setVolumeRef = c => {
this.volume = c;
}
2021-05-17 17:39:08 -07:00
setAudioRef = c => {
this.audio = c;
if (this.audio) {
this.setState({ volume: this.audio.volume, muted: this.audio.muted });
}
}
setCanvasRef = c => {
this.canvas = c;
this.visualizer.setCanvas(c);
}
componentDidMount() {
window.addEventListener('scroll', this.handleScroll);
window.addEventListener('resize', this.handleResize, { passive: true });
}
componentDidUpdate(prevProps, prevState) {
if (prevProps.src !== this.props.src || this.state.width !== prevState.width || this.state.height !== prevState.height || prevProps.accentColor !== this.props.accentColor) {
this._clear();
this._draw();
}
}
componentWillUnmount() {
window.removeEventListener('scroll', this.handleScroll);
window.removeEventListener('resize', this.handleResize);
if (!this.state.paused && this.audio && this.props.deployPictureInPicture) {
this.props.deployPictureInPicture('audio', this._pack());
}
}
togglePlay = () => {
if (!this.audioContext) {
this._initAudioContext();
}
if (this.state.paused) {
this.setState({ paused: false }, () => this.audio.play());
} else {
this.setState({ paused: true }, () => this.audio.pause());
}
}
handleResize = debounce(() => {
if (this.player) {
this._setDimensions();
}
}, 250, {
trailing: true,
});
2020-06-18 13:10:57 -07:00
handlePlay = () => {
this.setState({ paused: false });
2021-05-17 17:39:08 -07:00
if (this.audioContext && this.audioContext.state === 'suspended') {
this.audioContext.resume();
}
this._renderCanvas();
2020-06-18 13:10:57 -07:00
}
handlePause = () => {
this.setState({ paused: true });
2021-05-17 17:39:08 -07:00
if (this.audioContext) {
this.audioContext.suspend();
}
2020-06-18 13:10:57 -07:00
}
2021-05-17 17:39:08 -07:00
handleProgress = () => {
const lastTimeRange = this.audio.buffered.length - 1;
if (lastTimeRange > -1) {
this.setState({ buffer: Math.ceil(this.audio.buffered.end(lastTimeRange) / this.audio.duration * 100) });
}
}
toggleMute = () => {
const muted = !this.state.muted;
this.setState({ muted }, () => {
this.audio.muted = muted;
2020-06-18 13:10:57 -07:00
});
}
handleVolumeMouseDown = e => {
document.addEventListener('mousemove', this.handleMouseVolSlide, true);
document.addEventListener('mouseup', this.handleVolumeMouseUp, true);
document.addEventListener('touchmove', this.handleMouseVolSlide, true);
document.addEventListener('touchend', this.handleVolumeMouseUp, true);
this.handleMouseVolSlide(e);
e.preventDefault();
e.stopPropagation();
}
handleVolumeMouseUp = () => {
document.removeEventListener('mousemove', this.handleMouseVolSlide, true);
document.removeEventListener('mouseup', this.handleVolumeMouseUp, true);
document.removeEventListener('touchmove', this.handleMouseVolSlide, true);
document.removeEventListener('touchend', this.handleVolumeMouseUp, true);
}
handleMouseDown = e => {
document.addEventListener('mousemove', this.handleMouseMove, true);
document.addEventListener('mouseup', this.handleMouseUp, true);
document.addEventListener('touchmove', this.handleMouseMove, true);
document.addEventListener('touchend', this.handleMouseUp, true);
this.setState({ dragging: true });
2020-06-18 13:24:49 -07:00
this.audio.pause();
2020-06-18 13:10:57 -07:00
this.handleMouseMove(e);
e.preventDefault();
e.stopPropagation();
}
handleMouseUp = () => {
document.removeEventListener('mousemove', this.handleMouseMove, true);
document.removeEventListener('mouseup', this.handleMouseUp, true);
document.removeEventListener('touchmove', this.handleMouseMove, true);
document.removeEventListener('touchend', this.handleMouseUp, true);
this.setState({ dragging: false });
2020-06-18 13:24:49 -07:00
this.audio.play();
2020-06-18 13:10:57 -07:00
}
handleMouseMove = throttle(e => {
const { x } = getPointerPosition(this.seek, e);
2021-05-17 17:39:08 -07:00
const currentTime = this.audio.duration * x;
2020-06-18 13:10:57 -07:00
if (!isNaN(currentTime)) {
2021-05-17 17:39:08 -07:00
this.setState({ currentTime }, () => {
this.audio.currentTime = currentTime;
});
2020-06-18 13:10:57 -07:00
}
2021-05-17 17:39:08 -07:00
}, 15);
2020-06-18 13:10:57 -07:00
2021-05-17 17:39:08 -07:00
handleTimeUpdate = () => {
this.setState({
currentTime: this.audio.currentTime,
duration: this.audio.duration,
});
}
handleMouseVolSlide = throttle(e => {
const { x } = getPointerPosition(this.volume, e);
if (!isNaN(x)) {
2021-05-17 17:39:08 -07:00
this.setState({ volume: x }, () => {
this.audio.volume = x;
});
}
}, 15);
handleScroll = throttle(() => {
if (!this.canvas || !this.audio) {
return;
}
const { top, height } = this.canvas.getBoundingClientRect();
const inView = (top <= (window.innerHeight || document.documentElement.clientHeight)) && (top + height >= 0);
if (!this.state.paused && !inView) {
2020-06-18 13:24:49 -07:00
this.audio.pause();
2021-05-17 17:39:08 -07:00
if (this.props.deployPictureInPicture) {
this.props.deployPictureInPicture('audio', this._pack());
}
this.setState({ paused: true });
2020-06-18 13:10:57 -07:00
}
2021-05-17 17:39:08 -07:00
}, 150, { trailing: true });
2020-06-18 13:10:57 -07:00
2021-05-17 17:39:08 -07:00
handleMouseEnter = () => {
this.setState({ hovered: true });
2020-06-18 13:10:57 -07:00
}
2021-05-17 17:39:08 -07:00
handleMouseLeave = () => {
this.setState({ hovered: false });
2020-06-18 13:10:57 -07:00
}
handleLoadedData = () => {
2021-05-17 17:39:08 -07:00
const { autoPlay, currentTime, volume, muted } = this.props;
this.setState({ duration: this.audio.duration });
if (currentTime) {
this.audio.currentTime = currentTime;
2020-06-18 13:10:57 -07:00
}
2021-05-17 17:39:08 -07:00
if (volume !== undefined) {
this.audio.volume = volume;
}
if (muted !== undefined) {
this.audio.muted = muted;
}
if (autoPlay) {
this.togglePlay();
2020-06-18 13:10:57 -07:00
}
}
2021-05-17 17:39:08 -07:00
_initAudioContext() {
2022-07-21 10:19:36 -07:00
// eslint-disable-next-line compat/compat
2021-05-17 17:39:08 -07:00
const AudioContext = window.AudioContext || window.webkitAudioContext;
const context = new AudioContext();
const source = context.createMediaElementSource(this.audio);
this.visualizer.setAudioContext(context, source);
source.connect(context.destination);
this.audioContext = context;
2020-06-18 13:10:57 -07:00
}
2021-05-17 17:39:08 -07:00
handleDownload = () => {
fetch(this.props.src).then(res => res.blob()).then(blob => {
const element = document.createElement('a');
const objectURL = URL.createObjectURL(blob);
2020-06-18 13:10:57 -07:00
2021-05-17 17:39:08 -07:00
element.setAttribute('href', objectURL);
element.setAttribute('download', fileNameFromURL(this.props.src));
document.body.appendChild(element);
element.click();
document.body.removeChild(element);
URL.revokeObjectURL(objectURL);
}).catch(err => {
console.error(err);
});
2020-06-18 13:10:57 -07:00
}
2021-05-17 17:39:08 -07:00
_renderCanvas() {
requestAnimationFrame(() => {
if (!this.audio) return;
2020-06-18 13:10:57 -07:00
2021-05-17 17:39:08 -07:00
this.handleTimeUpdate();
this._clear();
this._draw();
2020-06-18 13:10:57 -07:00
2021-05-17 17:39:08 -07:00
if (!this.state.paused) {
this._renderCanvas();
}
});
}
2020-06-18 13:10:57 -07:00
2021-05-17 17:39:08 -07:00
_clear() {
this.visualizer.clear(this.state.width, this.state.height);
}
_draw() {
this.visualizer.draw(this._getCX(), this._getCY(), this._getAccentColor(), this._getRadius(), this._getScaleCoefficient());
}
_getRadius() {
return parseInt(((this.state.height || this.props.height) - (PADDING * this._getScaleCoefficient()) * 2) / 2);
}
_getScaleCoefficient() {
return (this.state.height || this.props.height) / 982;
}
_getCX() {
return Math.floor(this.state.width / 2) || null;
}
_getCY() {
return Math.floor(this._getRadius() + (PADDING * this._getScaleCoefficient())) || null;
}
_getAccentColor() {
return this.props.accentColor || '#ffffff';
}
_getBackgroundColor() {
return this.props.backgroundColor || '#000000';
}
_getForegroundColor() {
return this.props.foregroundColor || '#ffffff';
}
seekBy(time) {
const currentTime = this.audio.currentTime + time;
if (!isNaN(currentTime)) {
this.setState({ currentTime }, () => {
this.audio.currentTime = currentTime;
});
2020-06-18 13:10:57 -07:00
}
2021-05-17 17:39:08 -07:00
}
2020-06-18 13:10:57 -07:00
2021-05-17 17:39:08 -07:00
handleAudioKeyDown = e => {
// On the audio element or the seek bar, we can safely use the space bar
// for playback control because there are no buttons to press
if (e.key === ' ') {
e.preventDefault();
e.stopPropagation();
this.togglePlay();
}
}
handleKeyDown = e => {
switch (e.key) {
2022-05-11 14:06:35 -07:00
case 'k':
e.preventDefault();
e.stopPropagation();
this.togglePlay();
break;
case 'm':
e.preventDefault();
e.stopPropagation();
this.toggleMute();
break;
case 'j':
e.preventDefault();
e.stopPropagation();
this.seekBy(-10);
break;
case 'l':
e.preventDefault();
e.stopPropagation();
this.seekBy(10);
break;
2021-05-17 17:39:08 -07:00
}
}
render() {
const { src, intl, alt, editable } = this.props;
const { paused, muted, volume, currentTime, buffer, dragging } = this.state;
const duration = this.state.duration || this.props.duration;
const progress = Math.min((currentTime / duration) * 100, 100);
2020-06-18 13:10:57 -07:00
2021-05-17 17:39:08 -07:00
return (
<div className={classNames('audio-player', { editable })} ref={this.setPlayerRef} style={{ backgroundColor: this._getBackgroundColor(), color: this._getForegroundColor(), width: '100%', height: this.props.fullscreen ? '100%' : (this.state.height || this.props.height) }} onMouseEnter={this.handleMouseEnter} onMouseLeave={this.handleMouseLeave} tabIndex='0' onKeyDown={this.handleKeyDown}>
<audio
2020-06-18 13:10:57 -07:00
src={src}
2021-05-17 17:39:08 -07:00
ref={this.setAudioRef}
preload='auto'
2020-06-18 13:10:57 -07:00
onPlay={this.handlePlay}
onPause={this.handlePause}
onProgress={this.handleProgress}
2021-05-17 17:39:08 -07:00
onLoadedData={this.handleLoadedData}
crossOrigin='anonymous'
/>
2020-06-18 13:10:57 -07:00
2021-05-17 17:39:08 -07:00
<canvas
role='button'
tabIndex='0'
className='audio-player__canvas'
width={this.state.width}
height={this.state.height}
style={{ width: '100%', position: 'absolute', top: 0, left: 0 }}
ref={this.setCanvasRef}
onClick={this.togglePlay}
onKeyDown={this.handleAudioKeyDown}
title={alt}
aria-label={alt}
/>
2020-06-18 13:10:57 -07:00
2021-10-04 22:44:49 -07:00
{this.props.poster && <img
2021-05-17 17:39:08 -07:00
src={this.props.poster}
alt=''
width={(this._getRadius() - TICK_SIZE) * 2 || null}
height={(this._getRadius() - TICK_SIZE) * 2 || null}
style={{ position: 'absolute', left: this._getCX(), top: this._getCY(), transform: 'translate(-50%, -50%)', borderRadius: '50%', pointerEvents: 'none' }}
2021-10-04 22:44:49 -07:00
/>}
2020-06-18 13:10:57 -07:00
2021-05-17 17:39:08 -07:00
<div className='video-player__seek' onMouseDown={this.handleMouseDown} ref={this.setSeekRef}>
<div className='video-player__seek__buffer' style={{ width: `${buffer}%` }} />
<div className='video-player__seek__progress' style={{ width: `${progress}%`, backgroundColor: this._getAccentColor() }} />
<span
className={classNames('video-player__seek__handle', { active: dragging })}
tabIndex='0'
style={{ left: `${progress}%`, backgroundColor: this._getAccentColor() }}
onKeyDown={this.handleAudioKeyDown}
/>
</div>
2020-06-18 13:10:57 -07:00
2021-05-17 17:39:08 -07:00
<div className='video-player__controls active'>
<div className='video-player__buttons-bar'>
<div className='video-player__buttons left'>
<button type='button' title={intl.formatMessage(paused ? messages.play : messages.pause)} aria-label={intl.formatMessage(paused ? messages.play : messages.pause)} className='player-button' onClick={this.togglePlay}><Icon src={paused ? require('@tabler/icons/player-play.svg') : require('@tabler/icons/player-pause.svg')} /></button>
<button type='button' title={intl.formatMessage(muted ? messages.unmute : messages.mute)} aria-label={intl.formatMessage(muted ? messages.unmute : messages.mute)} className='player-button' onClick={this.toggleMute}><Icon src={muted ? require('@tabler/icons/volume-3.svg') : require('@tabler/icons/volume.svg')} /></button>
2021-05-17 17:39:08 -07:00
<div className={classNames('video-player__volume', { active: this.state.hovered })} ref={this.setVolumeRef} onMouseDown={this.handleVolumeMouseDown}>
<div className='video-player__volume__current' style={{ width: `${volume * 100}%`, backgroundColor: this._getAccentColor() }} />
2020-06-18 13:10:57 -07:00
<span
2021-05-17 17:39:08 -07:00
className='video-player__volume__handle'
2020-06-18 13:10:57 -07:00
tabIndex='0'
2021-05-17 17:39:08 -07:00
style={{ left: `${volume * 100}%`, backgroundColor: this._getAccentColor() }}
2020-06-18 13:10:57 -07:00
/>
</div>
2021-05-17 17:39:08 -07:00
<span className='video-player__time'>
<span className='video-player__time-current'>{formatTime(Math.floor(currentTime))}</span>
{duration && (<>
<span className='video-player__time-sep'>/</span>
<span className='video-player__time-total'>{formatTime(Math.floor(duration))}</span>
</>)}
2020-06-24 19:53:25 -07:00
</span>
2021-05-17 17:39:08 -07:00
</div>
2020-06-18 13:10:57 -07:00
2021-05-17 17:39:08 -07:00
<div className='video-player__buttons right'>
<a
title={intl.formatMessage(messages.download)}
aria-label={intl.formatMessage(messages.download)}
className='video-player__download__icon player-button'
href={this.props.src}
download
target='_blank'
>
<Icon src={require('@tabler/icons/download.svg')} />
2021-05-17 17:39:08 -07:00
</a>
2020-06-18 13:10:57 -07:00
</div>
</div>
</div>
</div>
);
}
2020-06-24 19:53:25 -07:00
}