View file maccman-flarevideo-ee31f65/javascripts/flarevideo.js

File size: 15.81Kb
(function($){

var idleEvents = "mousemove keydown DOMMouseScroll mousewheel mousedown reset.idle";
$.fn.idleTimer = function(options){
  options  = options || {};
  var element  = $(this);

  var idle     = false;  
  var timeout  = options.timeout  || 4000; // 3 seconds
  var interval = options.interval || 1000; // 1 second
  var timeFromLastEvent = 0;
  
  var reset = function(){ 
    if (idle) element.trigger("idle", false);
    idle = false;
    timeFromLastEvent = 0; 
  };
  var check = function(){
    if (timeFromLastEvent >= timeout) {
      reset();
      idle = true;
      element.trigger("idle", true);
    } else {
      timeFromLastEvent += interval;
    }
  };
  
  element.bind(idleEvents, reset);
  var loop = setInterval(check, interval);
  element.unload(function(){
    clearInterval(loop);
  });
};

var testVideoElement = $("<video />")[0];
var nativeSupport    = (typeof testVideoElement.canPlayType != 'undefined');
var nativeFullScreenSupport = (typeof testVideoElement.webkitEnterFullScreen != 'undefined');

// webkitEnterFullScreen fails under Chrome at the moment
if (navigator.userAgent.match('Chrome')) nativeFullScreenSupport = false;

var testRangeElement   = $("<input type='range' />")[0];
var nativeRangeSupport = (typeof testRangeElement.max != 'undefined');

var defaults = {
  autoplay: false,
  controls: false,
  preload:  "auto",
  poster:   null,
  srcs:     [],
  keyShortcut: true,
  flashSrc:    "media/FlareVideo.swf"
};

defaults.useNative           = nativeSupport;
defaults.useNativeFullScreen = nativeFullScreenSupport;

var FlareVideo = function(parent, options){
  this._class  = FlareVideo;
  
  this.parent  = parent;
  this.options = $.extend({}, defaults, options);
  this.sources = this.options.srcs || this.options.sources;
  this.useNative = this.options.useNative;
  // Only use full screen on HTML 5 atm
  this.options.useFullScreen = !!this.useNative;

  this.state        = null;
  this.canPlay      = false;
  this.inFullScreen = false;
  this.loaded       = false;
  this.readyList    = [];
  
  this.setupElement();
  
  if (this.options.useNative) {
    this.setupNative();
  } else {
    this.setupFlash();
  }
  
  this.ready($.proxy(function(){
    this.setupEvents();
    this.setupControls();
    this.change("initial");
    this.load();
    
    this.element.idleTimer();
    this.element.bind("idle", $.proxy(this.idle, this));
    this.element.bind("state.fv", $.proxy(function(){
      this.element.trigger("reset.idle");
    }, this))
  }, this));
};

FlareVideo.fn = FlareVideo.prototype;

// Public methods

FlareVideo.fn.ready = function(callback){
  this.readyList.push(callback);  
  if (this.loaded) callback.call(this);
};

FlareVideo.fn.load = function(srcs){
  if (srcs)
    this.sources = srcs;
  
  if (typeof this.sources == "string")
    this.sources = {src:this.sources};
  
  if (!$.isArray(this.sources))
    this.sources = [this.sources];
    
  this.ready(function(){
    this.change("loading");
    this.video.loadSources(this.sources);
  });
};

FlareVideo.fn.play = function(){
  this.video.play();
};

FlareVideo.fn.pause = function(){
  this.video.pause();
};

FlareVideo.fn.stop = function(){
  this.seek(0);
  this.pause();
};

FlareVideo.fn.togglePlay = function(){
  if (this.state == "playing") {
    this.pause();
  } else {
    this.play();
  }
};

FlareVideo.fn.fullScreen = function(state){
  if (typeof state == "undefined") state = true;
  this.inFullScreen = state;
  if (this.options.useNativeFullScreen) {
    this.video[state ? "enterFullScreen" : "exitFullScreen"]();
  } else {
    (state ? $("body") : this.parent).prepend(this.element);
    var isPlaying = (this.state == "playing");
    this.element[state ? "addClass" : "removeClass"]("fullScreen");
    if (isPlaying) this.play();
  }
};

FlareVideo.fn.toggleFullScreen = function(){
  this.fullScreen(!this.inFullScreen);
};

FlareVideo.fn.seek = function(offset){
  this.video.setCurrentTime(offset);
}

FlareVideo.fn.setVolume = function(num){
  this.video.setVolume(num);
};

FlareVideo.fn.getVolume = function(){
  return this.video.getVolume();
};

FlareVideo.fn.mute = function(state){
  if (typeof state == "undefined") state = true;
  this.setVolume(state ? 1 : 0);
};

FlareVideo.fn.remove = function(){
  this.element.remove();
};

FlareVideo.fn.bind = function(){
  this.videoElement.bind.apply(this.videoElement, arguments);
};

FlareVideo.fn.one = function(){
  this.videoElement.one.apply(this.videoElement, arguments);
};

FlareVideo.fn.trigger = function(){
  this.videoElement.trigger.apply(this.videoElement, arguments);
};

// Proxy jQuery events
var events = ["click", "dblclick", 
              "onerror", "onloadeddata", "oncanplay", 
              "ondurationchange", "ontimeupdate",
              "onpause", "onplay", "onended", "onvolumechange"];
for (var i=0; i < events.length; i++) {
  (function(){
    var functName = events[i];
    var eventName = functName.replace(/^(on)/, "");
    FlareVideo.fn[functName] = function(){
      var args = $.makeArray(arguments);
      args.unshift(eventName);
      this.bind.apply(this, args);
    };
  })();
}

// Private methods

FlareVideo.fn.triggerReady = function(){
  for (var i in this.readyList) {
		this.readyList[i].call(this);
	}
	this.loaded = true;
};

FlareVideo.fn.setupElement = function(){
  this.element = $("<div />");
  this.element.addClass("flareVideo");
  this.parent.append(this.element);
};

FlareVideo.fn.idle = function(e, toggle){
  if (toggle) {
    if (this.state == "playing")
      this.element.addClass("idle");
  } else {
    this.element.removeClass("idle");
  }
};

FlareVideo.fn.change = function(state){
  this.state = state;
  this.element.attr("data-state", this.state);
  this.element.trigger("state.fv", this.state);
}

FlareVideo.fn.setupNative = function(){
  this.videoElement = $("<video />");
  this.videoElement.addClass("video");
  this.videoElement.attr({
    width:    this.options.width,
    height:   this.options.height,
    poster:   this.options.poster,
    autoplay: this.options.autoplay,
    preload:  this.options.preload,
    controls: this.options.controls,
    autobuffer: this.options.autobuffer
  });
  
  this.element.append(this.videoElement);
  this.video = this.videoElement[0];
  
  var self = this;
  
  this.video.loadSources = function(srcs){
    self.videoElement.empty();
    for (var i in srcs) {
      var srcEl = $("<source />");
      srcEl.attr(srcs[i]);
      self.videoElement.append(srcEl);
    }
    self.video.load();
  };
  
  this.video.getStartTime   = function(){ return(this.startTime || 0); };
  this.video.getEndTime     = function(){
    if (this.duration == Infinity && this.buffered) {
      return(this.buffered.end(this.buffered.length-1));
    } else {
      return((this.startTime || 0) + this.duration);
    }
  };
  
  this.video.getCurrentTime = function(){ 
    try {
      return this.currentTime;
    } catch(e) {
      return 0;
    }
  };
  
  var self = this;

  this.video.setCurrentTime = function(val){ this.currentTime = val; }
  this.video.getVolume      = function(){ return this.volume; };
  this.video.setVolume      = function(val){ this.volume = val; };
  this.video.enterFullScreen = function(){ 
    // Because we don't know when full screen is exited
    self.inFullScreen = false;
    this.webkitEnterFullScreen(); 
  };
  this.video.exitFullScreen  = function(){ this.webkitExitFullScreen(); };
  
  this.videoElement.dblclick($.proxy(function(){
    this.toggleFullScreen();
  }, this));
    
  this.triggerReady();
};

FlareVideo.fn.setupFlash = function(){
  if (!this._class.flashInstance) this._class.flashInstance = [];
  
  var flashID = this._class.flashInstance.length;
  this._class.flashInstance[flashID] = this;
  
  this.element.addClass("flash");
  
  this.videoElement = $("<div />");
  this.videoElement.addClass("video"); 
  
  this.videoElement.flash({ 
    src: this.options.flashSrc, 
    wmode: "opaque",
    flashvars: {flashID:flashID},
    allowScriptAccess: "sameDomain",
    allowFullScreen: true,
    width: "100%",
    height: "100%"
  },{ 
    version: 9, 
    expressInstall: true,
  });
  
  this.video = this.videoElement.find("embed")[0];
  if (!this.video) throw 'Flash Player not installed';
  
  if ($.browser.msie)
    this.fixExternalInterface();
  
  var self = this;
  this.video.loadSources = function(srcs){
    if (!srcs) return;
    var source = self._class.flashSources(srcs)[0];
    if (!source || !source.src) return;
    this.loadSource(source.src);
  };
  
  this.element.append(this.videoElement);
};

// External Interface and IE is broken
FlareVideo.fn.fixExternalInterface = function(){
  var __flash__addCallback = function(instance, name) {
    instance[name] = function () { 
      return eval(instance.CallFunction("<invoke name=\"" + name + "\" returntype=\"javascript\">" + 
                  __flash__argumentsToXML(arguments,0) + "</invoke>"));
    }
  };
  var methods = ["loadSource", "getStartTime", "getCurrentTime", 
                 "setCurrentTime", "getEndTime", "getVolume", 
                 "setVolume", "play", "pause"];
  for(var i in methods)
    __flash__addCallback(this.video, methods[i]);
};

FlareVideo.flashSources = function(sources){
  return($.grep(sources, function(i){
    return(i.type.match(/flv/) || i.type.match(/mp4/));
  }));
};
FlareVideo.fn.flashSources = function(){ 
  return this._class.flashSources(this.sources); 
};

FlareVideo.eiTrigger = function(id, name){
  try {
    this.flashInstance[id].trigger(name);
  } catch(e) {
    console.error(e);
  }
};

FlareVideo.eiTriggerReady = function(id) {
  try {
    this.flashInstance[id].triggerReady();
  } catch(e) {
    console.error(e);
  }
};

FlareVideo.fn.setupButtons = function(){
  var play = $("<div />");
  play.addClass("play");
  play.text("Play");
  play.click($.proxy(function(){
    if (!this.canPlay) return;
    this.play();
  }, this));
  this.controls.append(play);
  
  var pause = $("<div />");
  pause.addClass("pause");
  pause.text("Pause");
  pause.click($.proxy(function(){
    if (!this.canPlay) return;
    this.pause();
  }, this));
  this.controls.append(pause);

  var fullScreen = $("<div />");
  fullScreen.addClass("fullScreen");
  fullScreen.text("Full Screen");
  fullScreen.click($.proxy(this.toggleFullScreen, this));
  if (!this.options.useFullScreen) fullScreen.addClass("disabled");
  this.controls.append(fullScreen);
};

FlareVideo.fn.createRange = function(){
  if (nativeRangeSupport) {
    var result = $("<input type='range' />");
    result.attr({step: "any"});
    result.getValue   = result.val;
    result.setValue   = result.val;
    result.setOptions = result.attr;
  } else {
    var result = $("<div />");    
    if (!result.slider)
      throw "jQuery UI with the slider component is required."
    
    result.slider();
    
    var currentValue = 0;
    result.getValue = function(){ 
      return currentValue;
    };
    
    result.setValue = function(value){
      if (result.find(".ui-slider-handle:first").hasClass("ui-state-active")) return;
      currentValue = value;
      result.slider("option", "value", value);
    };
        
    result.setOptions = function(options){
      result.slider("option", options);
    };
    
    result.bind("slidestop", function(e, ui){ 
      currentValue = ui.value;
      result.trigger("change") 
    });    
  }
  return result;
}

FlareVideo.fn.setupSeek = function(){
  var seek = $("<div />");
  seek.addClass("seek");
  
  var seekRange = this.createRange();
  seekRange.addClass("seekRange");
  seekRange.setValue(0);

  seekRange.change($.proxy(function(){
    this.seek(seekRange.getValue());
  }, this));

  this.ondurationchange($.proxy(function(){
    seekRange.setOptions({
      min:   this.video.getStartTime(),
      max:   this.video.getEndTime(),
      value: this.video.getCurrentTime()
    });
  }, this));
  
  this.ontimeupdate($.proxy(function(){
    seekRange.setOptions({
      max: this.video.getEndTime(),
      value: this.video.getCurrentTime()
    });
  }, this));
  
  seek.append(seekRange);
  this.controls.append(seek);
};

FlareVideo.fn.setupVolume = function(){
  var volume = $("<div />");
  volume.addClass("volume");
  
  var volRange = this.createRange();
  volRange.setOptions({
    max:  1,
    step: 0.1
  });
  
  volRange.addClass("volRange");
  volRange.change($.proxy(function(){
    this.setVolume(volRange.getValue());
  }, this));
  this.onvolumechange($.proxy(function(){
    volRange.setValue(this.getVolume());
  }, this));
  
  volRange.setValue(this.getVolume());
  
  var volMin = $("<div />");
  volMin.addClass("volMin");
  volMin.text("Minimum volume");
  volMin.click($.proxy(function(){
    this.setVolume(0);
  }, this));
  
  var volMax = $("<div />");
  volMax.addClass("volMax");
  volMax.text("Maximum volume");
  volMax.click($.proxy(function(){
    this.setVolume(1);
  }, this));
  
  volume.append(volMin);
  volume.append(volRange);
  volume.append(volMax);
  this.controls.append(volume);
};

FlareVideo.fn.setupTiming = function(){  
  var timeToGo = $("<div />");
  var timeLeft = $("<div />");
  
  timeToGo.addClass("timeMin");
  timeLeft.addClass("timeMax");
  
  var pad = function(num) {
    if (num < 10)
      return "0" + num;
    return num;
  }
  
  var secondsFormat = function(sec){
    var result  = [];
    
    var minutes = Math.floor( sec / 60 );
    var hours   = Math.floor( sec / 3600 );
    var seconds = (sec == 0) ? 0 : (sec % 60)
    seconds     = Math.round(seconds);
        
    if (hours > 0)
      result.push(pad(hours));
    
    result.push(pad(minutes));
    result.push(pad(seconds));
    
    return result.join(":");
  };
  
  this.ontimeupdate($.proxy(function(){    
    timeToGo.text(secondsFormat(this.video.getCurrentTime()));    
    timeLeft.text("-" + secondsFormat(this.video.getEndTime() - this.video.getCurrentTime()))
  }, this));
  
  this.videoElement.one("canplay", $.proxy(function(){
    this.videoElement.trigger("timeupdate");
  }, this));
  
  this.controls.append(timeToGo);
  this.controls.append(timeLeft);
};

FlareVideo.fn.setupControls = function(){
  // Use native controls
  if (this.options.controls) return;
  
  this.controls = $("<div />");
  this.controls.addClass("controls");
  this.controls.addClass("disabled");
  
  this.setupButtons();
  this.setupSeek();
  this.setupVolume();
  this.setupTiming();
  
  this.element.append(this.controls);
};

FlareVideo.fn.fallbackToFlash = function(){
  this.useNative = false;
  this.element.unload();
  this.remove()
  this.setupElement();
  this.setupFlash();
};

FlareVideo.fn.setupEvents = function(){
  this.onpause($.proxy(function(){
    this.element.removeClass("playing");
    this.change("paused");
  }, this));
  
  this.onplay($.proxy(function(){
    this.element.addClass("playing");
    this.change("playing");
  }, this));
  
  this.onended($.proxy(function(){
    this.element.removeClass("playing");
    this.fullScreen(false);
    this.stop()
    this.change("ended");
  }, this));

  this.onerror($.proxy(function(e){
    if (this.useNative) {
      if (this.video.error && this.video.error.code == 4) {
        var flashType = this.flashSources()[0];
        if (flashType) {
          this.fallbackToFlash();
        } else {
          console.error("Format not supported");
        }
      } else {
        console.error("Error - " + this.video.error);
      }
    } else {
      console.error("Flash error");
    }
  }, this));
  
  this.oncanplay($.proxy(function(){
    this.canPlay = true;
    this.controls.removeClass("disabled");
  }, this));
  
  if (this.options.keyShortcut);
    $(document).keydown($.proxy(function(e){
      if (e.keyCode == 32) { // Space
        this.togglePlay();
        return false;
      }
      
      if (e.keyCode == 27 && this.inFullScreen) { // Escape
        this.fullScreen(false);
        this.element.trigger("reset.idle");
        return false;
      }
      
    }, this));
};

$.fn.flareVideo = function(options, callback){
  return(new FlareVideo(this, options));
};

window.FlareVideo = FlareVideo;

})(jQuery);