Вход Регистрация
Файл: media/player/videojs/plugins/videojs-resolution-switcher-master/lib/videojs-resolution-switcher.js
Строк: 466
<?php
/*! videojs-resolution-switcher - 2015-7-26
 * Copyright (c) 2016 Kasper Moskwiak
 * Modified by Pierre Kraft and Derk-Jan Hartman
 * Licensed under the Apache-2.0 license. */

(function() {
  
/* jshint eqnull: true*/
  /* global require */
  
'use strict';
  var 
videojs null;
  if(
typeof window.videojs === 'undefined' && typeof require === 'function') {
    
videojs = require('video.js');
  } else {
    
videojs window.videojs;
  }

  (function(
windowvideojs) {
    var 
videoJsResolutionSwitcher,
      
defaults = {
        
uitrue
      
};

    
/*
     * Resolution menu item
     */
    
var MenuItem videojs.getComponent('MenuItem');
    var 
ResolutionMenuItem videojs.extend(MenuItem, {
      
constructor: function(playeroptions){
        
options.selectable true;
        
// Sets this.player_, this.options_ and initializes the component
        
MenuItem.call(thisplayeroptions);
        
this.src options.src;

        
player.on('resolutionchange'videojs.bind(thisthis.update));
      }
    } );
    
ResolutionMenuItem.prototype.handleClick = function(event){
      
MenuItem.prototype.handleClick.call(this,event);
      
this.player_.currentResolution(this.options_.label);
    };
    
ResolutionMenuItem.prototype.update = function(){
      var 
selection this.player_.currentResolution();
      
this.selected(this.options_.label === selection.label);
    };
    
MenuItem.registerComponent('ResolutionMenuItem'ResolutionMenuItem);

    
/*
     * Resolution menu button
     */
    
var MenuButton videojs.getComponent('MenuButton');
    var 
ResolutionMenuButton videojs.extend(MenuButton, {
      
constructor: function(playeroptions){
        
this.label document.createElement('span');
        
options.label 'Quality';
        
// Sets this.player_, this.options_ and initializes the component
        
MenuButton.call(thisplayeroptions);
        
this.el().setAttribute('aria-label','Quality');
        
this.controlText('Quality');

        if(
options.dynamicLabel){
          
videojs.addClass(this.label'vjs-resolution-button-label');
          
this.el().appendChild(this.label);
        }else{
          var 
staticLabel document.createElement('span');
          
videojs.addClass(staticLabel'vjs-menu-icon');
          
this.el().appendChild(staticLabel);
        }
        
player.on('updateSources'videojs.bindthisthis.update ) );
      }
    } );
    
ResolutionMenuButton.prototype.createItems = function(){
      var 
menuItems = [];
      var 
labels = (this.sources && this.sources.label) || {};

      
// FIXME order is not guaranteed here.
      
for (var key in labels) {
        if (
labels.hasOwnProperty(key)) {
          
menuItems.push(new ResolutionMenuItem(
            
this.player_,
            {
              
labelkey,
              
srclabels[key],
              
selectedkey === (this.currentSelection this.currentSelection.label false)
            })
          );
        }
      }
      return 
menuItems;
    };
    
ResolutionMenuButton.prototype.update = function(){
      
this.sources this.player_.getGroupedSrc();
      
this.currentSelection this.player_.currentResolution();
      
this.label.innerHTML this.currentSelection this.currentSelection.label '';
      return 
MenuButton.prototype.update.call(this);
    };
    
ResolutionMenuButton.prototype.buildCSSClass = function(){
      return 
MenuButton.prototype.buildCSSClass.callthis ) + ' vjs-resolution-button';
    };
    
MenuButton.registerComponent('ResolutionMenuButton'ResolutionMenuButton);

    
/**
     * Initialize the plugin.
     * @param {object} [options] configuration for the plugin
     */
    
videoJsResolutionSwitcher = function(options) {
      var 
settings videojs.mergeOptions(defaultsoptions),
          
player this,
          
groupedSrc = {},
          
currentSources = {},
          
currentResolutionState = {};

      
/**
       * Updates player sources or returns current source URL
       * @param   {Array}  [src] array of sources [{src: '', type: '', label: '', res: ''}]
       * @returns {Object|String|Array} videojs player object if used as setter or current source URL, object, or array of sources
       */
      
player.updateSrc = function(src){
        
//Return current src if src is not given
        
if(!src){ return player.src(); }

        
// Only add those sources which we can (maybe) play
        
src src.filter( function(source) {
          try {
            return ( 
player.canPlayTypesource.type ) !== '' );
          } catch (
e) {
            
// If a Tech doesn't yet have canPlayType just add it
            
return true;
          }
        });
        
//Sort sources
        
this.currentSources src.sort(compareResolutions);
        
this.groupedSrc bucketSources(this.currentSources);
        
// Pick one by default
        
var chosen chooseSrc(this.groupedSrcthis.currentSources);
        
this.currentResolutionState = {
          
labelchosen.label,
          
sourceschosen.sources
        
};

        
player.trigger('updateSources');
        
player.setSourcesSanitized(chosen.sourceschosen.label);
        
player.trigger('resolutionchange');
        return 
player;
      };

      
/**
       * Returns current resolution or sets one when label is specified
       * @param {String}   [label]         label name
       * @param {Function} [customSourcePicker] custom function to choose source. Takes 2 arguments: sources, label. Must return player object.
       * @returns {Object}   current resolution object {label: '', sources: []} if used as getter or player object if used as setter
       */
      
player.currentResolution = function(labelcustomSourcePicker){
        if(
label == null) { return this.currentResolutionState; }

        
// Lookup sources for label
        
if(!this.groupedSrc || !this.groupedSrc.label || !this.groupedSrc.label[label]){
          return;
        }
        var 
sources this.groupedSrc.label[label];
        
// Remember player state
        
var currentTime player.currentTime();
        var 
isPaused player.paused();

        
// Hide bigPlayButton
        
if(!isPaused && this.player_.options_.bigPlayButton){
          
this.player_.bigPlayButton.hide();
        }

        
// Change player source and wait for loadeddata event, then play video
        // loadedmetadata doesn't work right now for flash.
        // Probably because of https://github.com/videojs/video-js-swf/issues/124
        // If player preload is 'none' and then loadeddata not fired. So, we need timeupdate event for seek handle (timeupdate doesn't work properly with flash)
        
var handleSeekEvent 'loadeddata';
        if(
this.player_.techName_ !== 'Youtube' && this.player_.preload() === 'none' && this.player_.techName_ !== 'Flash') {
          
handleSeekEvent 'timeupdate';
        }
        
player
          
.setSourcesSanitized(sourceslabelcustomSourcePicker || settings.customSourcePicker)
          .
one(handleSeekEvent, function() {
            
player.currentTime(currentTime);
            
player.handleTechSeeked_();
            if(!
isPaused){
              
// Start playing and hide loadingSpinner (flash issue ?)
              
player.play().handleTechSeeked_();
            }
            
player.trigger('resolutionchange');
          });
        return 
player;
      };

      
/**
       * Returns grouped sources by label, resolution and type
       * @returns {Object} grouped sources: { label: { key: [] }, res: { key: [] }, type: { key: [] } }
       */
      
player.getGroupedSrc = function(){
        return 
this.groupedSrc;
      };

      
player.setSourcesSanitized = function(sourceslabelcustomSourcePicker) {
        
this.currentResolutionState = {
          
labellabel,
          
sourcessources
        
};
        if(
typeof customSourcePicker === 'function'){
          return 
customSourcePicker(playersourceslabel);
        }
        
player.src(sources.map(function(src) {
          return {
srcsrc.srctypesrc.typeressrc.res};
        }));
        return 
player;
      };

      
/**
       * Method used for sorting list of sources
       * @param   {Object} a - source object with res property
       * @param   {Object} b - source object with res property
       * @returns {Number} result of comparation
       */
      
function compareResolutions(ab){
        if(!
a.res || !b.res){ return 0; }
        return (+
b.res)-(+a.res);
      }

      
/**
       * Group sources by label, resolution and type
       * @param   {Array}  src Array of sources
       * @returns {Object} grouped sources: { label: { key: [] }, res: { key: [] }, type: { key: [] } }
       */
      
function bucketSources(src){
        var 
resolutions = {
          
label: {},
          
res: {},
          
type: {}
        };
        
src.map(function(source) {
          
initResolutionKey(resolutions'label'source);
          
initResolutionKey(resolutions'res'source);
          
initResolutionKey(resolutions'type'source);

          
appendSourceToKey(resolutions'label'source);
          
appendSourceToKey(resolutions'res'source);
          
appendSourceToKey(resolutions'type'source);
        });
        return 
resolutions;
      }

      function 
initResolutionKey(resolutionskeysource) {
        if(
resolutions[key][source[key]] == null) {
          
resolutions[key][source[key]] = [];
        }
      }

      function 
appendSourceToKey(resolutionskeysource) {
        
resolutions[key][source[key]].push(source);
      }

      
/**
       * Choose src if option.default is specified
       * @param   {Object} groupedSrc {res: { key: [] }}
       * @param   {Array}  src Array of sources sorted by resolution used to find high and low res
       * @returns {Object} {res: string, sources: []}
       */
      
function chooseSrc(groupedSrcsrc){
        var 
selectedRes settings['default']; // use array access as default is a reserved keyword
        
var selectedLabel '';
        if (
selectedRes === 'high') {
          
selectedRes src[0].res;
          
selectedLabel src[0].label;
        } else if (
selectedRes === 'low' || selectedRes == null || !groupedSrc.res[selectedRes]) {
          
// Select low-res if default is low or not set
          
selectedRes src[src.length 1].res;
          
selectedLabel src[src.length -1].label;
        } else if (
groupedSrc.res[selectedRes]) {
          
selectedLabel groupedSrc.res[selectedRes][0].label;
        }

        return {
resselectedReslabelselectedLabelsourcesgroupedSrc.res[selectedRes]};
      }

      function 
initResolutionForYt(player){
        
// Map youtube qualities names
        
var _yts = {
          
highres: {res1080label'1080'yt'highres'},
          
hd1080: {res1080label'1080'yt'hd1080'},
          
hd720: {res720label'720'yt'hd720'},
          
large: {res480label'480'yt'large'},
          
medium: {res360label'360'yt'medium'},
          
small: {res240label'240'yt'small'},
          
tiny: {res144label'144'yt'tiny'},
          
auto: {res0label'auto'yt'auto'}
        };
        
// Overwrite default sourcePicker function
        
var _customSourcePicker = function(_player_sources_label){
          
// Note that setPlayebackQuality is a suggestion. YT does not always obey it.
          
player.tech_.ytPlayer.setPlaybackQuality(_sources[0]._yt);
          
player.trigger('updateSources');
          return 
player;
        };
        
settings.customSourcePicker _customSourcePicker;

        
// Init resolution
        
player.tech_.ytPlayer.setPlaybackQuality('auto');

        
// This is triggered when the resolution actually changes
        
player.tech_.ytPlayer.addEventListener('onPlaybackQualityChange', function(event){
          for(var 
res in _yts) {
            if(
res.yt === event.data) {
              
player.currentResolution(res.label_customSourcePicker);
              return;
            }
          }
        });

        
// We must wait for play event
        
player.one('play', function(){
          var 
qualities player.tech_.ytPlayer.getAvailableQualityLevels();
          var 
_sources = [];

          
qualities.map(function(q){
            
_sources.push({
              
srcplayer.src().src,
              
typeplayer.src().type,
              
label_yts[q].label,
              
res_yts[q].res,
              
_yt_yts[q].yt
            
});
          });

          
player.groupedSrc bucketSources(_sources);
          var 
chosen = {label'auto'res0sourcesplayer.groupedSrc.label.auto};

          
this.currentResolutionState = {
            
labelchosen.label,
            
sourceschosen.sources
          
};

          
player.trigger('updateSources');
          
player.setSourcesSanitized(chosen.sourceschosen.label_customSourcePicker);
        });
      }

      
player.ready(function(){
        if( 
settings.ui ) {
          var 
menuButton = new ResolutionMenuButton(playersettings);
          
player.controlBar.resolutionSwitcher player.controlBar.el_.insertBefore(menuButton.el_player.controlBar.getChild('fullscreenToggle').el_);
          
player.controlBar.resolutionSwitcher.dispose = function(){
            
this.parentNode.removeChild(this);
          };
        }
        if(
player.options_.sources.length 1){
          
// tech: Html5 and Flash
          // Create resolution switcher for videos form <source> tag inside <video>
          
player.updateSrc(player.options_.sources);
        }

        if(
player.techName_ === 'Youtube'){
         
// tech: YouTube
         
initResolutionForYt(player);
        }
      });

    };

    
// register the plugin
    
videojs.plugin('videoJsResolutionSwitcher'videoJsResolutionSwitcher);
  })(
windowvideojs);
})();
?>
Онлайн: 0
Реклама