JavaScript WebRTC Video Recorder and Download as MP4 Video

In this tutorial, I will teach you how to make a Javascript WebRTC Video Recorder. You can also play/download the recorded MP4 video file.

The complete source code of our Javascript WebRTC Video Recorder is given below.

Javascript WebRTC Video Recorder Source Code

index.html

<!DOCTYPE html>
<html>
    <head>
        <title>JavaScript WebRTC Video Recorder</title>
        <link type="text/css" rel="stylesheet" href="./style.css" />
    </head>
    <body>
        <div id="container">
            <h1>Video Recorder</h1>
            <p>This page allows you to record video from your camera and save it in mp4/h.264 format. At the moment it is intended to work
              for Chrome, Firefox and Microsoft Edge. Note: there is no limit for the amount of time you record. Everything is leveraged
              by the web browser. No plugins are required.</p>
            <p>Note: this video recorder works properly with the latest versions of Chrome, Firefox and Microsoft Edge. Internet Explorer
              11 does not support WebRTC, hence this demo will not work there.
              <a href="https://html5test.com/compare/browser/ie-11/chrome-64/firefox/edge.html"
                target="_blank">https://html5test.com/compare/browser/ie-11/chrome-64/firefox/edge.html</a>
            </p>
            <table>
              <tr>
                <td>
                  <label for="videoSource">Video Source
                  <select name="" id="videoSource"></select>
                </label></td>
                <td style="display:none;">
                  <label for="audioSource">Audio Source
                    <select name="" id="audioSource"></select>
                  </label>
                </td>
                <td>
              </tr>
              
             
            </table>
            <div id="videoMain">
              <video muted autoplay id="videoContainer"></video>
            </div>
            <div id="videoPreview" style="display:none">
              <input type="hidden" name="videoBitrate" id="videoBitrate" value="4000000">
              <input type="hidden" name="audioBitrate" id="audioBitrate" value="320000">
              <video controls id="recordedVideo"></video>
              <button id="play" disabled>Play</button>
            </div>
            <div class="log"></div>
        
            <div>
              <button id="record" disabled>Start Recording</button>
              <button id="preview" disabled>See Recording</button>
              <button id="download" disabled>Download</button>
            </div>
        </div>

        <script src="https://code.jquery.com/jquery-3.6.1.min.js"></script>
        <script src="./script.js"></script>
    </body>
</html>

style.css

.hidden {
    display: none;
  }
  
  .highlight {
    background-color: #eee;
    font-size: 1.2em;
    margin: 0 0 30px 0;
    padding: 0.2em 1.5em;
  }
  .warning {
    color: red;
    font-weight: 400;
  }
  
  div#errorMsg p {
    color: #F00;
  }
  
  body {
    font-family: 'Roboto', sans-serif;
    font-weight: 300;
  }
  
  a {
  color: #6fa8dc;
  font-weight: 300;
  text-decoration: none;
  }
  
  a:hover {
  color: #3d85c6;
  text-decoration: underline;
  }
  
  a#viewSource {
  display: block;
  margin: 1.3em 0 0 0;
  border-top: 1px solid #999;
  padding: 1em 0 0 0;
  }
  
  div#links a {
  display: block;
  line-height: 1.3em;
  margin: 0 0 1.5em 0;
  }
  
  div.outputSelector {
  margin: -1.3em 0 2em 0;
  }
  
  @media screen and (min-width: 1000px) {
  /* hack! to detect non-touch devices */
    div#links a {
          line-height: 0.8em;
    }
  }
  
  h1 a {
  font-weight: 300;
  margin: 0 10px 0 0;
  white-space: nowrap;
  }
  
  audio {
  max-width: 100%;
  }
  
  body {
  font-family: 'Roboto', sans-serif;
  margin: 0;
  padding: 1em;
  word-break: break-word;
  }
  
  button {
  background-color: #d84a38;
  border: none;
  border-radius: 2px;
  color: white;
  font-family: 'Roboto', sans-serif;
  font-size: 0.8em;
  margin: 0 0 1em 0;
  padding: 0.5em 0.7em 0.6em 0.7em;
  }
  
  button:active {
  background-color: #cf402f;
  }
  
  button:hover {
  background-color: #cf402f;
  }
  
  button[disabled] {
  color: #ccc;
  }
  
  button[disabled]:hover {
  background-color: #d84a38;
  }
  
  canvas {
    background-color: #ccc;
    max-width: 100%;
    width: 100%;
  }
  
  code {
  font-family: 'Roboto', sans-serif;
  font-weight: 400;
  }
  
  div#container {
  margin: 0 auto 0 auto;
  max-width: 40em;
  padding: 1em 1.5em 1.3em 1.5em;
  }
  
  div#links {
      padding: 0.5em 0 0 0;
  }
  
  h1 {
  border-bottom: 1px solid #ccc;
  font-family: 'Roboto', sans-serif;
  font-weight: 500;
  margin: 0 0 0.8em 0;
  padding: 0 0 0.2em 0;
  }
  
  h2 {
  color: #444;
  font-size: 1em;
  font-weight: 500;
  line-height: 1.2em;
  margin: 0 0 0.8em 0;
  }
  
  h3 {
  border-top: 1px solid #eee;
  color: #666;
  font-weight: 500;
  margin: 20px 0 10px 0;
  padding: 10px 0 0 0;
  white-space: nowrap;
  }
  
  html {
  /* avoid annoying page width change
  when moving from the home page */
  overflow-y: scroll;
  }
  
  img {
  border: none;
  max-width: 100%;
  }
  
  input[type=radio] {
  position: relative;
  top: -1px;
  }
  
  p {
  color: #444;
  font-weight: 300;
  line-height: 1.6em;
  }
  
  p#data {
  border-top: 1px dotted #666;
  font-family: Courier New, monospace;
  line-height: 1.3em;
  max-height: 1000px;
  overflow-y: auto;
  padding: 1em 0 0 0;
  }
  
  p.borderBelow {
  border-bottom: 1px solid #aaa;
  padding: 0 0 20px 0;
  }
  
  section p:last-of-type {
  margin: 0;
  }
  
  section {
    border-bottom: 1px solid #eee;
    margin: 0 0 30px 0;
    padding: 0 0 20px 0;
  }
  
  section:last-of-type {
    border-bottom: none;
    padding: 0 0 1em 0;
  }
  
  select {
    margin: 0 1em 1em 0;
    position: relative;
    top: -1px;
  }
  
  h1 span {
    white-space: nowrap;
  }
  
  strong {
    font-weight: 500;
  }
  
  textarea {
    font-family: 'Roboto', sans-serif;
  }
  
  video {
    background: #222;
    margin: 0 0 20px 0;
    width: 100%;
  }
  
  @media screen and (max-width: 650px) {
    .highlight {
      font-size: 1em;
      margin: 0 0 20px 0;
      padding: 0.2em 1em;
    }
    h1 {
      font-size: 24px;
    }
  }
  
  @media screen and (max-width: 550px) {
    button:active {
      background-color: darkRed;
    }
    h1 {
      font-size: 22px;
    }
  }
  
  @media screen and (max-width: 450px) {
    h1 {
      font-size: 20px;
    }
  }
  
  
   button {
    margin: 0 3px 10px 0;
    padding-left: 2px;
    padding-right: 2px;
    width: 99px;
  }
  
  button:last-of-type {
    margin: 0;
  }
  
  p.borderBelow {
    margin: 0 0 20px 0;
    padding: 0 0 20px 0;
  }
  
  video {
    height: auto;
    margin: 0 12px 20px 0;
    vertical-align: top;
    min-width: 100%;
  }
  
  
  video:last-of-type {
    margin: 0 0 20px 0;
  }
  
  video#videoRecorder {
    margin: 0 20px 20px 0;
  }
  
  @media screen and (max-width: 500px) {
    button {
      font-size: 0.8em;
      width: calc(33% - 5px);
    }
  }
  
  @media screen and (max-width: 720px) {
    video {
      height: calc((50vw - 48px) * 3 / 4);
      margin: 0 10px 10px 0;
      width: calc(50vw - 48px);
    }
  
    video#videoRecorder {
      margin: 0 10px 10px 0;
    }
  }
  
  #log, .log {
    font-size: 10px;
    margin-bottom: 20px;
  }

script.js

/**
 * [videoRecorder is the class for video recording]
 * @type {Class}
 */
 var videoRecorder = class videoRecorder {
    constructor(selectVideoSources, selectAudioSources, selectVideoBitrate, selectAudioBitrate, DOMVideoObject, recordButton, downloadButton, recordedVideo, previewButton, videoMain, videoPreview, countdown = 30) {
      /**
       * [videoSources holds a list of found video sources]
       * @type {Array}
       */
      this.videoSources = [];
      /**
       * [audioSources holds a list of found audio sources]
       * @type {Array}
       */
      this.audioSources = [];
      /**
       * [logging allows to see logging information through the console.]
       * @type {Boolean}
       */
      this.logging = false;
      /**
       * [countdownTimer takes the parameter countdown which is numeric. it is the total allowed time of recording]
       * @type {Parameter|Numeric}
       */
      this.countdownTimer = countdown;
      /**
       * [selectVideoSources takes the parameter selectVideoSources which is a string. it is a querySelector for the video source selection source: '#myselect']
       * @type {DOM}
       */
      this.selectVideoSources = document.querySelector(selectVideoSources);
      /**
       * [selectAudioSources takes the parameter selectAudioSources which is a string. it is a querySelector for the video source selection source: '#myselect']
       * @type {DOM}
       */
      this.selectAudioSources = document.querySelector(selectAudioSources);
      /**
       * [selectVideoBitrate takes the parameter selectVideoBitrate which is a string. it is a querySelector for the audio source selection input: '#myselect']
       * @type {DOM}
       */
      this.selectVideoBitrate = document.querySelector(selectVideoBitrate);
      /**
       * [selectAudioBitrate takes the parameter countdown which is a string. it is a querySelector for the video bitrate selection input: '#myselect']
       * @type {DOM}
       */
      this.selectAudioBitrate = document.querySelector(selectAudioBitrate);
      /**
       * [previewButton takes the parameter previewButton which is a string. it is a querySelector for the button that launches the preview: '#myselect']
       * @type {DOM}
       */
      this.previewButton = document.querySelector(previewButton);
      /**
       * [DOMVideoObject takes the parameterDOMVideoObject which is a string. it is a querySelector for the main video output: '#myselect']
       * @type {DOM}
       */
      this.DOMVideoObject = document.querySelector(DOMVideoObject);
      /**
       * [videoPreview takes the parameter videoPreview which is a string. it is a querySelector for the container for the video that was recorded]
       * @type {DOM}
       */
      this.videoPreview = document.querySelector(videoPreview);
      /**
       * [videoMain takes the parameter videoMain which is a string. it is a querySelector for the container for the main video]
       * @type {DOM}
       */
      this.videoMain = document.querySelector(videoMain);
      /**
       * [portVideoPreviewWidth is the default width for the recorded and preview video]
       * @type {Number}
       */
      this.portVideoPreviewWidth = 640;
      /**
       * [portVideoPreviewHeight is the default height for the recorded and preview video]
       * @type {Number}
       */
      this.portVideoPreviewHeight = 360;
      /**
       * [recorderOptions are the default encoding settings for the browser to use]
       * @type {Object}
       */
      this.recorderOptions = {
        audioBitsPerSecond: 128000,
        videoBitsPerSecond: 4000000,
        mimeType: 'video/mp4'
      };
      /**
       * [recordedSomething tell to us if we recorded something]
       * @type {Boolean}
       */
      this.recordedSomething = false;
      /**
       * [stream is an object that contains the stream passed from the device. it shares common data with window.stream]
       * @type {Object}
       */
      this.stream = {};
      /**
       * [mediaRecorder will be the MediaRecorded object to be used to record blob data]
       * @type {[type]}
       */
      this.mediaRecorder = null;
      /**
       * [recordedBlobs are raw data recorded in an array by the browser.]
       * @type {Array}
       */
      this.recordedBlobs = [];
      /**
       * [recordButton takes the parameter recordButton which is a string. it is a querySelector for the container for the record button]
       * @type {String}
       */
      this.recordButton = document.querySelector(recordButton);
      /**
       * [previewButtonLabel is the default string for the button that will play the recording]
       * @type {String}
       */
      this.previewButtonLabel =  'Play Recording';
      /**
       * [playingPreview is the default string for the button that will playback the recording]
       * @type {String}
       */
      this.playingPreview =      'Playback Rec';
      /**
       * [recordButtonLabel is the default string for the button that will start the recording]
       * @type {String}
       */
      this.recordButtonLabel =   'Start Recording';
      /**
       * [downloadButtonLabel is the default string for the button that will download the recording]
       * @type {String}
       */
      this.downloadButtonLabel = 'Download Video';
      /**
       * [downloadButton takes the parameter downloadButton which is a string. it is a querySelector for the download button]
       * @type {DOM}
       */
      this.downloadButton = document.querySelector(downloadButton);
      /**
       * [recordedVideo takes the parameter recordedVideo which is a string. it is a querySelector for the playback button for the recorded video]
       * @type {DOM}
       */
      this.recordedVideo = document.querySelector(recordedVideo);
      /**
       * [log is a queryselector for the log division (we will write logging data on it)]
       * @type {DOM}
       */
      this.log = document.querySelector('.log');
      /**
       * [requiredResolutions is an array of allowed resolutions to be used by the recorder.]
       * @type {Array}
       */
      this.requiredResolutions = [
        {
          'label': '720p(HD)',
          'width': 1280,
          'height': 720,
          'ratio': '16:9'
        }, {
          'label': '360p(nHD)',
          'width': 640,
          'height': 360,
          'ratio': '16:9'
        }, {
          'label': '480p',
          'width': 640,
          'height': 480,
          'ratio': '4:3'
        }
      ];
      /**
       * [blockedposter is a poster that displays that the access was denied to access devices]
       * @type {String}
       */
      this.blockedposter = '';
      /**
       * [poster is a poster that displays that a device is busy]
       * @type {String}
       */
      this.poster = '';
      /**
       * [nodeviceselected shows a poster that says that no device was selected]
       * @type {String}
       */
      this.nodeviceselected = '';
      /**
       * [selectdevice shows a poster requesting the user to select a device]
       * @type {String}
       */
      this.selectdevice = '';
      /**
       * [nullDevice is an hypotetical and imaginary device hanging on the clouds of many unknown players, also used to tell the browser to use something that does not exists
       * to allow us to detect video devices more efficiently]
       * @type {String}
       */
      this.nullDevice = '@playerme/__NULL_DEVICE__';
      this.previewButton.textContent = this.previewButtonLabel;
      this.recordButton.textContent = this.recordButtonLabel;
      this.downloadButton.textContent = this.downloadButtonLabel;
      
    }
    /**
     * [hasGetUserMedia return to us if the current browser uses GetUserMedia instance]
     * @return {Boolean} [true: browser uses GetUserMedia, false: F you]
     */
    hasGetUserMedia() {
      return !!(navigator.getUserMedia || navigator.webkitGetUserMedia || navigator.mozGetUserMedia || navigator.msGetUserMedia);
    }
    
    /**
     * [init initializes the full script]
     * @return {Void}
     */
    init() {
      const self = this;
      self.DOMVideoObject.srcObject = null;
      self.DOMVideoObject.poster = self.selectdevice;
      let foundNullDom = false;
      if (!self.hasGetUserMedia()) {
        alert('Incompatible browser to use video capturing features');
        return false;
      }
      if (self.selectVideoSources === null) {
        console.error('Needed a select/input to append video sources. not configured a required one');
        foundNullDom = true;
      }
      if (self.selectAudioSources === null) {
        console.error('Needed a select/input to append audio sources. not configured a required one');
        foundNullDom = true;
      }
      if (self.selectVideoBitrate === null) {
        console.error('Needed a select/input to append video bitrate. not configured a required one');
        foundNullDom = true;
      }
      if (self.selectAudioBitrate === null) {
        console.error('Needed a select/input to append audio bitrate. not configured a required one');
        foundNullDom = true;
      }
      if (self.DOMVideoObject === null) {
        console.error('Needed a DOM for Video. not configured a required one');
        foundNullDom = true;
      }
      if (foundNullDom) {
        throw ('Please check the messages above regarding the need of a dom missing object');
      }
      self.selectAudioSources.onchange = function () { self.getStream(self); };
      self.selectVideoSources.onchange = function () { self.getStream(self); };
      self.recordButton.onclick = function () { self.toggleRecording(self); };
      self.downloadButton.onclick = function () { self.download(self); };
      self.previewButton.onclick = function () { self.togglePreview(self);};
      self.recordedVideo.addEventListener('error', function (ev) {
        console.error('MediaRecording.recordedMedia.error()');
        alert(`Your browser can not play ${self.recordedVideo.src} media clip. 
        event: ${JSON.stringify(ev)}`);
      }, true);
      //first let's fetch the available video and mic, so we can load them on the list of the devices by name
      /**
       * [startup is a promise object that allows the access to devices]
       * @return {Promise} [The startup function]
       */
      var startup = function() {
        return navigator.mediaDevices.enumerateDevices().then(x => {
          return navigator.mediaDevices.getUserMedia({ audio: true, video: self.nullDevice });
        });
      };
      /**
       * [videofail is a failback of startup. if the default device is likely busy, this will
       * help us to list all the video and audio devices]
       * @return {Promise]} [a promise that contains the list of devices]
       */
      var videofail = function() {
        return navigator.mediaDevices.enumerateDevices().then(x => {
          return navigator.mediaDevices.getUserMedia({ audio: true });
        });
      };
      /**
       * [getDevices allow us to filter the devices and put them into the selection boxes]
       * @return {Promise} [a promise that contains the list of devices]
       */
      var getDevices = function(){
        let self = vr;
        if (window.stream) {
          window.stream.getTracks().forEach(function (track) {
            track.stop();
          });
        }
        return navigator.mediaDevices.enumerateDevices().then(x => {
          if(self.logging) console.log('listDevices',x);
          let videodevices = [];
          let audiodevices = [];
          let option;
  
          videodevices = x.filter(function(e){
            return e.kind === 'videoinput';
          });
          audiodevices = x.filter(function (e) {
            return e.kind === 'audioinput';
          });
          if(self.logging) console.log([videodevices,audiodevices]);
          for (let i = 0; i !== videodevices.length; ++i) {
            if ($(self.selectVideoSources).find(`option[value='${videodevices[i].deviceId}']`).length > 0){
              $(self.selectVideoSources)
                .find(`option[value='${videodevices[i].deviceId}']`)
                .text(videodevices[i].label || 'camera ' + (self.selectVideoSources.length + 1));
            } else {
              $(`<option value="${videodevices[i].deviceId}">${videodevices[i].label || 'camera ' + (self.selectVideoSources.length + 1)}</option>`).appendTo($(self.selectVideoSources));
            }
          }
          for (let i = 0; i !== audiodevices.length; ++i) {
            if ($(self.selectAudioSources).find(`option[value='${audiodevices[i].deviceId}']`).length > 0) {
              $(self.selectAudioSources)
                .find(`option[value='${audiodevices[i].deviceId}']`)
                .text(audiodevices[i].label || 'camera ' + (self.selectAudioSources.length + 1));
            } else {
              $(`<option value="${audiodevices[i].deviceId}">${audiodevices[i].label || 'camera ' + (self.selectAudioSources.length + 1)}</option>`).appendTo($(self.selectAudioSources));
            }
          }
          // now we set a default empty value for the selects
          if ($(self.selectVideoSources).find(`option[value='${self.nullDevice}']`).length === 0) {
            $(`<option value="${self.nullDevice}">None</option>`)
              .appendTo($(self.selectVideoSources));
            $('<option value="null" selected="selected">Select a camera</option>')
              .prependTo($(self.selectVideoSources));
          }
        });
      };
      startup()
        .then( x => {
          if(self.logging) console.log(x);
          if (x instanceof MediaStream){
            x.stop();
          }
          getDevices();
        })
        .catch( y => {
          if(self.logging) console.log(y);
          videofail()
            .then( x => {
              if(self.logging) console.log(x);
              getDevices();
            })
            .catch( z => {
              self.DOMVideoObject.srcObject = null;
              if (z.name === 'NotAllowedError'){
                self.DOMVideoObject.poster = self.blockedposter;
              }
            });
        })
        .then(z => {
          if(self.logging) console.log('video fails, using only audio', z);
          getDevices();
        });
    }
    /**
     * [getStream allow us to get the stream of the selected media and start to display the required media into the main video container.
     * this also tests each resolution and display the available resolution into the main video container. if no video is displayed, a 
     * poster is displayed that the device is not compatible with the required resolutions.]
     * @param  {Object} self [this class]
     */
    getStream(self) {
      if (window.stream) {
        window.stream.getTracks().forEach(function (track) {
          track.stop();
        });
      }
      if (self.selectVideoSources.value === self.nullDevice){
        self.DOMVideoObject.srcObject = null;
        self.DOMVideoObject.poster = self.nodeviceselected;
        self.recordButton.disabled = true;
        return;
      } else if (self.selectVideoSources.value === 'null') {
        self.DOMVideoObject.srcObject = null;
        self.DOMVideoObject.poster = self.selectdevice;
        self.recordButton.disabled = true;
        return;
      }
      // testing modes....
      let currentResolution = 0;
      function testAndRun(self, currentResolution) {
        const cameraname = self.selectVideoSources.options[self.selectVideoSources.options.selectedIndex].text;
        try {
          var constrains = {
            audio: {
              deviceId: { exact: self.selectAudioSources.value }
            },
            video: {
              width: { exact: self.requiredResolutions[currentResolution].width },
              height: { exact: self.requiredResolutions[currentResolution].height },
              framerate: { ideal: 30, max: 60 },
              deviceId: { exact: self.selectVideoSources.value }
            }
          };
          if(self.logging) console.log(`${cameraname}: Resolution set to ${self.requiredResolutions[currentResolution].width} x ${self.requiredResolutions[currentResolution].height}`);
          navigator.mediaDevices.getUserMedia(constrains)
            .then(function (stream) {
              /** this conditional makes sure if microsofot edge have videotracks but not return video. This  means there is another process using the required video device... */
              var warn = `${cameraname}: the camera is being used by another application, or the device is not ready.`;
              var working = `${cameraname}: Resolution ${self.requiredResolutions[currentResolution].width} x ${self.requiredResolutions[currentResolution].height} working, using this resolution`;
              if (stream.getVideoTracks().length === 0) {
                self.DOMVideoObject.srcObject = null;
                console.warn(warn);
                self.recordButton.disabled = true;
                self.log.innerHTML = warn;
                if (window.stream) {
                  window.stream.getTracks().forEach(function (track) {
                    track.stop();
                  });
                }
                self.DOMVideoObject.poster = self.poster;
              } else {
                if(self.logging) console.log(working);
                self.log.innerHTML = working;
                self.recordButton.disabled = false;
                self.gotStream(stream);
              }
            })
            .catch(e => {
              var warn = `${cameraname}: the camera is being used by another application, or the device is not ready.`;
              if(self.logging) console.log('error found:', e);
              if (e.name === 'NotReadableError') {
                self.DOMVideoObject.srcObject = null;
                console.warn(warn);
                self.log.innerHTML = warn;
                if (window.stream) {
                  window.stream.getTracks().forEach(function (track) {
                    track.stop();
                  });
                }
                self.recordButton.disabled = true;
                self.DOMVideoObject.poster = self.poster;
              } else {
                var w = `${cameraname}: Resolution ${self.requiredResolutions[currentResolution].width} x ${self.requiredResolutions[currentResolution].height} for selected camera does not work, switching for more suitable resolution...`;
                console.warn(w);
                self.log.innerHTML = w;
                setTimeout(function () {
                  currentResolution++;
                  testAndRun(self, currentResolution);
                }, 500);
              }
            });
        } catch (e) {
          var warn;
          try{
            warn = `${cameraname}: Resolution ${self.requiredResolutions[currentResolution].width} x ${self.requiredResolutions[currentResolution].height} for selected camera does not work, switching for more suitable resolution...`;
            self.DOMVideoObject.srcObject = null;
            if(self.logging) console.log('error found:', e);
            console.warn(warn);
            self.log.innerHTML = warn;
            setTimeout(function () {
              currentResolution++;
              testAndRun(self, currentResolution);
            }, 500);
          } catch (e) {
            if (window.stream) {
              window.stream.getTracks().forEach(function (track) {
                track.stop();
              });
            }
            warn =`${cameraname}: This camera cannot display a proper resolution. It could be the device is being used by another application or the camera does not support the required resolutions. Please choose another device.`;
            self.DOMVideoObject.srcObject = null;
            self.log.innerHTML = warn;
            self.recordButton.disabled = true;
            self.DOMVideoObject.poster = self.poster;
          }
        }
      }
      testAndRun(self, currentResolution);
    }
    /**
     * [gotStream allow us to get the stream from the selected main video device output on the page]
     * @param  {Object} stream [the data stream from the selected device]
     */
    gotStream(stream) {
      if (window.stream) {
        window.stream.getTracks().forEach(function (track) {
          track.stop();
        });
      }
      var self = this;
      self.stream = stream;
      window.stream = stream;
      self.DOMVideoObject.srcObject = stream;
    }
    /**
     * [handleError is a basic error handler]
     * @param  {Object} err [the Error Object]
     * @throws {String}     [an string with the error details.]
     */
    handleError(err) {
      if(self.logging) console.log(err);
      console.error(err.name + ': ' + err.message);
      throw (err.name + ': ' + err.message);
    }
    /**
     * [toggleRecording toggles between recording mode and non recording mode]
     * @param  {Object} self [this class]
     */
    toggleRecording(self) {
      $(self.videoMain).show();
      $(self.videoPreview).hide();
      if (self.recordButton.textContent === 'Start Recording') {
        self.startRecording(self);
        self.recordedSomething = true;
        self.previewButton.disabled = true;
      } else {
        self.stopRecording(self);
        if(self.recordedSomething){
          self.previewButton.disabled = false;
          self.recordedSomething = false;
        }
        self.recordButton.textContent = 'Start Recording';
        self.downloadButton.disabled = false;
      }
    }
    /**
     * [togglePreview toggles between displaying the recording video and the view to get ready a new recording.]
     * @param  {Object} self [this class]
     */
    togglePreview(self){
      $(self.videoMain).hide();
      $(self.videoPreview).show();
      if($(self.videoPreview).css('display') === 'block'){
        self.play(self);
      }
    }
    /**
     * [startRecording starts to record the media through the provided options]
     * @param  {Object} self [this class]
     */
    startRecording(self) {
      $(self.recordedVideo).css({
        'width': self.portVideoPreviewWidth,
        'height': self.portVideoPreviewHeight
      });
      self.recordedBlobs = [];
      self.recorderOptions.mimeType = 'video/webm;codecs=h264';
      if (!MediaRecorder.isTypeSupported(self.recorderOptions.mimeType)) {
        if(self.logging) console.log(self.recorderOptions.mimeType + ' is not Supported');
        self.recorderOptions.mimeType = 'video/webm;codecs=vp9';
        if (!MediaRecorder.isTypeSupported(self.recorderOptions.mimeType)) {
          if(self.logging) console.log(self.recorderOptions.mimeType + ' is not Supported');
          self.recorderOptions.mimeType = 'video/webm;codecs=vp8';
          if (!MediaRecorder.isTypeSupported(self.recorderOptions.mimeType)) {
            if(self.logging) console.log(self.recorderOptions.mimeType + ' is not Supported');
            self.recorderOptions.mimeType = 'video/webm';
            if (!MediaRecorder.isTypeSupported(self.recorderOptions.mimeType)) {
              self.recorderOptions.mimeType = '';
              console.warn('Not able to find a suitable container. mimeType will be empty');
            }
          }
        }
      }
      try {
        self.recorderOptions.audioBitsPerSecond = parseInt(self.selectAudioBitrate.value, 10);
        self.recorderOptions.videoBitsPerSecond = parseInt(self.selectVideoBitrate.value, 10);
        self.mediaRecorder = new MediaRecorder(self.stream, self.recorderOptions);
        if(self.logging) console.log('Created MediaRecorder', self.mediaRecorder, 'with options', self.recorderOptions);
        self.recordButton.textContent = 'Stop Recording';
        self.mediaRecorder.onstop = self.handleStop;
        self.mediaRecorder.ondataavailable = function (e) {
          self.handleDataAvailable(e, self);
        };
        self.mediaRecorder.start(10);
        self.downloadButton.disabled = true;
        self.timer(self);
        if(self.logging) console.log('MediaRecorder started', self.mediaRecorder);
      } catch (e) {
        console.error('Exception while creating MediaRecorder: ' + e);
        alert(`Exception while creating MediaRecorder: ${e}.mimeType: ${self.recorderOptions.mimeType}`);
        return;
      }
    }
    /**
     * [handleDataAvailable pushes blob data to the recorded blobs array]
     * @param  {event} event [event related to the captured blob data]
     * @param  {self}  self  [this class]
     */
    handleDataAvailable(event, self) {
      if (event.data && event.data.size > 0) {
        self.recordedBlobs.push(event.data);
      }
    }
    /**
     * [stopRecording stops the recording]
     * @param  {Object} self [this class]
     */
    stopRecording(self) {
      self.mediaRecorder.stop();
      if(self.logging) console.log('Recorded Blobs:', self.recordedBlobs);
      self.recordedVideo.controls = false;
    }
    /**
     * [play displays the recorded blob into the video player]
     * @param  {Object} self [this class]
     */
    play(self) {
      $(self.recordedVideo).css({
        'width': self.portVideoPreviewWidth,
        'height': self.portVideoPreviewHeight
      });
      var superBuffer = new Blob(self.recordedBlobs, { type: 'video/webm' });
      self.recordedVideo.src = window.URL.createObjectURL(superBuffer);
      self.recordedVideo.onload = function(){
        self.recordedVideo.currentTime = 0;
        self.recordedVideo.play();
      };
      // workaround for non-seekable video taken from
      // https://bugs.chromium.org/p/chromium/issues/detail?id=642012#c23
      self.recordedVideo.addEventListener('loadedmetadata', function () {
        self.recordedVideo.currentTime = 0;
        self.recordedVideo.controls = false;
        self.recordedVideo.play();
        self.previewButton.textContent = self.playingPreview;
        self.previewButton.disabled = true;
        self.downloadButton.disabled = true;
        self.recordButton.disabled = true;
      });
      self.recordedVideo.addEventListener('error',function(e){
        if(self.logging) console.log('error', e);
      });
      self.recordedVideo.addEventListener('timeupdate',function(e){
        self.previewButton.textContent = Math.floor(this.currentTime);
        $.event.trigger({
          type: 'timerOnPlayback',
          message: Math.floor(this.currentTime),
          time: new Date()
        });
      });
      self.recordedVideo.addEventListener('progress', function(e){
        if(self.logging) console.log('progress',e);
      });
      self.recordedVideo.addEventListener('ended', function (e) {
        self.previewButton.textContent = self.previewButtonLabel;
        self.previewButton.disabled = false;
        self.downloadButton.disabled = false;
        self.recordButton.disabled = false;
      });
    }
    /**
     * [handleStop is an event message that tell us that the recorder stopped]
     * @param  {Object} event [the event that comes after an stop video]
     */
    handleStop(event) {
      if(self.logging) console.log('Recorder stopped: ', event);
    }
    /**
     * [download processes the blob and converts everything into a proper media to be 'downloaded']
     * @param  {Object} self [this class]
     */
    download(self) {
      var blob = new Blob(self.recordedBlobs, { type: 'video/webm' });
      var url = window.URL.createObjectURL(blob);
      var a = document.createElement('a');
      a.style.display = 'none';
      a.href = url;
      a.download = 'test.mp4';
      document.body.appendChild(a);
      a.click();
      setTimeout(function () {
        document.body.removeChild(a);
        window.URL.revokeObjectURL(url);
      }, 100);
    }
    /**
     * [timer will process a timer that will count down the allowed time of recording]
     * @param  {Object} self [this class]
     */
    timer(self){
      var count = self.countdownTimer;
      var counter = setInterval(timer,1000);
      self.recordButton.textContent = count;
      self.recordButton.disabled = true;
      
      function timer() {
        count = count - 1;
        $.event.trigger({
          type: 'timerOnRecording',
          message: count,
          time : new Date()
        });
        self.recordButton.textContent = count;
        if(count <= 0){
          clearInterval(counter);
          self.recordButton.textContent = self.recordButtonLabel;
          self.stopRecording(self);
          self.previewButton.disabled = false;
          self.downloadButton.disabled = true;
          self.recordButton.disabled = false;
          $.event.trigger({
            type: 'timerOnRecording',
            message: false,
            time: new Date()
          });
          return;
        }
      }
    }
  };
  
  let vr;
  
  $(function(){
    vr = new videoRecorder('#videoSource', '#audioSource', '#videoBitrate', '#audioBitrate','#videoContainer','#record','#download','#recordedVideo','#preview','#videoMain','#videoPreview',15);
    vr.init();
  });
  
  /**
   * Useful events to be used: timerOnPlayback will display the current playback time of the 
   * recorded video in seconds
   */
  $(window).on('timerOnPlayback',function(e){
    console.log('timerOnPlayback',e.message);
  });
  /**
   * timerOnRecording will display the current time of recording in seconds
   */
  $(window).on('timerOnRecording', function (e) {
    console.log('timerOnRecording', e.message);
  });

Run the Project

Open the index.html file using a web server. For local development, you can use software like XAMPP.

Now you need to allow camera and microphone access.

Allow camera and microphone access on Google Chrome

Now select the camera you want to use.

JavaScript WebRTC select camera for recording

Once you select the camera, the camera output will start showing on the screen.

JavaScript WebRTC start recording video

Press the “Start Recording” button to record your video using JavaScript WebRTC.

After recording, you can play or download the MP4 video file using the “Play Recording” and “Download Video” buttons respectively.

JavaScript WebRTC play or download video

Here’s a screenshot of the downloaded MP4 file.

test.mp4 downloaded

Leave a Comment

This site uses Akismet to reduce spam. Learn how your comment data is processed.