mirror of
https://github.com/Stremio/stremio-web.git
synced 2026-05-18 11:01:48 +00:00
157 lines
5.6 KiB
JavaScript
157 lines
5.6 KiB
JavaScript
const React = require('react');
|
|
const PropTypes = require('prop-types');
|
|
const classnames = require('classnames');
|
|
const { useDeepEqualEffect, useBinaryState } = require('stremio/common');
|
|
const BufferingLoader = require('./BufferingLoader');
|
|
const ControlBar = require('./ControlBar');
|
|
const Video = require('./Video');
|
|
const usePlayer = require('./usePlayer');
|
|
const styles = require('./styles');
|
|
|
|
const Player = ({ urlParams }) => {
|
|
const player = usePlayer(urlParams);
|
|
const [subtitlesPickerOpen, , , toggleSubtitlesPicker] = useBinaryState(false);
|
|
const [state, setState] = React.useReducer(
|
|
(state, nextState) => ({
|
|
...state,
|
|
...nextState
|
|
}),
|
|
{
|
|
paused: null,
|
|
time: null,
|
|
duration: null,
|
|
buffering: null,
|
|
volume: null,
|
|
muted: null,
|
|
subtitlesTracks: [],
|
|
selectedSubtitlesTrackId: null,
|
|
subtitlesSize: null,
|
|
subtitlesDelay: null,
|
|
subtitlesTextColor: null,
|
|
subtitlesBackgroundColor: null,
|
|
subtitlesOutlineColor: null
|
|
}
|
|
);
|
|
const videoRef = React.useRef(null);
|
|
const dispatch = React.useCallback((args) => {
|
|
if (videoRef.current !== null) {
|
|
videoRef.current.dispatch(args);
|
|
}
|
|
}, []);
|
|
const onImplementationChanged = React.useCallback((manifest) => {
|
|
manifest.props.forEach((propName) => {
|
|
dispatch({ observedPropName: propName });
|
|
});
|
|
}, []);
|
|
const onPropChanged = React.useCallback((propName, propValue) => {
|
|
setState({ [propName]: propValue });
|
|
}, []);
|
|
const onEnded = React.useCallback(() => {
|
|
console.log('ended');
|
|
}, []);
|
|
const onError = React.useCallback((error) => {
|
|
console.error(error);
|
|
}, []);
|
|
const onPlayRequested = React.useCallback(() => {
|
|
dispatch({ propName: 'paused', propValue: false });
|
|
}, []);
|
|
const onPauseRequested = React.useCallback(() => {
|
|
dispatch({ propName: 'paused', propValue: true });
|
|
}, []);
|
|
const onMuteRequested = React.useCallback(() => {
|
|
dispatch({ propName: 'muted', propValue: true });
|
|
}, []);
|
|
const onUnmuteRequested = React.useCallback(() => {
|
|
dispatch({ propName: 'muted', propValue: false });
|
|
}, []);
|
|
const onVolumeChangeRequested = React.useCallback((volume) => {
|
|
dispatch({ propName: 'volume', propValue: volume });
|
|
}, []);
|
|
const onSeekRequested = React.useCallback((time) => {
|
|
dispatch({ propName: 'time', propValue: time });
|
|
}, []);
|
|
useDeepEqualEffect(() => {
|
|
if (player.selected === null || player.selected.stream === null) {
|
|
dispatch({ commandName: 'stop' });
|
|
} else {
|
|
dispatch({
|
|
commandName: 'load',
|
|
commandArgs: {
|
|
stream: player.selected.stream
|
|
}
|
|
});
|
|
}
|
|
}, [player.selected && player.selected.stream]);
|
|
useDeepEqualEffect(() => {
|
|
dispatch({
|
|
commandName: 'addSubtitlesTracks',
|
|
commandArgs: {
|
|
tracks: player.subtitles_resources
|
|
.filter((subtitles_resource) => subtitles_resource.content.type === 'Ready')
|
|
.reduce((tracks, subtitles_resource) => {
|
|
return tracks.concat(subtitles_resource.content.content);
|
|
}, [])
|
|
}
|
|
});
|
|
}, [player.subtitles_resources]);
|
|
return (
|
|
<div className={styles['player-container']}>
|
|
<Video
|
|
ref={videoRef}
|
|
className={styles['layer']}
|
|
onEnded={onEnded}
|
|
onError={onError}
|
|
onPropValue={onPropChanged}
|
|
onPropChanged={onPropChanged}
|
|
onImplementationChanged={onImplementationChanged}
|
|
/>
|
|
<div className={styles['layer']} />
|
|
{
|
|
state.buffering ?
|
|
<BufferingLoader className={styles['layer']} />
|
|
:
|
|
null
|
|
}
|
|
<ControlBar
|
|
className={classnames(styles['layer'], styles['control-bar-layer'])}
|
|
paused={state.paused}
|
|
time={state.time}
|
|
duration={state.duration}
|
|
volume={state.volume}
|
|
muted={state.muted}
|
|
subtitlesTracks={state.subtitlesTracks}
|
|
selectedSubtitlesTrackId={state.selectedSubtitlesTrackId}
|
|
subtitlesSize={state.subtitlesSize}
|
|
subtitlesDelay={state.subtitlesDelay}
|
|
subtitlesTextColor={state.subtitlesTextColor}
|
|
subtitlesBackgroundColor={state.subtitlesBackgroundColor}
|
|
subtitlesOutlineColor={state.subtitlesOutlineColor}
|
|
onPlayRequested={onPlayRequested}
|
|
onPauseRequested={onPauseRequested}
|
|
onMuteRequested={onMuteRequested}
|
|
onUnmuteRequested={onUnmuteRequested}
|
|
onVolumeChangeRequested={onVolumeChangeRequested}
|
|
onSeekRequested={onSeekRequested}
|
|
onToggleSubtitlesPicker={toggleSubtitlesPicker}
|
|
/>
|
|
{
|
|
subtitlesPickerOpen ?
|
|
<div className={styles['layer']} />
|
|
:
|
|
null
|
|
}
|
|
</div>
|
|
);
|
|
};
|
|
|
|
Player.propTypes = {
|
|
urlParams: PropTypes.exact({
|
|
stream: PropTypes.string,
|
|
transportUrl: PropTypes.string,
|
|
type: PropTypes.string,
|
|
id: PropTypes.string,
|
|
videoId: PropTypes.string
|
|
})
|
|
};
|
|
|
|
module.exports = Player;
|