Learn More

Try out the world’s first micro-repo!

Learn More

Developing a React Video Player with Personalized Controls

Developing a React Video Player with Personalized Controls
Stylized image of a React video player.

Web technology has developed a lot over the years; Javascript, in particular, has played an enormous role in it. By involving React, the development and use of UI have become better and more efficient. With React, including an audio-video player on any website is easy. A simple React player plugin may assist you in performing the task in a few minutes. In this article, we'll talk about how to build a customizable React video player from scratch.

Introduction

React-Player is a React component that plays audio-visual files from various URLs, including file paths, YouTube links, Facebook links, Twitch links, SoundCloud links, Streamable links, Vimeo links, Wistia links, Mixcloud links, DailyMotion links, Kaltura links, and so on.

In this post, we'll use the React-Player component to build a video player (which you can install with the React Player npm), while focusing more on its functionalities than its appearance. To follow this tutorial, you'll need to have Node.js installed locally on your machine and have some familiarity with Material UI, Javascript, and React.

Features of React-Player

  • Customizable and easy-to-use
  • Runs on diffrent URLs, including file paths
  • Supports video looping and play back rates

Installing Dependencies

Running the following commands in the terminal allows us to start by setting up the project and installing the necessary dependencies:

yarn create react-app react-video-player

cd react-video-player

yarn add @mui/material @emotion/react @emotion/styled @mui/icons-material react-player screenfull

yarn start

Let's start by importing ReactPlayer into our App.js file and updating it.

import ReactPlayer from "react-player";
import "./App.css";
import ReactPlayer from "react-player";
import { Container } from "@mui/material";

function App() {
return (
<div className="video_container">
<div>
<h2>React player</h2>
</div>
<Container maxWidth="md" justify="center">
<div className="player__wrapper">
<ReactPlayer
className="player"
url="https://bucket-viewer.s3.amazonaws.com/viewer1664370329252.mp4"
width="100%"
height="100%"
playing={true}
muted={true}
/>
<Control />
</div>
</Container>
</div>
);
}

The react-player component is now inside a Container wrapper. Additionally, we supplied the link to the example video as the URL prop's value. Finally, we also provided the player with a set width and height of 100%, so it will be responsive.

Then, add the following code to your App.css file.

.App {
text-align: center;
}
.video_container {
display: flex;
flex-direction: column;
justify-content: center;
align-items: center;
width: 100%;
}

.player__wrapper {
position: relative;
}

.player {
border: 2px solid #7b2cbf;
object-fit: cover;
padding: 0;
margin: 0;
}

h2 {
color: #7b2cbf;
}

To enable playback within our video player React, we added the playing and muted props to the preceding code. In addition, we set a value of true in place of the default value of false for the props. As a result, the playing footage created by the above code is shown in the picture below:

Beautiful nature scenes in our React player.

Observe that the video player is not interactive and lacks any controls. The react-player package contains a prop called control that, by default, has a value of false. When the value is modified to true, the React player controls are automatically added.

React Player Functionalities

Let's begin by creating a new folder in the src folder named Components. Then, create the Control.jsx and Control.css files as two new files inside the newly formed folder.

src
├── App.css
├── App.js
├── Components
├── Control.css
├──Control.jsx

Let’s Code

First, we’ll import the required packages for our Control.jsx file:

import React from "react";
import { makeStyles, Slider, withStyles, Button, Tooltip, Popover,Grid

} from "@material-ui/core";
import {
FastForward,
FastRewind,
Pause,
PlayArrow,
SkipNext,
VolumeUp,
} from "@material-ui/icons";
import "./Control.css";

The control.jsx file is made up of three containers:

  1. The heading container
  2. The middle container (Contains the Play, Rewind and Fast forward buttons)
  3. The bottom container (Contains the Slider, volume and playback rate buttons)

The Heading Container

<div className="top_container">
<h2>Video PLayer</h2>
</div>

The Middle Container

<div className="mid__container">
<div className="icon__btn">
<FastRewind fontSize="medium" />
</div>

<div className="icon__btn">
<Pause fontSize="medium" />
</div>

<div className="icon__btn">
<FastForward fontSize="medium" />
</div>
</div>

The Bottom Container

<div className="bottom__container">
<div className="slider__container">
<PrettoSlider />
</div>
<div className="control__box">
<div className="inner__controls">
<div className="icon__btn">
<PlayArrow fontSize="medium" />
</div>
<div className="icon__btn">
<SkipNext fontSize="medium" />
</div>
<div className="icon__btn">
<VolumeUp fontSize="medium" />
</div>

<Slider
className={`${classes.volumeSlider}`} />
<span>5/20</span>
</div>
</div>
</div>

Our control.jsx React player component will be like this:

const Control = () => {
return(
<div className="control_Container">
// <-- The Heading Container -->
// <-- The Middle Container -->
// <-- The Bottom Container -->
</div>
)
}

Let's discuss the .control_Container class-named div element. It’s a distinct container that, if hovered above the custom video player, appears as an overlay.

The following are the styles for our Control.jsx file:

.control_Container {
background-color: rgba(0, 0, 0, 0.6);
position: absolute;
top: 0;
bottom: 0;
right: 0;
left: 0;
flex-direction: column;
z-index: 1;
display: flex;
justify-content: space-between;
}

.top_container {
display: flex;
align-items: center;
justify-content: space-between;
margin: 5px 20px;
}

.mid__container {
display: flex;
justify-content: center;
align-items: center;
}

.icon__btn {
padding: 0 10px;
color: #7b2cbf;
}

.slider__container {
/* width: 100%; */
display: flex;
align-items: center;
padding: 0 16px;
}

.control__box {
display: flex;
align-items: center;
justify-content: space-between;
}

.inner__controls {
display: flex;
padding: 10px 0;
align-items: center;
width: 50%;
}

span {
color: #9556cc;
font-size: 0.8rem;
margin-left: 10px;
}

.second__control {
display: flex;
align-items: center;
}

Additionally, we style some of the components by adding the Material UI's makeStyles and withStyles to our Control. Jsx

const useStyles = makeStyles({
volumeSlider: {
width: "100px",
color: "#9556CC",
},

bottomIcons: {
color: "#999",
padding: "12px 8px",


"&:hover": {
color: "#fff",
},
},
});

const PrettoSlider = withStyles({
root: {
height: "20px",
color: "#9556CC",
display: "flex",
justifyContent: "center",
alignItems: "center",
},
thumb: {
height: 20,
width: 20,
backgroundColor: "#9556CC",
border: "2px solid currentColor",
marginTop: -3,
marginLeft: -12,
"&:focus, &:hover, &$active": {
boxShadow: "inherit",
},
},
active: {},
valueLabel: {
left: "calc(-50% + 4px)",
},
track: {
height: 5,
borderRadius: 4,
width: "100%",
},
rail: {
height: 5,
borderRadius: 4,
},
})(Slider);

Adding the following code to your App.js file will import React’s Controls component, which we also need to do:

import "./App.css";
import ReactPlayer from "react-player";
import { Container } from "@material-ui/core";
import Control from "./Components/Control";

function App() {
return (
<div className="video_container">
<div>
<h2>React player</h2>
</div>
<Container maxWidth="md" justify="center">
<div className="player__wrapper">
<ReactPlayer
className="player"
url="https://bucket-viewer.s3.amazonaws.com/viewer1664370329252.mp4"
width="100%"
height="100%"
playing={true}
muted={true}
/>
<Control />
</div>
</Container>
</div>
);
}

export default App;

If you followed closely, our React custom video player ought to appear like this:

Our finished custom React player.

Handling Play and Pause

In this section, we'll concentrate on the capabilities and logic of the custom controls for the React player, starting with playing and pausing.

Navigate to the App.js file, where we'll begin by defining a state named videoState. An object with different properties represents this state:

const [videoState, setVideoState] = useState({
playing: true,
muted: false,
volume: 0.5,
played: 0,
seeking: false,
Buffer : true
});
//Destructuring the properties from the videoState
const {playing, muted, volume, playbackRate, played, seeking, buffer} = videoState

We destructure the videoState using the ES6 syntax to get its properties.

Following that, we need to create a function that toggles the play and pause capabilities. To do this, we'll spread the previous state and toggle the necessary state.

const playPauseHandler = () => {
//plays and pause the video (toggling)
setVideoState({ ...videoState, playing: !videoState.playing });
};

Replace the true values for the playing and muted properties with the appropriate values of playing and mute from the destructured videoState in the ReactPlayer component.

<div className="player__wrapper">
<ReactPlayer
className="player"
url="https://bucket-viewer.s3.amazonaws.com/viewer1664370329252.mp4"
width="100%"
height="100%"
playing={playing}
muted={muted}
/>
<Control onPlayPause={playPauseHandler} playing={playing}/>
</div>

The playing prop in ReactPlayer is used to set the value to true or false, playing or pausing the video.

To allow the play and pause buttons to access this feature, we pass in the function to the Control component as a prop in the example above.

We’d also update our Control.jsx file by passing the onPlayPause into the onClick of our pause button.

<div className="icon__btn" onClick={onPlayPause}>
<Pause fontSize="medium" />
</div>

The playing attribute value provided within the videoState, which we passed as a prop to the Control component, is used to render the icon using the ternary operator conditionally. For example, the play icon should appear when the video is paused, and when it is playing, the pause icon should appear.

<div className="icon__btn" onClick={onPlayPause}>
{playing ? (
<Pause fontSize="medium" />
) : (
<PlayArrow fontSize="medium" />
)}{" "}
</div>

With the addition of these functionalities, we can easily play and pause our video, like so:

The React player, now capable of being played and paused.

Handling Rewind and Fast Forward

We want to fast-forward the video by 10 seconds and rewind it by 5 seconds, much like with other video players. The ReactPlayer must be referenced using the useRef hook to obtain the video's current time before implementing this feature.

We’ll begin by importing the useRef hook and creating a reference for the React Player.

import ReactPlayer from "react-player";
import { useRef } from "react";

function App() {
//react player reference
const videoPlayerRef = useRef(null);
return (
<ReactPlayer
ref={videoPlayerRef} //updating the react player ref
className="player"
url="https://bucket-viewer.s3.amazonaws.com/viewer1664370329252.mp4"
width="100%"
height="100%"
playing={playing}
muted={muted}
/>
);
}

export default App;

Let's implement a rewindHandler function that we'll call anytime the rewind button is double-tapped. This function will use two methods for the rewind and fast-forward features obtained from the videoPlayerRef.

  • seekTo: Seek the given number of seconds, or a fraction if the amount is between 0 and 1
  • getCurrentTime(): Returns the number of seconds that have been played
const rewindHandler = () => {
//Rewinds the video player reducing 5
videoPlayerRef.current.seekTo(videoPlayerRef.current.getCurrentTime() - 5);
};

The rewindHandler subtracts 5 seconds from the current video time, which is what you're thinking about when you think about the fastForwardHandler adding 10 seconds to the current video time.

const fastFowardHandler = () => {
//FastFowards the video player by adding 10
videoPlayerRef.current.seekTo(videoPlayerRef.current.getCurrentTime() + 10);
};

Next, we pass the functions as props to the Control component:

<Control
onPlayPause={playPauseHandler}
playing={playing}
onRewind={rewindHandler}
onForward ={handleFastFoward }
/>

Then, we receive the props and pass the two functions into their respective buttons.

<div className="icon__btn" onDoubleClick={onRewind}>
<FastRewind fontSize="medium" />
</div>


<div className="icon__btn">
<FastForward fontSize="medium" onDoubleClick={onForward}/>
</div>

We can now fast-forward and rewind our video using the code above. However, you'll also notice that the video player's slider doesn't move to the current time when we rewind or fast-forward the video. Let's fix that, since we're problem solvers, right?

Seek Functionality

The ReactPlayer has an onProgress prop callback for this feature, which is a Callback that contains played and loaded progress as a fraction as well as playedSeconds and loaded seconds.

const progressHandler = (state) => {

if (!seeking) {
setVideoState({ ...videoState, ...state });
}
};

The progressHandler function accepts an argument called state. This state argument indicates an object with contained states. Every time the video player seek bar (the player's timing) updates, some sets of states are altered, and new values are returned. We keep the values of the previously modified States and our videoState properties in this function. We want this to happen, but only when the value of the seeking property in our videoState is negated.

Next, we pass in the progressHandler function to the onProgess prop in the ReactPlayer:

<ReactPlayer
ref={videoPlayerRef}
className="player"
url="https://bucket-viewer.s3.amazonaws.com/viewer1664370329252.mp4"
width="100%"
height="100%"
playing={playing}
muted={muted}
onProgress = {progressHandler}
/>

Then, to update our seek slider to the appropriate time of the video, we pass the played state that we've destructured from the videoState to the control component as a prop.

<Control
onPlayPause={playPauseHandler}
playing={playing}
onRewind={rewindHandler}
onForward ={handleFastFoward }
played ={played}
/>

The PrettoSlider in our control component then has to be updated. The maximum value for our PrettoSlider must be 100, and the minimum value must be set to 0. The value prop is the slider's current e.target.value, which multiplies the value of the played props by 100.

<PrettoSlider
min = {0}
max = {100}
value = {played * 100}
/>

Furthermore, we need to write functions that will enable us to use the video player slider to look for a specific moment. To do this, we'd create a seekHandler and a seekMouseUpHandler to help us achieve this feature.

const seekHandler = (e, value) => {
setVideoState({ ...videoState, played: parseFloat(value) / 100 });
};

const seekMouseUpHandler = (e, value ) => {
setVideoState({ ...videoState, seeking: false });
videoPlayerRef.current.seekTo(value / 100);
};

We then pass these functions as props to the control component. The parameter given to the function is the value we immediately receive whenever the slider moves. Then, since played only accepts values between 0 and 1, we only updated the played value to the value argument divided by 100 when updating the videoState.

Additionally, we gave the mouseSeekUpHandler function an argument called value. Spreading the previous state and changing only the seeking state value to false, which allows us to update the videoState. Then, we change the video's current time to the desired time.

<Control
onPlayPause={playPauseHandler}
playing={playing}
onRewind={rewindHandler}
onForward ={handleFastFoward }
played ={played}
onSeek ={seekHandler}
onSeekMouseUp ={seekMouseUpHandler}
/>

To use these two functions, the PrettoSlider has two props which we can use for this functionality: onChange and onChangeCommitted. According to the MUI docs, the onChange prop is a Callback function that fires whenever the slider's value changes, while the onChangeCommitted is a Callback function that fires when the mouse moves up.

This is what our video player should look like now:

The current functions of our video player.

Volume Functionality

We should have the option to mute and increase/decrease the video’s volume in our React player. The icon should then change to ‘muted’ or ‘volume-up’, depending on the volume setting in that situation.

Let’s head over to our App.js file.

const volumeChangeHandler = (e, value) => {
const newVolume = parseFloat(value) / 100;
setVideoState({
...videoState,
volume: newVolume,
muted: Number(newVolume) === 0 ? true : false, // volume === 0 then muted
})

};

const volumeSeekUpHandler = (e, value) => {
const newVolume = parseFloat(value) / 100;
setVideoState({
...videoState,
volume: newVolume,
muted: newVolume === 0 ? true : false,
})};

There is a volume prop for this functionality in the ReactPlayer where we pass the destructured volume from our videoState.

<ReactPlayer
ref={videoPlayerRef}
className="player"
url="https://bucket-viewer.s3.amazonaws.com/viewer1664370329252.mp4"
width="100%"
height="100%"
playing={playing}
volume = {volume}
muted={muted}
onProgress={progressHandler}
/>

The volume slider needs to define two functions, one for the onChange event and the other for the onChangeCommitted event, just like the PrettoSlider.

Now, we pass them as props to the control component:

<Control
onPlayPause={playPauseHandler}
playing={playing}
onRewind={rewindHandler}
onForward={handleFastFoward}
played={played}
onSeek={seekHandler}
onSeekMouseUp={seekMouseUpHandler}

Volume={volume}
onVolumeChangeHandler = {volumeChangeHandler}
onVolumeSeekUp = {volumeSeekUpHandler}
/>

The volume slider is then updated by including new props, the onChange event, the onChangeCommitted event, and value, and passing the onVolumeChangeHandler, onVolumeSeekUp, and volume props.

<Slider
className={`${classes.volumeSlider}`}
onChange={onVolumeChangeHandler}
value={volume * 100}
onChangeCommitted={onVolumeSeekUp}
/>

Handling the Muted State

We ought to be able to choose between muted and volume up whenever the mute button is clicked. The symbol should switch to ‘muted’ or ‘volume-up’ depending on the volume level.

We’ll start by writing a function that handles the mute functionality in our App.js file.

const muteHandler = () => {
//Mutes the video player
setVideoState({ ...videoState, muted: !videoState.muted });
};

In the muteHandler, we retained everything in the videoState and changed the muted property to have a value opposite of the current muted state, thereby toggling it and then passing the muted props to the control component.

<Control
onPlayPause={playPauseHandler}
playing={playing}
onRewind={rewindHandler}
onForward={handleFastFoward}
played={played}
onSeek={seekHandler}
onSeekMouseUp={seekMouseUpHandler}
volume ={volume}
onVolumeChangeHandler = {volumeChangeHandler}
onVolumeSeekUp = {volumeSeekUpHandler}
mute = {muted}
onMute = {muteHandler}
/>

<div className="icon__btn" onClick={onMute} >
{mute ? (
<VolumeOff fontSize="medium" />
) : (
<VolumeUp fontSize="medium" />
)}
</div>

The value of the videoState field's muted state determines how the two icons in the above code would render conditionally. It also passes in a function to the wrapper, which toggles the mute functionality.

This is how our video should function:

Raising and lowering the volume of our video.

Video Time Functionality

Video players, as we all know, typically show the video's current time. To accomplish this, we use some instance methods that React Player offers:

  • getCurrentTime: Returns the number of seconds that have been played.
  • getDuration: Returns the currently playing media’s total duration (in seconds).
  • getSecondsLoaded: Returns the number of seconds that have been loaded.
const currentTime = videoPlayerRef.current? videoPlayerRef.current.getCurrentTime(): "00:00";

const duration = videoPlayerRef.current? videoPlayerRef.current.getDuration(): "00:00";

We still need to format the returned time according to our preferences, even though the currentTime and duration are rendered conditionally using ternary operators.

A function that formats the timing must be created. After accepting an argument, this function will return the time in a specific format.

To use the function in App.js, we create a new file called Format.js inside the component folder.

export const formatTime = (time) => {
//formarting duration of video
if (isNaN(time)) {
return "00:00";
}

const date = new Date(time * 1000);
const hours = date.getUTCHours();
const minutes = date.getUTCMinutes();
const seconds = date.getUTCSeconds().toString().padStart(2, "0");
if (hours) {
//if video have hours
return `${hours}:${minutes.toString().padStart(2, "0")} `;
} else return `${minutes}:${seconds}`;
};

Following that, import the formatTime method found in the Format.js file. Therefore, in the App.js file, we can use it. Then, two formatTime calls are made, with the currentTime and duration being passed as arguments.

const formatCurrentTime = formatTime(currentTime)

const formatDuration = formatTime(duration)

We then pass the formatCurrentTime and formatDuration as props to the control component.

<Control
onPlayPause={playPauseHandler}
playing={playing}
onRewind={rewindHandler}
onForward={handleFastFoward}
played={played}
onSeek={seekHandler}
onSeekMouseUp={seekMouseUpHandler}
volume ={volume}
onVolumeChangeHandler = {volumeChangeHandler}
onVolumeSeekUp = {volumeSeekUpHandler}
mute = {muted}
onMute = {muteHandler}
duration = {formatDuration}
currentTime = {formatCurrentTime}
/>

Finally, modify the default timing value to this:

<span>{ currentTime} : {duration}</span>

Now, you can watch the length of the video as it plays and see how much time has passed. Watch the timing as the seek bar is dragged.

Adjusting the time in our custom video player.

The Control container div is frequently resting on the player, as you can see. Let's make it more interesting by having it resemble other video players— let’s make it visible whenever we hover on the video player.

Creating a Disappearing Seek Bar

To achieve this feature, we also need to create a reference of the div with classname of control_Container in the App.js file:

const controlRef = useRef(null)

And pass it to the control component:

<Control
controlRef = {controlRef}
onPlayPause={playPauseHandler}
playing={playing}
onRewind={rewindHandler}
onForward={handleFastFoward}
played={played}
onSeek={seekHandler}
onSeekMouseUp={seekMouseUpHandler}
volume ={volume}
onVolumeChangeHandler = {volumeChangeHandler}
onVolumeSeekUp = {volumeSeekUpHandler}
mute = {muted}
onMute = {muteHandler}
playRate = {playbackRate}
onPlayRate = {playBackRateHandler}
duration = {formatDuration}
currentTime = {formatCurrentTime}
/>

We then pass it to the ref property on the control_Container div in the control component:

const Control = ({controlRef}) => {
return (
<div className="control_Container" ref ={controlRef}>
// <-- body of the control component --->

</div>
);
};

export default Control;

Next, we head over to our App.js and declare a new variable called count which will be equal to 0:

Let count = 0

const App = () => {
return (
<div>
// <-- body of the App component --->

</div>
);
};

export default App;

We include an if statement inside the progressHandler function to determine whether the count exceeds 3, at which point we use the controlRef and set the visibility to hidden. By doing so, the control component div is hidden.

const progressHandler = (state) => {
if (count > 3){

// toggling player control container

controlRef.current.style.visibility = "hidden";

} else if (controlRef.current.style.visibility === "visible") {
count += 1;
}

if (!seeking) {
setVideoState({ ...videoState, ...state });
}
};

We must create another function to make the control div visible whenever the player is hovered over.

const mouseMoveHandler = () => {
controlRef.current.style.visibility = "visible";
count = 0;
};

const App = () => {
return (
<div>
// <-- body of the App component --->
<div onMouseDown = {mouseMoveHandler} >
<ReactPlayer/>
<Control/>
</div>
</div>
);
};

export default App;

Our player now looks like this:

The React player with a disappearing seek bar.

Buffering Functionality in our React Player

When we watch videos online, they frequently stall, primarily because of poor network connections. Let's add a similar feature to our video player.

The ReactPlayer provides two props that accept a callback for this feature: onBuffer and onBufferEnd.

const bufferStartHandler = () => {
console.log("Bufering.......");
setVideoState({...videoState , buffer: true})
};

const bufferEndHandler = () => {
console.log("buffering stoped ,,,,,,play");
setVideoState({...videoState , buffer: false})
};

These two functions will be passed to the onBuffer and onBufferEnd.

<ReactPlayer
ref={videoPlayerRef}
className="player"
url="https://bucket-viewer.s3.amazonaws.com/viewer1664370329252.mp4"
width="100%"
height="100%"
playing={playing}
volume={volume}
muted={muted}
onProgress={progressHandler}
playbackRate={playBackRateHandler}
onBuffer={bufferStartHandler}
onBufferEnd={bufferEndHandler}
/>

The last step alerts users when a video freezes. The buffer state must be true to render a load effectively.

{buffer && <p>Loading</p>}
Our buffering video.

Observe the Loading text when the video freezes.

Conclusion

This post taught us how to construct and modify a video player using the react-player package and Material UI to style and import the required icons. Of course, you can always improve the build by including a few special features to make the player entirely custom.

Here is the URL to the GitHub repository.

Table of Contents

React

Video

Tutorial

More from Pieces
Subscribe to our newsletter
Join our growing developer community by signing up for our monthly newsletter, The Pieces Post.

We help keep you in flow with product updates, new blog content, power tips and more!
Thank you for joining our community! Stay tuned for the next edition.
Oops! Something went wrong while submitting the form.