Build a WebRTC Video Recorder and Downloads as MP4 Video in Browser Using Javascript Full Project For Beginners

 

 

 

 

 

index.html

 

 

<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>

 

 

 

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);
});

Leave a Reply