player, Video HTML streaming

Cross browser video player

I tried using videojs for playing encrypted HLS video in all common browsers. It didn’t go well. I had a lot of trouble to get it working in Safari browser on MacOS (seeking). The best thing I did was writing my own video player. Here is an example of it.

Example of index.html:

<html>
<head>
   <title></title>
   <meta charset="utf-8" />
   <link rel="stylesheet" type="text/css" href="test.css" />
   <script src="Static/js/hls.js"></script>
   <script src="Static/js/test.js"></script>
</head>
<body>
   <div id ="video-container">
      <video id ="video" autoplay>
         <source src ="hls/master.m3u8" type="application/x-mpegURL" />
         Your browser does not support the HLS video or the video tag.
      </video>
   </div>
</body>
</html>

Example of test.js file:

var INIT_VOLUME = 0.3;
 
function isFullscreen() {
   if (document.fullscreenElement || document.webkitCurrentFullScreenElement ||
      document.webkitFullscreenElement || document.mozFullScreenElement || document.msFullscreenElement) {
      return true;
   }
   return false;
}
function requestFullscreen(el) {
   if (el.requestFullscreen) {
      el.requestFullscreen();
   }
   else if (el.mozRequestFullScreen) {
      el.mozRequestFullScreen();
   }
   else if (el.webkitRequestFullScreen) {
      el.webkitRequestFullScreen();
   }
   else if (el.webkitRequestFullscreen) {
      el.webkitRequestFullscreen();
   }
   else if (el.msRequestFullscreen) {
      el.msRequestFullscreen();
   }
}
function cancelFullscreen() {
   if (document.exitFullscreen) {
      document.exitFullscreen();
   }
   else if (document.mozCancelFullScreen) {
      document.mozCancelFullScreen();
   }
   else if (document.webkitCancelFullScreen) {
      document.webkitCancelFullScreen();
   }
   else if (document.msExitFullscreen) {
      document.msExitFullscreen();
   }
}
function fromSeconds(value) {
   var hours = Math.floor(value / 3600);
   value = value - hours * 3600;
   var minutes = Math.floor(value / 60);
   value = value - minutes * 60;
   var hoursStr = ("0" + hours).slice(-2);
   var minutesStr = ("0" + minutes).slice(-2);
   var secondsStr = ("0" + Math.round(value)).slice(-2);
   if (hours > 0)
      return hoursStr + ":" + minutesStr + ":" + secondsStr;
   return minutesStr + ":" + secondsStr;
}
function createControls() {
   $('#video-container').append('< div id="video-controls">' +
      '<button id="play"><i class="icon icon-play"></i></button>' +
      '<button id="volume-btn"><i class="icon icon-volume"></i></button>' +
      '<span id="current-time">00:00</span>' +
      '< div id="progress">' +
      '< div id="progress-buf"></div>' +
      '< div id="progress-bar"></div>' +
      '</div>' +
      '<span id="total-time"></span>' +
      '<button id="fullscreen"><i class="icon icon-fullscreen"></i></button>' +
      '<button id="levels">auto</button>' +
      '</div>' +
      '< div class="loader"></div>');
   var volumeBar = $('< div id="volume" style="display:none">< div id="volume-bar">< div id="volume-value" style="height: ' + INIT_VOLUME * 100 + '%"></div></div></div>');
   $('#video-controls').append(volumeBar);
   $('#volume-btn').click(function () {
      if (volumeBar.is(":visible")) {
         $(this).removeClass('active');
         volumeBar.hide();
      }
      else {
         $(this).addClass('active');
         volumeBar.show();
      }
   });
}
function createLevelButtons(hls) {
   var levels = hls.levelController._levels;
   var list = $('<ul id="video-levels" style="display:none">');
   for (var i = 0; i < levels.length; i++) {
      list.append('<li level="' + i + '">' + levels[i].height + 'p</li>');
   }
   list.append('<li level="-1" style="display:none">auto</li>');
   $('#video-controls').append(list);
   $("body").on("click""#video-levels li"function () {
      var level = parseInt($(this).attr("level"));
      hls.currentLevel = level; //immediate (nextLevel for later change)
      $('#levels').text($(this).text());
      list.hide();
      $("#video-levels li").show();
      $(this).hide();
      $('#levels').removeClass('active');
      $(".loader").show();
      $('#progress-buf').width(0);
   });
   $('#levels').click(function () {
      if (list.is(":visible")) {
         $(this).removeClass('active');
         list.hide();
      }
      else {
         $(this).addClass('active');
         list.show();
      }
   });
}
 
$(document).ready(function () {
   var video = document.getElementById('video');
   if (Hls.isSupported()) {
      createControls();
      var vid = $("#video");
      var getLoadedPercentage = function () {
         try {
            var range = 0;
            var bf = video.buffered;
            var time = video.currentTime;
            while (!(bf.start(range) <= time && time <= bf.end(range))) {
               range += 1;
            }
            return bf.end(range) / video.duration;
         }
         catch (e) {
            return 0;
         }
      };
      $(window).resize(function () {
         $('#video-controls').width(vid.width());
      });
      vid.bind('webkitfullscreenchange mozfullscreenchange msfullscreenchange fullscreenchange resize orientationchange'function () {
         $('#video-controls').width(vid.width());
      });
      var hls = new Hls();
      hls.loadSource('hls/master.m3u8');
      hls.attachMedia(video);
      hls.on(Hls.Events.MANIFEST_PARSED, function () {
         createLevelButtons(hls);
         video.volume = INIT_VOLUME;
         video.play();
      });
      hls.on(Hls.Events.BUFFER_APPENDED, function () {
         var loadEndPercentage = getLoadedPercentage();
         $('#progress-buf').width((loadEndPercentage * 100+ "%");
      });
      $('#play').click(function () {
         if (video.paused)
            video.play();
         else
            video.pause();
      });
      $('#volume-bar').click(function (e) {
         var y = e.pageY - this.offsetTop - $(this).parent()[0].offsetTop - $(this).parent().parent()[0].offsetTop;
         var position = 1 - (y / $('#volume-bar').height());
         video.volume = position;
         $('#volume-value').css('height', (position * 100+ '%');
      });
      $('#progress').click(function (e) {
         var x = e.pageX - this.offsetLeft - $(this).parent()[0].offsetLeft;
         var position = x / $('#progress').width();
         video.currentTime = position * video.duration;
         $('#progress-bar').css('width', (position * 100+ '%');
      });
      $('#fullscreen').click(function () {
         if (isFullscreen()) {
            cancelFullscreen();
            $('body').removeClass("fullscreen");
            $(this).removeClass('active');
         }
         else {
            var videoContainerEl = videoContainer[0];
            requestFullscreen(videoContainerEl);
            $('body').addClass("fullscreen");
            $(this).addClass('active');
         }
      });
      vid.on('timeupdate'function () {
         var percentage = video.currentTime / video.duration;
         $('#progress-bar').css('width', (100 * percentage) + '%');
         $('#current-time').text(fromSeconds(video.currentTime));
      });
      vid.on('loadedmetadata'function () {
         $('#total-time').text(fromSeconds(video.duration));
         $('#video-controls').width(vid.width());
      });
      vid.on('playing'function () {
         $('#play').find('i').removeClass('icon-play').addClass('icon-pause');
         $(".loader").hide();
      });
      vid.on('seeking'function () {
         $(".loader").show();
      });
      vid.on('seeked'function () {
         $(".loader").hide();
      });
      vid.on('pause'function () {
         $('#play').find('i').removeClass('icon-pause').addClass('icon-play');
      });
      vid.on('loadstart'function (event) {
         $(".loader").show();
      });
      vid.on('canplay'function (event) {
         $(".loader").hide();
      });
      vid.on('waiting'function (event) {
         $(".loader").show();
      });
   }
   else {
      video.setAttribute("controls""controls");
   }
});
 

Example of test.css file:

.icon-fullscreen {
      background-imageurl('');
   }
 
   .icon-play {
      background-imageurl('');
   }
 
   .icon-pause {
      background-imageurl('');
   }
 
   .icon-volume {
      background-imageurl('');
   }
 
   .icon {
      background-size30px;
      width30px;
      height30px;
   }
 
   #video-wrapper {
      positionabsolute;
      top0;
      left0;
      right0;
      bottom0;
      z-index2147483646;
      max-width100%;
      marginauto;
   }
 
   #video {
      height50%;
      positionabsolute;
      top0;
      left0;
      right0;
      bottom0;
      z-index2147483645;
      max-width100%;
      marginauto;
      background-colorblack;
   }
 
   #video-controls {
      positionabsolute;
      top75%;
      left0;
      right0;
      bottom0;
      z-index2147483647;
      max-width100%;
      marginauto;
      height40px;
      margin-top0;
      background-color#1d35e4;
      colorwhite;
   }
 
   body {
      text-aligncenter;
   }
 
      html,
      body,
      body.fullcreen > main {
         height100%;
      }
 
   #video-container {
      heightcalc(100% - 4px);
      margin0;
      padding0;
   }
 
   main {
      heightcalc(100% - 64px);
   }
 
   #progress {
      height10px;
      widthcalc(100% - 301px);
      background-colorwhite;
      displayinline-block;
      margin-left5px;
      cursorpointer;
   }
 
      #progress > #progress-bar {
         height100%;
         background-color#366ac7;
         width0;
         margin-top-10px;
      }
 
      #progress > #progress-buf {
         height100%;
         background-color#d8d4d4;
         width0;
      }
 
   #video-controls > span {
      width60px;
      displayinline-block;
      height40px;
      padding-top10px;
   }
 
   #video-controls > span > p{
      margin:0;
   }
 
   #video-controls > span#total-time {
      margin-left5px;
   }
 
   #video-controls > button {
      width40px;
      height40px;
      bordernone;
      padding0;
      background-colortransparent;
      floatleft;
   }
 
      #video-controls > button.active {
         background-color#19abd0;
      }
 
      #video-controls > button:hover {
         background-color#19abd0;
      }
 
      #video-controls > button#levels {
         floatright;
         width50px;
      }
 
      #video-controls > button#fullscreen {
         floatright;
      }
 
   body.fullcreen > header,
   body.fullcreen > main > .progress {
      displaynone;
   }
 
   body.fullscreen #video,
   body.fullscreen #video-wrapper {
      heightcalc(100% - 40px);
      bottom40px;
   }
 
   body.fullscreen #video-controls {
      width100% !important;
      topcalc(100% - 40px);
   }
 
   #video-levels {
      positionabsolute;
      right40px;
      bottom25px;
      background-color#19abd0;
      padding5px;
      width50px;
      z-index2147483647;
   }
 
      #video-levels > li {
         cursorpointer;
      }
 
         #video-levels > li:hover {
            background-color#1d35e4;
         }
 
   #volume {
      positionabsolute;
      width40px;
      height100px;
      background-color#19abd0;
      bottom40px;
      left41px;
   }
 
   #volume-bar {
      background-colorwhite;
      widthcalc(100% - 20px);
      heightcalc(100% - 20px);
      margin10px;
      positionrelative;
      cursorpointer;
   }
 
   #volume-value {
      background-color#366ac7;
      bottom0;
      positionabsolute;
      width100%;
   }
 
   .loader {
      border16px solid #f3f3f3/* Light grey */
      border-top16px solid #1d35e4/* Blue */
      border-radius50%;
      width120px;
      height120px;
      animationspin 2s linear infinite;
      positionabsolute;
      topcalc(50% - 60px);
      leftcalc(50% - 60px);
      z-index2147483647;
   }
 
   @keyframes spin {
      0% {
         transformrotate(0deg);
      }
 
      100% {
         transformrotate(360deg);
      }
   }

You can use it. There can be minor problems, because I parsed this code from my working solution and I don’t want to share everything. Please replace “< div” with “<div”. The wordpress formatting is shit.