/* * jPlayer Plugin for jQuery JavaScript Library * http://www.jplayer.org * * Copyright (c) 2009 - 2011 Happyworm Ltd * Dual licensed under the MIT and GPL licenses. * - http://www.opensource.org/licenses/mit-license.php * - http://www.gnu.org/copyleft/gpl.html * * Author: Mark J Panaghiston * Version: 2.0.15 * Date: 21th June 2011 */ /* Code verified using http://www.jshint.com/ */ /*jshint asi:false, bitwise:false, boss:false, browser:true, curly:true, debug:false, eqeqeq:true, eqnull:false, evil:false, forin:false, immed:false, jquery:true, laxbreak:false, newcap:true, noarg:true, noempty:true, nonew:true, nomem:false, onevar:false, passfail:false, plusplus:false, regexp:false, undef:true, sub:false, strict:false, white:false */ /*global jQuery:false, ActiveXObject:false, alert:false */ (function($, undefined) { // Adapted from jquery.ui.widget.js (1.8.7): $.widget.bridge $.fn.jPlayer = function( options ) { var name = "jPlayer"; var isMethodCall = typeof options === "string", args = Array.prototype.slice.call( arguments, 1 ), returnValue = this; // allow multiple hashes to be passed on init options = !isMethodCall && args.length ? $.extend.apply( null, [ true, options ].concat(args) ) : options; // prevent calls to internal methods if ( isMethodCall && options.charAt( 0 ) === "_" ) { return returnValue; } if ( isMethodCall ) { this.each(function() { var instance = $.data( this, name ), methodValue = instance && $.isFunction( instance[options] ) ? instance[ options ].apply( instance, args ) : instance; if ( methodValue !== instance && methodValue !== undefined ) { returnValue = methodValue; return false; } }); } else { this.each(function() { var instance = $.data( this, name ); if ( instance ) { // instance.option( options || {} )._init(); // Orig jquery.ui.widget.js code: Not recommend for jPlayer. ie., Applying new options to an existing instance (via the jPlayer constructor) and performing the _init(). The _init() is what concerns me. It would leave a lot of event handlers acting on jPlayer instance and the interface. instance.option( options || {} ); // The new constructor only changes the options. Changing options only has basic support atm. } else { $.data( this, name, new $.jPlayer( options, this ) ); } }); } return returnValue; }; $.jPlayer = function( options, element ) { // allow instantiation without initializing for simple inheritance if ( arguments.length ) { this.element = $(element); this.options = $.extend(true, {}, this.options, options ); var self = this; this.element.bind( "remove.jPlayer", function() { self.destroy(); }); this._init(); } }; // End of: (Adapted from jquery.ui.widget.js (1.8.7)) // Emulated HTML5 methods and properties $.jPlayer.emulateMethods = "load play pause"; $.jPlayer.emulateStatus = "src readyState networkState currentTime duration paused ended playbackRate"; $.jPlayer.emulateOptions = "muted volume"; // Reserved event names generated by jPlayer that are not part of the HTML5 Media element spec $.jPlayer.reservedEvent = "ready resize error warning"; // Events generated by jPlayer $.jPlayer.event = { ready: "jPlayer_ready", resize: "jPlayer_resize", // Not implemented. error: "jPlayer_error", // Event error code in event.jPlayer.error.type. See $.jPlayer.error warning: "jPlayer_warning", // Event warning code in event.jPlayer.warning.type. See $.jPlayer.warning // Other events match HTML5 spec. loadstart: "jPlayer_loadstart", progress: "jPlayer_progress", suspend: "jPlayer_suspend", abort: "jPlayer_abort", emptied: "jPlayer_emptied", stalled: "jPlayer_stalled", play: "jPlayer_play", pause: "jPlayer_pause", loadedmetadata: "jPlayer_loadedmetadata", loadeddata: "jPlayer_loadeddata", waiting: "jPlayer_waiting", playing: "jPlayer_playing", canplay: "jPlayer_canplay", canplaythrough: "jPlayer_canplaythrough", seeking: "jPlayer_seeking", seeked: "jPlayer_seeked", timeupdate: "jPlayer_timeupdate", ended: "jPlayer_ended", ratechange: "jPlayer_ratechange", durationchange: "jPlayer_durationchange", volumechange: "jPlayer_volumechange" }; $.jPlayer.htmlEvent = [ // These HTML events are bubbled through to the jPlayer event, without any internal action. "loadstart", // "progress", // jPlayer uses internally before bubbling. // "suspend", // jPlayer uses internally before bubbling. "abort", // "error", // jPlayer uses internally before bubbling. "emptied", "stalled", // "play", // jPlayer uses internally before bubbling. // "pause", // jPlayer uses internally before bubbling. "loadedmetadata", "loadeddata", // "waiting", // jPlayer uses internally before bubbling. // "playing", // jPlayer uses internally before bubbling. // "canplay", // jPlayer fixes the volume (for Chrome) before bubbling. "canplaythrough", // "seeking", // jPlayer uses internally before bubbling. // "seeked", // jPlayer uses internally before bubbling. // "timeupdate", // jPlayer uses internally before bubbling. // "ended", // jPlayer uses internally before bubbling. "ratechange" // "durationchange" // jPlayer uses internally before bubbling. // "volumechange" // Handled by jPlayer in volume() method, primarily due to the volume fix (for Chrome) in the canplay event. [*] Need to review whether the latest Chrome still needs the fix sometime. ]; $.jPlayer.pause = function() { // $.each($.jPlayer.instances, function(i, element) { $.each($.jPlayer.prototype.instances, function(i, element) { if(element.data("jPlayer").status.srcSet) { // Check that media is set otherwise would cause error event. element.jPlayer("pause"); } }); }; $.jPlayer.timeFormat = { showHour: false, showMin: true, showSec: true, padHour: false, padMin: true, padSec: true, sepHour: ":", sepMin: ":", sepSec: "" }; $.jPlayer.convertTime = function(s) { var myTime = new Date(s * 1000); var hour = myTime.getUTCHours(); var min = myTime.getUTCMinutes(); var sec = myTime.getUTCSeconds(); var strHour = ($.jPlayer.timeFormat.padHour && hour < 10) ? "0" + hour : hour; var strMin = ($.jPlayer.timeFormat.padMin && min < 10) ? "0" + min : min; var strSec = ($.jPlayer.timeFormat.padSec && sec < 10) ? "0" + sec : sec; return (($.jPlayer.timeFormat.showHour) ? strHour + $.jPlayer.timeFormat.sepHour : "") + (($.jPlayer.timeFormat.showMin) ? strMin + $.jPlayer.timeFormat.sepMin : "") + (($.jPlayer.timeFormat.showSec) ? strSec + $.jPlayer.timeFormat.sepSec : ""); }; // Adapting jQuery 1.4.4 code for jQuery.browser. Required since jQuery 1.3.2 does not detect Chrome as webkit. $.jPlayer.uaBrowser = function( userAgent ) { var ua = userAgent.toLowerCase(); // Useragent RegExp var rwebkit = /(webkit)[ \/]([\w.]+)/; var ropera = /(opera)(?:.*version)?[ \/]([\w.]+)/; var rmsie = /(msie) ([\w.]+)/; var rmozilla = /(mozilla)(?:.*? rv:([\w.]+))?/; var match = rwebkit.exec( ua ) || ropera.exec( ua ) || rmsie.exec( ua ) || ua.indexOf("compatible") < 0 && rmozilla.exec( ua ) || []; return { browser: match[1] || "", version: match[2] || "0" }; }; // Platform sniffer for detecting mobile devices $.jPlayer.uaPlatform = function( userAgent ) { var ua = userAgent.toLowerCase(); // Useragent RegExp var rplatform = /(ipad|iphone|ipod|android|blackberry|playbook|windows ce|webos)/; var rtablet = /(ipad|playbook)/; var randroid = /(android)/; var rmobile = /(mobile)/; var platform = rplatform.exec( ua ) || []; var tablet = rtablet.exec( ua ) || !rmobile.exec( ua ) && randroid.exec( ua ) || []; return { platform: platform[1] || "", tablet: tablet[1] || "" }; }; $.jPlayer.browser = { }; $.jPlayer.platform = { }; var browserMatch = $.jPlayer.uaBrowser(navigator.userAgent); if ( browserMatch.browser ) { $.jPlayer.browser[ browserMatch.browser ] = true; $.jPlayer.browser.version = browserMatch.version; } var platformMatch = $.jPlayer.uaPlatform(navigator.userAgent); if ( platformMatch.platform ) { $.jPlayer.platform[ platformMatch.platform ] = true; $.jPlayer.platform.mobile = !platformMatch.tablet; $.jPlayer.platform.tablet = !!platformMatch.tablet; } $.jPlayer.prototype = { count: 0, // Static Variable: Change it via prototype. version: { // Static Object script: "2.0.15", needFlash: "2.0.9", flash: "unknown" }, options: { // Instanced in $.jPlayer() constructor swfPath: "js", // Path to Jplayer.swf. Can be relative, absolute or server root relative. solution: "html, flash", // Valid solutions: html, flash. Order defines priority. 1st is highest, supplied: "mp3", // Defines which formats jPlayer will try and support and the priority by the order. 1st is highest, preload: 'metadata', // HTML5 Spec values: none, metadata, auto. volume: 0.8, // The volume. Number 0 to 1. muted: false, wmode: "window", // Default Flash wmode is: window. Valid wmode: transparent, opaque, direct, gpu backgroundColor: "#000000", // To define the jPlayer div and Flash background color. cssSelectorAncestor: "#jp_container_1", cssSelector: { // * denotes properties that should only be required when video media type required. _cssSelector() would require changes to enable splitting these into Audio and Video defaults. videoPlay: ".jp-video-play", // * play: ".jp-play", pause: ".jp-pause", stop: ".jp-stop", seekBar: ".jp-seek-bar", playBar: ".jp-play-bar", mute: ".jp-mute", unmute: ".jp-unmute", volumeBar: ".jp-volume-bar", volumeBarValue: ".jp-volume-bar-value", currentTime: ".jp-current-time", duration: ".jp-duration", fullScreen: ".jp-full-screen", // * restoreScreen: ".jp-restore-screen" // * }, fullScreen: false, // globalVolume: false, // Not implemented: Set to make volume changes affect all jPlayer instances // globalMute: false, // Not implemented: Set to make mute changes affect all jPlayer instances idPrefix: "jp", // Prefix for the ids of html elements created by jPlayer. For flash, this must not include characters: . - + * / \ noConflict: "jQuery", emulateHtml: false, // Emulates the HTML5 Media element on the jPlayer element. errorAlerts: false, warningAlerts: false }, optionsAudio: { size: { width: "0px", height: "0px", cssClass: "" }, sizeFull: { width: "0px", height: "0px", cssClass: "" } }, optionsVideo: { size: { width: "480px", height: "270px", cssClass: "jp-video-270p" }, sizeFull: { width: "100%", height: "90%", cssClass: "jp-video-full" } }, instances: {}, // Static Object status: { // Instanced in _init() src: "", media: {}, paused: true, format: {}, formatType: "", waitForPlay: true, // Same as waitForLoad except in case where preloading. waitForLoad: true, srcSet: false, video: false, // True if playing a video seekPercent: 0, currentPercentRelative: 0, currentPercentAbsolute: 0, currentTime: 0, duration: 0, readyState: 0, networkState: 0, playbackRate: 1, ended: 0 /* Persistant status properties created dynamically at _init(): width height cssClass */ }, internal: { // Instanced in _init() ready: false // instance: undefined, // domNode: undefined, // htmlDlyCmdId: undefined }, solution: { // Static Object: Defines the solutions built in jPlayer. html: true, flash: true }, // 'MPEG-4 support' : canPlayType('video/mp4; codecs="mp4v.20.8"') format: { // Static Object mp3: { codec: 'audio/mpeg; codecs="mp3"', flashCanPlay: true, media: 'audio' }, m4a: { // AAC / MP4 codec: 'audio/mp4; codecs="mp4a.40.2"', flashCanPlay: true, media: 'audio' }, oga: { // OGG codec: 'audio/ogg; codecs="vorbis"', flashCanPlay: false, media: 'audio' }, wav: { // PCM codec: 'audio/wav; codecs="1"', flashCanPlay: false, media: 'audio' }, webma: { // WEBM codec: 'audio/webm; codecs="vorbis"', flashCanPlay: false, media: 'audio' }, fla: { // FLV / F4A codec: 'audio/x-flv', flashCanPlay: true, media: 'audio' }, m4v: { // H.264 / MP4 codec: 'video/mp4; codecs="avc1.42E01E, mp4a.40.2"', flashCanPlay: true, media: 'video' }, ogv: { // OGG codec: 'video/ogg; codecs="theora, vorbis"', flashCanPlay: false, media: 'video' }, webmv: { // WEBM codec: 'video/webm; codecs="vorbis, vp8"', flashCanPlay: false, media: 'video' }, flv: { // FLV / F4V codec: 'video/x-flv', flashCanPlay: true, media: 'video' } }, _init: function() { var self = this; this.element.empty(); this.status = $.extend({}, this.status); // Copy static to unique instance. this.internal = $.extend({}, this.internal); // Copy static to unique instance. this.internal.domNode = this.element.get(0); this.formats = []; // Array based on supplied string option. Order defines priority. this.solutions = []; // Array based on solution string option. Order defines priority. this.require = {}; // Which media types are required: video, audio. this.htmlElement = {}; // DOM elements created by jPlayer this.html = {}; // In _init()'s this.desired code and setmedia(): Accessed via this[solution], where solution from this.solutions array. this.html.audio = {}; this.html.video = {}; this.flash = {}; // In _init()'s this.desired code and setmedia(): Accessed via this[solution], where solution from this.solutions array. this.css = {}; this.css.cs = {}; // Holds the css selector strings this.css.jq = {}; // Holds jQuery selectors. ie., $(css.cs.method) this.ancestorJq = []; // Holds jQuery selector of cssSelectorAncestor. Init would use $() instead of [], but it is only 1.4+ this.options.volume = this._limitValue(this.options.volume, 0, 1); // Limit volume value's bounds. // Create the formats array, with prority based on the order of the supplied formats string $.each(this.options.supplied.toLowerCase().split(","), function(index1, value1) { var format = value1.replace(/^\s+|\s+$/g, ""); //trim if(self.format[format]) { // Check format is valid. var dupFound = false; $.each(self.formats, function(index2, value2) { // Check for duplicates if(format === value2) { dupFound = true; return false; } }); if(!dupFound) { self.formats.push(format); } } }); // Create the solutions array, with prority based on the order of the solution string $.each(this.options.solution.toLowerCase().split(","), function(index1, value1) { var solution = value1.replace(/^\s+|\s+$/g, ""); //trim if(self.solution[solution]) { // Check solution is valid. var dupFound = false; $.each(self.solutions, function(index2, value2) { // Check for duplicates if(solution === value2) { dupFound = true; return false; } }); if(!dupFound) { self.solutions.push(solution); } } }); this.internal.instance = "jp_" + this.count; this.instances[this.internal.instance] = this.element; // Check the jPlayer div has an id and create one if required. Important for Flash to know the unique id for comms. if(this.element.attr("id") === "") { this.element.attr("id", this.options.idPrefix + "_jplayer_" + this.count); } this.internal.self = $.extend({}, { id: this.element.attr("id"), jq: this.element }); this.internal.audio = $.extend({}, { id: this.options.idPrefix + "_audio_" + this.count, jq: undefined }); this.internal.video = $.extend({}, { id: this.options.idPrefix + "_video_" + this.count, jq: undefined }); this.internal.flash = $.extend({}, { id: this.options.idPrefix + "_flash_" + this.count, jq: undefined, swf: this.options.swfPath + ((this.options.swfPath !== "" && this.options.swfPath.slice(-1) !== "/") ? "/" : "") + "Jplayer.swf" }); this.internal.poster = $.extend({}, { id: this.options.idPrefix + "_poster_" + this.count, jq: undefined }); // Register listeners defined in the constructor $.each($.jPlayer.event, function(eventName,eventType) { if(self.options[eventName] !== undefined) { self.element.bind(eventType + ".jPlayer", self.options[eventName]); // With .jPlayer namespace. self.options[eventName] = undefined; // Destroy the handler pointer copy on the options. Reason, events can be added/removed in other ways so this could be obsolete and misleading. } }); // Determine if we require solutions for audio, video or both media types. this.require.audio = false; this.require.video = false; $.each(this.formats, function(priority, format) { self.require[self.format[format].media] = true; }); // Now required types are known, finish the options default settings. if(this.require.video) { this.options = $.extend(true, {}, this.optionsVideo, this.options ); } else { this.options = $.extend(true, {}, this.optionsAudio, this.options ); } this._setSize(); // update status and jPlayer element size // Create the poster image. this.htmlElement.poster = document.createElement('img'); this.htmlElement.poster.id = this.internal.poster.id; this.htmlElement.poster.onload = function() { // Note that this did not work on Firefox 3.6: poster.addEventListener("onload", function() {}, false); Did not investigate x-browser. if(!self.status.video || self.status.waitForPlay) { self.internal.poster.jq.show(); } }; this.element.append(this.htmlElement.poster); this.internal.poster.jq = $("#" + this.internal.poster.id); this.internal.poster.jq.css({'width': this.status.width, 'height': this.status.height}); this.internal.poster.jq.hide(); // Generate the required media elements this.html.audio.available = false; if(this.require.audio) { // If a supplied format is audio this.htmlElement.audio = document.createElement('audio'); this.htmlElement.audio.id = this.internal.audio.id; this.html.audio.available = !!this.htmlElement.audio.canPlayType; } this.html.video.available = false; if(this.require.video) { // If a supplied format is video this.htmlElement.video = document.createElement('video'); this.htmlElement.video.id = this.internal.video.id; this.html.video.available = !!this.htmlElement.video.canPlayType; } this.flash.available = this._checkForFlash(10); this.html.canPlay = {}; this.flash.canPlay = {}; $.each(this.formats, function(priority, format) { self.html.canPlay[format] = self.html[self.format[format].media].available && "" !== self.htmlElement[self.format[format].media].canPlayType(self.format[format].codec); self.flash.canPlay[format] = self.format[format].flashCanPlay && self.flash.available; }); this.html.desired = false; this.flash.desired = false; $.each(this.solutions, function(solutionPriority, solution) { if(solutionPriority === 0) { self[solution].desired = true; } else { var audioCanPlay = false; var videoCanPlay = false; $.each(self.formats, function(formatPriority, format) { if(self[self.solutions[0]].canPlay[format]) { // The other solution can play if(self.format[format].media === 'video') { videoCanPlay = true; } else { audioCanPlay = true; } } }); self[solution].desired = (self.require.audio && !audioCanPlay) || (self.require.video && !videoCanPlay); } }); // This is what jPlayer will support, based on solution and supplied. this.html.support = {}; this.flash.support = {}; $.each(this.formats, function(priority, format) { self.html.support[format] = self.html.canPlay[format] && self.html.desired; self.flash.support[format] = self.flash.canPlay[format] && self.flash.desired; }); // If jPlayer is supporting any format in a solution, then the solution is used. this.html.used = false; this.flash.used = false; $.each(this.solutions, function(solutionPriority, solution) { $.each(self.formats, function(formatPriority, format) { if(self[solution].support[format]) { self[solution].used = true; return false; } }); }); // Init solution active state and the event gates to false. this.html.active = false; this.html.audio.gate = false; this.html.video.gate = false; this.flash.active = false; this.flash.gate = false; // Set up the css selectors for the control and feedback entities. this._cssSelectorAncestor(this.options.cssSelectorAncestor); // If neither html nor flash are being used by this browser, then media playback is not possible. Trigger an error event. if(!(this.html.used || this.flash.used)) { this._error( { type: $.jPlayer.error.NO_SOLUTION, context: "{solution:'" + this.options.solution + "', supplied:'" + this.options.supplied + "'}", message: $.jPlayer.errorMsg.NO_SOLUTION, hint: $.jPlayer.errorHint.NO_SOLUTION }); } // Add the flash solution if it is being used. if(this.flash.used) { var htmlObj, flashVars = 'jQuery=' + encodeURI(this.options.noConflict) + '&id=' + encodeURI(this.internal.self.id) + '&vol=' + this.options.volume + '&muted=' + this.options.muted; // Code influenced by SWFObject 2.2: http://code.google.com/p/swfobject/ // Non IE browsers have an initial Flash size of 1 by 1 otherwise the wmode affected the Flash ready event. if($.browser.msie && Number($.browser.version) <= 8) { var objStr = ''; var paramStr = [ '', '', '', '', '' ]; htmlObj = document.createElement(objStr); for(var i=0; i < paramStr.length; i++) { htmlObj.appendChild(document.createElement(paramStr[i])); } } else { var createParam = function(el, n, v) { var p = document.createElement("param"); p.setAttribute("name", n); p.setAttribute("value", v); el.appendChild(p); }; htmlObj = document.createElement("object"); htmlObj.setAttribute("id", this.internal.flash.id); htmlObj.setAttribute("data", this.internal.flash.swf); htmlObj.setAttribute("type", "application/x-shockwave-flash"); htmlObj.setAttribute("width", "1"); // Non-zero htmlObj.setAttribute("height", "1"); // Non-zero createParam(htmlObj, "flashvars", flashVars); createParam(htmlObj, "allowscriptaccess", "always"); createParam(htmlObj, "bgcolor", this.options.backgroundColor); createParam(htmlObj, "wmode", this.options.wmode); } this.element.append(htmlObj); this.internal.flash.jq = $(htmlObj); } // Add the HTML solution if being used. if(this.html.used) { // The HTML Audio handlers if(this.html.audio.available) { this._addHtmlEventListeners(this.htmlElement.audio, this.html.audio); this.element.append(this.htmlElement.audio); this.internal.audio.jq = $("#" + this.internal.audio.id); } // The HTML Video handlers if(this.html.video.available) { this._addHtmlEventListeners(this.htmlElement.video, this.html.video); this.element.append(this.htmlElement.video); this.internal.video.jq = $("#" + this.internal.video.id); this.internal.video.jq.css({'width':'0px', 'height':'0px'}); // Using size 0x0 since a .hide() causes issues in iOS } } // Create the bridge that emulates the HTML Media element on the jPlayer DIV if( this.options.emulateHtml ) { this._emulateHtmlBridge(); } if(this.html.used && !this.flash.used) { // If only HTML, then emulate flash ready() call after 100ms. window.setTimeout( function() { self.internal.ready = true; self.version.flash = "n/a"; self._trigger($.jPlayer.event.ready); }, 100); } this._updateInterface(); this._updateButtons(false); this._updateVolume(this.options.volume); this._updateMute(this.options.muted); if(this.css.jq.videoPlay.length) { this.css.jq.videoPlay.hide(); } $.jPlayer.prototype.count++; // Change static variable via prototype. }, destroy: function() { // MJP: The background change remains. Would need to store the original to restore it correctly. // Reset the interface, remove seeking effect and times. this._resetStatus(); this._updateInterface(); this._seeked(); if(this.css.jq.currentTime.length) { this.css.jq.currentTime.text(""); } if(this.css.jq.duration.length) { this.css.jq.duration.text(""); } if(this.status.srcSet) { // Or you get a bogus error event this.pause(); // Pauses the media and clears any delayed commands used in the HTML solution. } $.each(this.css.jq, function(fn, jq) { // Remove any bindings from the interface controls. // Check selector is valid before trying to execute method. if(jq.length) { jq.unbind(".jPlayer"); } }); if( this.options.emulateHtml ) { this._destroyHtmlBridge(); } this.element.removeData("jPlayer"); // Remove jPlayer data this.element.unbind(".jPlayer"); // Remove all event handlers created by the jPlayer constructor this.element.empty(); // Remove the inserted child elements this.instances[this.internal.instance] = undefined; // Clear the instance on the static instance object }, enable: function() { // Plan to implement // options.disabled = false }, disable: function () { // Plan to implement // options.disabled = true }, _addHtmlEventListeners: function(mediaElement, entity) { var self = this; mediaElement.preload = this.options.preload; mediaElement.muted = this.options.muted; mediaElement.volume = this.options.volume; // Create the event listeners // Only want the active entity to affect jPlayer and bubble events. // Using entity.gate so that object is referenced and gate property always current mediaElement.addEventListener("progress", function() { if(entity.gate && !self.status.waitForLoad) { self._getHtmlStatus(mediaElement); self._updateInterface(); self._trigger($.jPlayer.event.progress); } }, false); mediaElement.addEventListener("timeupdate", function() { if(entity.gate && !self.status.waitForLoad) { self._getHtmlStatus(mediaElement); self._updateInterface(); self._trigger($.jPlayer.event.timeupdate); } }, false); mediaElement.addEventListener("durationchange", function() { if(entity.gate && !self.status.waitForLoad) { self.status.duration = this.duration; self._getHtmlStatus(mediaElement); self._updateInterface(); self._trigger($.jPlayer.event.durationchange); } }, false); mediaElement.addEventListener("play", function() { if(entity.gate && !self.status.waitForLoad) { self._updateButtons(true); self._trigger($.jPlayer.event.play); } }, false); mediaElement.addEventListener("playing", function() { if(entity.gate && !self.status.waitForLoad) { self._updateButtons(true); self._seeked(); self._trigger($.jPlayer.event.playing); } }, false); mediaElement.addEventListener("pause", function() { if(entity.gate && !self.status.waitForLoad) { self._updateButtons(false); self._trigger($.jPlayer.event.pause); } }, false); mediaElement.addEventListener("waiting", function() { if(entity.gate && !self.status.waitForLoad) { self._seeking(); self._trigger($.jPlayer.event.waiting); } }, false); mediaElement.addEventListener("canplay", function() { if(entity.gate && !self.status.waitForLoad) { mediaElement.volume = self._volumeFix(self.options.volume); self._trigger($.jPlayer.event.canplay); } }, false); mediaElement.addEventListener("seeking", function() { if(entity.gate && !self.status.waitForLoad) { self._seeking(); self._trigger($.jPlayer.event.seeking); } }, false); mediaElement.addEventListener("seeked", function() { if(entity.gate && !self.status.waitForLoad) { self._seeked(); self._trigger($.jPlayer.event.seeked); } }, false); mediaElement.addEventListener("suspend", function() { // Seems to be the only way of capturing that the iOS4 browser did not actually play the media from the page code. ie., It needs a user gesture. if(entity.gate && !self.status.waitForLoad) { self._seeked(); self._trigger($.jPlayer.event.suspend); } }, false); mediaElement.addEventListener("ended", function() { if(entity.gate && !self.status.waitForLoad) { // Order of the next few commands are important. Change the time and then pause. // Solves a bug in Firefox, where issuing pause 1st causes the media to play from the start. ie., The pause is ignored. if(!$.jPlayer.browser.webkit) { // Chrome crashes if you do this in conjunction with a setMedia command in an ended event handler. ie., The playlist demo. self.htmlElement.media.currentTime = 0; // Safari does not care about this command. ie., It works with or without this line. (Both Safari and Chrome are Webkit.) } self.htmlElement.media.pause(); // Pause otherwise a click on the progress bar will play from that point, when it shouldn't, since it stopped playback. self._updateButtons(false); self._getHtmlStatus(mediaElement, true); // With override true. Otherwise Chrome leaves progress at full. self._updateInterface(); self._trigger($.jPlayer.event.ended); } }, false); mediaElement.addEventListener("error", function() { if(entity.gate && !self.status.waitForLoad) { self._updateButtons(false); self._seeked(); if(self.status.srcSet) { // Deals with case of clearMedia() causing an error event. clearTimeout(self.internal.htmlDlyCmdId); // Clears any delayed commands used in the HTML solution. self.status.waitForLoad = true; // Allows the load operation to try again. self.status.waitForPlay = true; // Reset since a play was captured. if(self.status.video) { self.internal.video.jq.css({'width':'0px', 'height':'0px'}); } if(self._validString(self.status.media.poster)) { self.internal.poster.jq.show(); } if(self.css.jq.videoPlay.length) { self.css.jq.videoPlay.show(); } self._error( { type: $.jPlayer.error.URL, context: self.status.src, // this.src shows absolute urls. Want context to show the url given. message: $.jPlayer.errorMsg.URL, hint: $.jPlayer.errorHint.URL }); } } }, false); // Create all the other event listeners that bubble up to a jPlayer event from html, without being used by jPlayer. $.each($.jPlayer.htmlEvent, function(i, eventType) { mediaElement.addEventListener(this, function() { if(entity.gate && !self.status.waitForLoad) { self._trigger($.jPlayer.event[eventType]); } }, false); }); }, _getHtmlStatus: function(media, override) { var ct = 0, d = 0, cpa = 0, sp = 0, cpr = 0; if(media.duration) { // Fixes the duration bug in iOS, where the durationchange event occurs when media.duration is not always correct. this.status.duration = media.duration; } ct = media.currentTime; cpa = (this.status.duration > 0) ? 100 * ct / this.status.duration : 0; if((typeof media.seekable === "object") && (media.seekable.length > 0)) { sp = (this.status.duration > 0) ? 100 * media.seekable.end(media.seekable.length-1) / this.status.duration : 100; cpr = 100 * media.currentTime / media.seekable.end(media.seekable.length-1); } else { sp = 100; cpr = cpa; } if(override) { ct = 0; cpr = 0; cpa = 0; } this.status.seekPercent = sp; this.status.currentPercentRelative = cpr; this.status.currentPercentAbsolute = cpa; this.status.currentTime = ct; this.status.readyState = media.readyState; this.status.networkState = media.networkState; this.status.playbackRate = media.playbackRate; this.status.ended = media.ended; }, _resetStatus: function() { this.status = $.extend({}, this.status, $.jPlayer.prototype.status); // Maintains the status properties that persist through a reset. }, _trigger: function(eventType, error, warning) { // eventType always valid as called using $.jPlayer.event.eventType var event = $.Event(eventType); event.jPlayer = {}; event.jPlayer.version = $.extend({}, this.version); event.jPlayer.options = $.extend(true, {}, this.options); // Deep copy event.jPlayer.status = $.extend(true, {}, this.status); // Deep copy event.jPlayer.html = $.extend(true, {}, this.html); // Deep copy event.jPlayer.flash = $.extend(true, {}, this.flash); // Deep copy if(error) { event.jPlayer.error = $.extend({}, error); } if(warning) { event.jPlayer.warning = $.extend({}, warning); } this.element.trigger(event); }, jPlayerFlashEvent: function(eventType, status) { // Called from Flash if(eventType === $.jPlayer.event.ready && !this.internal.ready) { this.internal.ready = true; this.internal.flash.jq.css({'width':'0px', 'height':'0px'}); // Once Flash generates the ready event, minimise to zero as it is not affected by wmode anymore. this.version.flash = status.version; if(this.version.needFlash !== this.version.flash) { this._error( { type: $.jPlayer.error.VERSION, context: this.version.flash, message: $.jPlayer.errorMsg.VERSION + this.version.flash, hint: $.jPlayer.errorHint.VERSION }); } this._trigger(eventType); } if(this.flash.gate) { switch(eventType) { case $.jPlayer.event.progress: this._getFlashStatus(status); this._updateInterface(); this._trigger(eventType); break; case $.jPlayer.event.timeupdate: this._getFlashStatus(status); this._updateInterface(); this._trigger(eventType); break; case $.jPlayer.event.play: this._seeked(); this._updateButtons(true); this._trigger(eventType); break; case $.jPlayer.event.pause: this._updateButtons(false); this._trigger(eventType); break; case $.jPlayer.event.ended: this._updateButtons(false); this._trigger(eventType); break; case $.jPlayer.event.error: this.status.waitForLoad = true; // Allows the load operation to try again. this.status.waitForPlay = true; // Reset since a play was captured. if(this.status.video) { this.internal.flash.jq.css({'width':'0px', 'height':'0px'}); } if(this._validString(this.status.media.poster)) { this.internal.poster.jq.show(); } if(this.css.jq.videoPlay.length) { this.css.jq.videoPlay.show(); } if(this.status.video) { // Set up for another try. Execute before error event. this._flash_setVideo(this.status.media); } else { this._flash_setAudio(this.status.media); } this._error( { type: $.jPlayer.error.URL, context:status.src, message: $.jPlayer.errorMsg.URL, hint: $.jPlayer.errorHint.URL }); break; case $.jPlayer.event.seeking: this._seeking(); this._trigger(eventType); break; case $.jPlayer.event.seeked: this._seeked(); this._trigger(eventType); break; case $.jPlayer.event.ready: // The ready event is handled outside the switch statement. // Captured here otherwise 2 ready events would be generated if the ready event handler used setMedia. break; default: this._trigger(eventType); } } return false; }, _getFlashStatus: function(status) { this.status.seekPercent = status.seekPercent; this.status.currentPercentRelative = status.currentPercentRelative; this.status.currentPercentAbsolute = status.currentPercentAbsolute; this.status.currentTime = status.currentTime; this.status.duration = status.duration; // The Flash does not generate this information in this release this.status.readyState = 4; // status.readyState; this.status.networkState = 0; // status.networkState; this.status.playbackRate = 1; // status.playbackRate; this.status.ended = false; // status.ended; }, _updateButtons: function(playing) { this.status.paused = !playing; if(this.css.jq.play.length && this.css.jq.pause.length) { if(playing) { this.css.jq.play.hide(); this.css.jq.pause.show(); } else { this.css.jq.play.show(); this.css.jq.pause.hide(); } } }, _updateInterface: function() { if(this.css.jq.seekBar.length) { this.css.jq.seekBar.width(this.status.seekPercent+"%"); } if(this.css.jq.playBar.length) { this.css.jq.playBar.width(this.status.currentPercentRelative+"%"); } if(this.css.jq.currentTime.length) { this.css.jq.currentTime.text($.jPlayer.convertTime(this.status.currentTime)); } if(this.css.jq.duration.length) { this.css.jq.duration.text($.jPlayer.convertTime(this.status.duration)); } }, _seeking: function() { if(this.css.jq.seekBar.length) { this.css.jq.seekBar.addClass("jp-seeking-bg"); } }, _seeked: function() { if(this.css.jq.seekBar.length) { this.css.jq.seekBar.removeClass("jp-seeking-bg"); } }, setMedia: function(media) { /* media[format] = String: URL of format. Must contain all of the supplied option's video or audio formats. * media.poster = String: Video poster URL. * media.subtitles = String: * NOT IMPLEMENTED * URL of subtitles SRT file * media.chapters = String: * NOT IMPLEMENTED * URL of chapters SRT file * media.stream = Boolean: * NOT IMPLEMENTED * Designating actual media streams. ie., "false/undefined" for files. Plan to refresh the flash every so often. */ var self = this; this._seeked(); clearTimeout(this.internal.htmlDlyCmdId); // Clears any delayed commands used in the HTML solution. // Store the current html gates, since we need for clearMedia() conditions. var audioGate = this.html.audio.gate; var videoGate = this.html.video.gate; var supported = false; $.each(this.formats, function(formatPriority, format) { var isVideo = self.format[format].media === 'video'; $.each(self.solutions, function(solutionPriority, solution) { if(self[solution].support[format] && self._validString(media[format])) { // Format supported in solution and url given for format. var isHtml = solution === 'html'; if(isVideo) { if(isHtml) { self.html.audio.gate = false; self.html.video.gate = true; self.flash.gate = false; } else { self.html.audio.gate = false; self.html.video.gate = false; self.flash.gate = true; } } else { if(isHtml) { self.html.audio.gate = true; self.html.video.gate = false; self.flash.gate = false; } else { self.html.audio.gate = false; self.html.video.gate = false; self.flash.gate = true; } } // Clear media of the previous solution if: // - it was Flash // - changing from HTML to Flash // - the HTML solution media type (audio or video) remained the same. // Note that, we must be careful with clearMedia() on iPhone, otherwise clearing the video when changing to audio corrupts the built in video player. if(self.flash.active || (self.html.active && self.flash.gate) || (audioGate === self.html.audio.gate && videoGate === self.html.video.gate)) { self.clearMedia(); } else if(audioGate !== self.html.audio.gate && videoGate !== self.html.video.gate) { // If switching between html elements self._html_pause(); // Hide the video if it was being used. if(self.status.video) { self.internal.video.jq.css({'width':'0px', 'height':'0px'}); } self._resetStatus(); // Since clearMedia usually does this. Execute after status.video useage. } if(isVideo) { if(isHtml) { self._html_setVideo(media); self.html.active = true; self.flash.active = false; } else { self._flash_setVideo(media); self.html.active = false; self.flash.active = true; } if(self.css.jq.videoPlay.length) { self.css.jq.videoPlay.show(); } self.status.video = true; } else { if(isHtml) { self._html_setAudio(media); self.html.active = true; self.flash.active = false; } else { self._flash_setAudio(media); self.html.active = false; self.flash.active = true; } if(self.css.jq.videoPlay.length) { self.css.jq.videoPlay.hide(); } self.status.video = false; } supported = true; return false; // Exit $.each } }); if(supported) { return false; // Exit $.each } }); if(supported) { // Set poster after the possible clearMedia() command above. IE had issues since the IMG onload event occurred immediately when cached. ie., The clearMedia() hide the poster. if(this._validString(media.poster)) { if(this.htmlElement.poster.src !== media.poster) { // Since some browsers do not generate img onload event. this.htmlElement.poster.src = media.poster; } else { this.internal.poster.jq.show(); } } else { this.internal.poster.jq.hide(); // Hide if not used, since clearMedia() does not always occur above. ie., HTML audio <-> video switching. } this.status.srcSet = true; this.status.media = $.extend({}, media); this._updateButtons(false); this._updateInterface(); } else { // jPlayer cannot support any formats provided in this browser // Pause here if old media could be playing. Otherwise, playing media being changed to bad media would leave the old media playing. if(this.status.srcSet && !this.status.waitForPlay) { this.pause(); } // Reset all the control flags this.html.audio.gate = false; this.html.video.gate = false; this.flash.gate = false; this.html.active = false; this.flash.active = false; // Reset status and interface. this._resetStatus(); this._updateInterface(); this._updateButtons(false); // Hide the any old media this.internal.poster.jq.hide(); if(this.html.used && this.require.video) { this.internal.video.jq.css({'width':'0px', 'height':'0px'}); } if(this.flash.used) { this.internal.flash.jq.css({'width':'0px', 'height':'0px'}); } // Send an error event this._error( { type: $.jPlayer.error.NO_SUPPORT, context: "{supplied:'" + this.options.supplied + "'}", message: $.jPlayer.errorMsg.NO_SUPPORT, hint: $.jPlayer.errorHint.NO_SUPPORT }); } }, clearMedia: function() { this._resetStatus(); this._updateButtons(false); this.internal.poster.jq.hide(); clearTimeout(this.internal.htmlDlyCmdId); if(this.html.active) { this._html_clearMedia(); } else if(this.flash.active) { this._flash_clearMedia(); } }, load: function() { if(this.status.srcSet) { if(this.html.active) { this._html_load(); } else if(this.flash.active) { this._flash_load(); } } else { this._urlNotSetError("load"); } }, play: function(time) { time = (typeof time === "number") ? time : NaN; // Remove jQuery event from click handler if(this.status.srcSet) { if(this.html.active) { this._html_play(time); } else if(this.flash.active) { this._flash_play(time); } } else { this._urlNotSetError("play"); } }, videoPlay: function(e) { // Handles clicks on the play button over the video poster this.play(); }, pause: function(time) { time = (typeof time === "number") ? time : NaN; // Remove jQuery event from click handler if(this.status.srcSet) { if(this.html.active) { this._html_pause(time); } else if(this.flash.active) { this._flash_pause(time); } } else { this._urlNotSetError("pause"); } }, pauseOthers: function() { var self = this; $.each(this.instances, function(i, element) { if(self.element !== element) { // Do not this instance. if(element.data("jPlayer").status.srcSet) { // Check that media is set otherwise would cause error event. element.jPlayer("pause"); } } }); }, stop: function() { if(this.status.srcSet) { if(this.html.active) { this._html_pause(0); } else if(this.flash.active) { this._flash_pause(0); } } else { this._urlNotSetError("stop"); } }, playHead: function(p) { p = this._limitValue(p, 0, 100); if(this.status.srcSet) { if(this.html.active) { this._html_playHead(p); } else if(this.flash.active) { this._flash_playHead(p); } } else { this._urlNotSetError("playHead"); } }, _muted: function(muted) { this.options.muted = muted; if(this.html.used) { this._html_mute(muted); } if(this.flash.used) { this._flash_mute(muted); } this._updateMute(muted); this._updateVolume(this.options.volume); this._trigger($.jPlayer.event.volumechange); }, mute: function(mute) { // mute is either: undefined (true), an event object (true) or a boolean (muted). mute = mute === undefined ? true : !!mute; this._muted(mute); }, unmute: function(unmute) { // unmute is either: undefined (true), an event object (true) or a boolean (!muted). unmute = unmute === undefined ? true : !!unmute; this._muted(!unmute); }, _updateMute: function(mute) { if(this.css.jq.mute.length && this.css.jq.unmute.length) { if(mute) { this.css.jq.mute.hide(); this.css.jq.unmute.show(); } else { this.css.jq.mute.show(); this.css.jq.unmute.hide(); } } }, volume: function(v) { v = this._limitValue(v, 0, 1); this.options.volume = v; if(this.html.used) { this._html_volume(v); } if(this.flash.used) { this._flash_volume(v); } this._updateVolume(v); this._trigger($.jPlayer.event.volumechange); }, volumeBar: function(e) { // Handles clicks on the volumeBar if(!this.options.muted && this.css.jq.volumeBar.length) { // Ignore clicks when muted var offset = this.css.jq.volumeBar.offset(); var x = e.pageX - offset.left; var w = this.css.jq.volumeBar.width(); var v = x/w; this.volume(v); } }, volumeBarValue: function(e) { // Handles clicks on the volumeBarValue this.volumeBar(e); }, _updateVolume: function(v) { v = this.options.muted ? 0 : v; if(this.css.jq.volumeBarValue.length) { this.css.jq.volumeBarValue.width((v*100)+"%"); } }, _volumeFix: function(v) { // Need to review if this is still necessary on latest Chrome var rnd = 0.001 * Math.random(); // Fix for Chrome 4: Fix volume being set multiple times before playing bug. var fix = (v < 0.5) ? rnd : -rnd; // Fix for Chrome 4: Solves volume change before play bug. (When new vol == old vol Chrome 4 does nothing!) return (v + fix); // Fix for Chrome 4: Event solves initial volume not being set correctly. }, _cssSelectorAncestor: function(ancestor) { var self = this; this.options.cssSelectorAncestor = ancestor; this._removeUiClass(); this.ancestorJq = ancestor ? $(ancestor) : []; // Would use $() instead of [], but it is only 1.4+ if(ancestor && this.ancestorJq.length !== 1) { // So empty strings do not generate the warning. this._warning( { type: $.jPlayer.warning.CSS_SELECTOR_COUNT, context: ancestor, message: $.jPlayer.warningMsg.CSS_SELECTOR_COUNT + this.ancestorJq.length + " found for cssSelectorAncestor.", hint: $.jPlayer.warningHint.CSS_SELECTOR_COUNT }); } this._addUiClass(); $.each(this.options.cssSelector, function(fn, cssSel) { self._cssSelector(fn, cssSel); }); }, _cssSelector: function(fn, cssSel) { var self = this; if(typeof cssSel === 'string') { if($.jPlayer.prototype.options.cssSelector[fn]) { if(this.css.jq[fn] && this.css.jq[fn].length) { this.css.jq[fn].unbind(".jPlayer"); } this.options.cssSelector[fn] = cssSel; this.css.cs[fn] = this.options.cssSelectorAncestor + " " + cssSel; if(cssSel) { // Checks for empty string this.css.jq[fn] = $(this.css.cs[fn]); } else { this.css.jq[fn] = []; // To comply with the css.jq[fn].length check before its use. As of jQuery 1.4 could have used $() for an empty set. } if(this.css.jq[fn].length) { var handler = function(e) { self[fn](e); $(this).blur(); return false; }; this.css.jq[fn].bind("click.jPlayer", handler); // Using jPlayer namespace } if(cssSel && this.css.jq[fn].length !== 1) { // So empty strings do not generate the warning. ie., they just remove the old one. this._warning( { type: $.jPlayer.warning.CSS_SELECTOR_COUNT, context: this.css.cs[fn], message: $.jPlayer.warningMsg.CSS_SELECTOR_COUNT + this.css.jq[fn].length + " found for " + fn + " method.", hint: $.jPlayer.warningHint.CSS_SELECTOR_COUNT }); } } else { this._warning( { type: $.jPlayer.warning.CSS_SELECTOR_METHOD, context: fn, message: $.jPlayer.warningMsg.CSS_SELECTOR_METHOD, hint: $.jPlayer.warningHint.CSS_SELECTOR_METHOD }); } } else { this._warning( { type: $.jPlayer.warning.CSS_SELECTOR_STRING, context: cssSel, message: $.jPlayer.warningMsg.CSS_SELECTOR_STRING, hint: $.jPlayer.warningHint.CSS_SELECTOR_STRING }); } }, seekBar: function(e) { // Handles clicks on the seekBar if(this.css.jq.seekBar) { var offset = this.css.jq.seekBar.offset(); var x = e.pageX - offset.left; var w = this.css.jq.seekBar.width(); var p = 100*x/w; this.playHead(p); } }, playBar: function(e) { // Handles clicks on the playBar this.seekBar(e); }, currentTime: function(e) { // Handles clicks on the text // Added to avoid errors using cssSelector system for the text }, duration: function(e) { // Handles clicks on the text // Added to avoid errors using cssSelector system for the text }, // Options code adapted from ui.widget.js (1.8.7). Made changes so the key can use dot notation. To match previous getData solution in jPlayer 1. option: function(key, value) { var options = key; // Enables use: options(). Returns a copy of options object if ( arguments.length === 0 ) { return $.extend( true, {}, this.options ); } if(typeof key === "string") { var keys = key.split("."); // Enables use: options("someOption") Returns a copy of the option. Supports dot notation. if(value === undefined) { var opt = $.extend(true, {}, this.options); for(var i = 0; i < keys.length; i++) { if(opt[keys[i]] !== undefined) { opt = opt[keys[i]]; } else { this._warning( { type: $.jPlayer.warning.OPTION_KEY, context: key, message: $.jPlayer.warningMsg.OPTION_KEY, hint: $.jPlayer.warningHint.OPTION_KEY }); return undefined; } } return opt; } // Enables use: options("someOptionObject", someObject}). Creates: {someOptionObject:someObject} // Enables use: options("someOption", someValue). Creates: {someOption:someValue} // Enables use: options("someOptionObject.someOption", someValue). Creates: {someOptionObject:{someOption:someValue}} options = {}; var opts = options; for(var j = 0; j < keys.length; j++) { if(j < keys.length - 1) { opts[keys[j]] = {}; opts = opts[keys[j]]; } else { opts[keys[j]] = value; } } } // Otherwise enables use: options(optionObject). Uses original object (the key) this._setOptions(options); return this; }, _setOptions: function(options) { var self = this; $.each(options, function(key, value) { // This supports the 2 level depth that the options of jPlayer has. Would review if we ever need more depth. self._setOption(key, value); }); return this; }, _setOption: function(key, value) { var self = this; // The ability to set options is limited at this time. switch(key) { case "volume" : this.volume(value); break; case "muted" : this._muted(value); break; case "cssSelectorAncestor" : this._cssSelectorAncestor(value); // Set and refresh all associations for the new ancestor. break; case "cssSelector" : $.each(value, function(fn, cssSel) { self._cssSelector(fn, cssSel); // NB: The option is set inside this function, after further validity checks. }); break; case "fullScreen" : if(this.options[key] !== value) { // if changed this._removeUiClass(); this.options[key] = value; this._refreshSize(); } break; case "size" : if(!this.options.fullScreen && this.options[key].cssClass !== value.cssClass) { this._removeUiClass(); } this.options[key] = $.extend({}, this.options[key], value); // store a merged copy of it, incase not all properties changed. this._refreshSize(); break; case "sizeFull" : if(this.options.fullScreen && this.options[key].cssClass !== value.cssClass) { this._removeUiClass(); } this.options[key] = $.extend({}, this.options[key], value); // store a merged copy of it, incase not all properties changed. this._refreshSize(); break; case "emulateHtml" : if(this.options[key] !== value) { // To avoid multiple event handlers being created, if true already. this.options[key] = value; if(value) { this._emulateHtmlBridge(); } else { this._destroyHtmlBridge(); } } break; } return this; }, // End of: (Options code adapted from ui.widget.js) _refreshSize: function() { this._setSize(); // update status and jPlayer element size this._addUiClass(); // update the ui class this._updateSize(); // update internal sizes }, _setSize: function() { // Determine the current size from the options if(this.options.fullScreen) { this.status.width = this.options.sizeFull.width; this.status.height = this.options.sizeFull.height; this.status.cssClass = this.options.sizeFull.cssClass; } else { this.status.width = this.options.size.width; this.status.height = this.options.size.height; this.status.cssClass = this.options.size.cssClass; } // Set the size of the jPlayer area. this.element.css({'width': this.status.width, 'height': this.status.height}); }, _addUiClass: function() { if(this.ancestorJq.length) { this.ancestorJq.addClass(this.status.cssClass); } }, _removeUiClass: function() { if(this.ancestorJq.length) { this.ancestorJq.removeClass(this.status.cssClass); } }, _updateSize: function() { // The poster uses show/hide so can simply resize it. this.internal.poster.jq.css({'width': this.status.width, 'height': this.status.height}); // Video html or flash resized if necessary at this time. if(!this.status.waitForPlay) { if(this.html.active && this.status.video) { // Only if video media this.internal.video.jq.css({'width': this.status.width, 'height': this.status.height}); } else if(this.flash.active) { this.internal.flash.jq.css({'width': this.status.width, 'height': this.status.height}); } } }, fullScreen: function() { this._setOption("fullScreen", true); }, restoreScreen: function() { this._setOption("fullScreen", false); }, _html_initMedia: function() { if(this.status.srcSet && !this.status.waitForPlay) { this.htmlElement.media.pause(); } if(this.options.preload !== 'none') { this._html_load(); } this._trigger($.jPlayer.event.timeupdate); // The flash generates this event for its solution. }, _html_setAudio: function(media) { var self = this; // Always finds a format due to checks in setMedia() $.each(this.formats, function(priority, format) { if(self.html.support[format] && media[format]) { self.status.src = media[format]; self.status.format[format] = true; self.status.formatType = format; return false; } }); this.htmlElement.media = this.htmlElement.audio; this._html_initMedia(); }, _html_setVideo: function(media) { var self = this; // Always finds a format due to checks in setMedia() $.each(this.formats, function(priority, format) { if(self.html.support[format] && media[format]) { self.status.src = media[format]; self.status.format[format] = true; self.status.formatType = format; return false; } }); this.htmlElement.media = this.htmlElement.video; this._html_initMedia(); }, _html_clearMedia: function() { if(this.htmlElement.media) { if(this.htmlElement.media.id === this.internal.video.id) { this.internal.video.jq.css({'width':'0px', 'height':'0px'}); } this.htmlElement.media.pause(); this.htmlElement.media.src = ""; this.htmlElement.media.load(); // Stops an old, "in progress" download from continuing the download. Triggers the loadstart, error and emptied events, due to the empty src. Also an abort event if a download was in progress. } }, _html_load: function() { if(this.status.waitForLoad) { this.status.waitForLoad = false; this.htmlElement.media.src = this.status.src; this.htmlElement.media.load(); } clearTimeout(this.internal.htmlDlyCmdId); }, _html_play: function(time) { var self = this; this._html_load(); // Loads if required and clears any delayed commands. this.htmlElement.media.play(); // Before currentTime attempt otherwise Firefox 4 Beta never loads. if(!isNaN(time)) { try { this.htmlElement.media.currentTime = time; } catch(err) { this.internal.htmlDlyCmdId = setTimeout(function() { self.play(time); }, 100); return; // Cancel execution and wait for the delayed command. } } this._html_checkWaitForPlay(); }, _html_pause: function(time) { var self = this; if(time > 0) { // We do not want the stop() command, which does pause(0), causing a load operation. this._html_load(); // Loads if required and clears any delayed commands. } else { clearTimeout(this.internal.htmlDlyCmdId); } // Order of these commands is important for Safari (Win) and IE9. Pause then change currentTime. this.htmlElement.media.pause(); if(!isNaN(time)) { try { this.htmlElement.media.currentTime = time; } catch(err) { this.internal.htmlDlyCmdId = setTimeout(function() { self.pause(time); }, 100); return; // Cancel execution and wait for the delayed command. } } if(time > 0) { // Avoids a setMedia() followed by stop() or pause(0) hiding the video play button. this._html_checkWaitForPlay(); } }, _html_playHead: function(percent) { var self = this; this._html_load(); // Loads if required and clears any delayed commands. try { if((typeof this.htmlElement.media.seekable === "object") && (this.htmlElement.media.seekable.length > 0)) { this.htmlElement.media.currentTime = percent * this.htmlElement.media.seekable.end(this.htmlElement.media.seekable.length-1) / 100; } else if(this.htmlElement.media.duration > 0 && !isNaN(this.htmlElement.media.duration)) { this.htmlElement.media.currentTime = percent * this.htmlElement.media.duration / 100; } else { throw "e"; } } catch(err) { this.internal.htmlDlyCmdId = setTimeout(function() { self.playHead(percent); }, 100); return; // Cancel execution and wait for the delayed command. } if(!this.status.waitForLoad) { this._html_checkWaitForPlay(); } }, _html_checkWaitForPlay: function() { if(this.status.waitForPlay) { this.status.waitForPlay = false; if(this.css.jq.videoPlay.length) { this.css.jq.videoPlay.hide(); } if(this.status.video) { this.internal.poster.jq.hide(); this.internal.video.jq.css({'width': this.status.width, 'height': this.status.height}); } } }, _html_volume: function(v) { if(this.html.audio.available) { this.htmlElement.audio.volume = v; } if(this.html.video.available) { this.htmlElement.video.volume = v; } }, _html_mute: function(m) { if(this.html.audio.available) { this.htmlElement.audio.muted = m; } if(this.html.video.available) { this.htmlElement.video.muted = m; } }, _flash_setAudio: function(media) { var self = this; try { // Always finds a format due to checks in setMedia() $.each(this.formats, function(priority, format) { if(self.flash.support[format] && media[format]) { switch (format) { case "m4a" : case "fla" : self._getMovie().fl_setAudio_m4a(media[format]); break; case "mp3" : self._getMovie().fl_setAudio_mp3(media[format]); break; } self.status.src = media[format]; self.status.format[format] = true; self.status.formatType = format; return false; } }); if(this.options.preload === 'auto') { this._flash_load(); this.status.waitForLoad = false; } } catch(err) { this._flashError(err); } }, _flash_setVideo: function(media) { var self = this; try { // Always finds a format due to checks in setMedia() $.each(this.formats, function(priority, format) { if(self.flash.support[format] && media[format]) { switch (format) { case "m4v" : case "flv" : self._getMovie().fl_setVideo_m4v(media[format]); break; } self.status.src = media[format]; self.status.format[format] = true; self.status.formatType = format; return false; } }); if(this.options.preload === 'auto') { this._flash_load(); this.status.waitForLoad = false; } } catch(err) { this._flashError(err); } }, _flash_clearMedia: function() { this.internal.flash.jq.css({'width':'0px', 'height':'0px'}); // Must do via CSS as setting attr() to zero causes a jQuery error in IE. try { this._getMovie().fl_clearMedia(); } catch(err) { this._flashError(err); } }, _flash_load: function() { try { this._getMovie().fl_load(); } catch(err) { this._flashError(err); } this.status.waitForLoad = false; }, _flash_play: function(time) { try { this._getMovie().fl_play(time); } catch(err) { this._flashError(err); } this.status.waitForLoad = false; this._flash_checkWaitForPlay(); }, _flash_pause: function(time) { try { this._getMovie().fl_pause(time); } catch(err) { this._flashError(err); } if(time > 0) { // Avoids a setMedia() followed by stop() or pause(0) hiding the video play button. this.status.waitForLoad = false; this._flash_checkWaitForPlay(); } }, _flash_playHead: function(p) { try { this._getMovie().fl_play_head(p); } catch(err) { this._flashError(err); } if(!this.status.waitForLoad) { this._flash_checkWaitForPlay(); } }, _flash_checkWaitForPlay: function() { if(this.status.waitForPlay) { this.status.waitForPlay = false; if(this.css.jq.videoPlay.length) { this.css.jq.videoPlay.hide(); } if(this.status.video) { this.internal.poster.jq.hide(); this.internal.flash.jq.css({'width': this.status.width, 'height': this.status.height}); } } }, _flash_volume: function(v) { try { this._getMovie().fl_volume(v); } catch(err) { this._flashError(err); } }, _flash_mute: function(m) { try { this._getMovie().fl_mute(m); } catch(err) { this._flashError(err); } }, _getMovie: function() { return document[this.internal.flash.id]; }, _checkForFlash: function (version) { // Function checkForFlash adapted from FlashReplace by Robert Nyman // http://code.google.com/p/flashreplace/ var flashIsInstalled = false; var flash; if(window.ActiveXObject){ try{ flash = new ActiveXObject(("ShockwaveFlash.ShockwaveFlash." + version)); flashIsInstalled = true; } catch(e){ // Throws an error if the version isn't available } } else if(navigator.plugins && navigator.mimeTypes.length > 0){ flash = navigator.plugins["Shockwave Flash"]; if(flash){ var flashVersion = navigator.plugins["Shockwave Flash"].description.replace(/.*\s(\d+\.\d+).*/, "$1"); if(flashVersion >= version){ flashIsInstalled = true; } } } return flashIsInstalled; }, _validString: function(url) { return (url && typeof url === "string"); // Empty strings return false }, _limitValue: function(value, min, max) { return (value < min) ? min : ((value > max) ? max : value); }, _urlNotSetError: function(context) { this._error( { type: $.jPlayer.error.URL_NOT_SET, context: context, message: $.jPlayer.errorMsg.URL_NOT_SET, hint: $.jPlayer.errorHint.URL_NOT_SET }); }, _flashError: function(error) { this._error( { type: $.jPlayer.error.FLASH, context: this.internal.flash.swf, message: $.jPlayer.errorMsg.FLASH + error.message, hint: $.jPlayer.errorHint.FLASH }); }, _error: function(error) { this._trigger($.jPlayer.event.error, error); if(this.options.errorAlerts) { this._alert("Error!" + (error.message ? "\n\n" + error.message : "") + (error.hint ? "\n\n" + error.hint : "") + "\n\nContext: " + error.context); } }, _warning: function(warning) { this._trigger($.jPlayer.event.warning, undefined, warning); if(this.options.warningAlerts) { this._alert("Warning!" + (warning.message ? "\n\n" + warning.message : "") + (warning.hint ? "\n\n" + warning.hint : "") + "\n\nContext: " + warning.context); } }, _alert: function(message) { alert("jPlayer " + this.version.script + " : id='" + this.internal.self.id +"' : " + message); }, _emulateHtmlBridge: function() { var self = this, methods = $.jPlayer.emulateMethods; // Emulate methods on jPlayer's DOM element. $.each( $.jPlayer.emulateMethods.split(/\s+/g), function(i, name) { self.internal.domNode[name] = function(arg) { self[name](arg); }; }); // Bubble jPlayer events to its DOM element. $.each($.jPlayer.event, function(eventName,eventType) { var nativeEvent = true; $.each( $.jPlayer.reservedEvent.split(/\s+/g), function(i, name) { if(name === eventName) { nativeEvent = false; return false; } }); if(nativeEvent) { self.element.bind(eventType + ".jPlayer.jPlayerHtml", function() { // With .jPlayer & .jPlayerHtml namespaces. self._emulateHtmlUpdate(); var domEvent = document.createEvent("Event"); domEvent.initEvent(eventName, false, true); self.internal.domNode.dispatchEvent(domEvent); }); } // The error event would require a special case }); // IE9 has a readyState property on all elements. The document should have it, but all (except media) elements inherit it in IE9. This conflicts with Popcorn, which polls the readyState. }, _emulateHtmlUpdate: function() { var self = this; $.each( $.jPlayer.emulateStatus.split(/\s+/g), function(i, name) { self.internal.domNode[name] = self.status[name]; }); $.each( $.jPlayer.emulateOptions.split(/\s+/g), function(i, name) { self.internal.domNode[name] = self.options[name]; }); }, _destroyHtmlBridge: function() { var self = this; // Bridge event handlers are also removed by destroy() through .jPlayer namespace. this.element.unbind(".jPlayerHtml"); // Remove all event handlers created by the jPlayer bridge. So you can change the emulateHtml option. // Remove the methods and properties var emulated = $.jPlayer.emulateMethods + " " + $.jPlayer.emulateStatus + " " + $.jPlayer.emulateOptions; $.each( emulated.split(/\s+/g), function(i, name) { delete self.internal.domNode[name]; }); } }; $.jPlayer.error = { FLASH: "e_flash", NO_SOLUTION: "e_no_solution", NO_SUPPORT: "e_no_support", URL: "e_url", URL_NOT_SET: "e_url_not_set", VERSION: "e_version" }; $.jPlayer.errorMsg = { FLASH: "jPlayer's Flash fallback is not configured correctly, or a command was issued before the jPlayer Ready event. Details: ", // Used in: _flashError() NO_SOLUTION: "No solution can be found by jPlayer in this browser. Neither HTML nor Flash can be used.", // Used in: _init() NO_SUPPORT: "It is not possible to play any media format provided in setMedia() on this browser using your current options.", // Used in: setMedia() URL: "Media URL could not be loaded.", // Used in: jPlayerFlashEvent() and _addHtmlEventListeners() URL_NOT_SET: "Attempt to issue media playback commands, while no media url is set.", // Used in: load(), play(), pause(), stop() and playHead() VERSION: "jPlayer " + $.jPlayer.prototype.version.script + " needs Jplayer.swf version " + $.jPlayer.prototype.version.needFlash + " but found " // Used in: jPlayerReady() }; $.jPlayer.errorHint = { FLASH: "Check your swfPath option and that Jplayer.swf is there.", NO_SOLUTION: "Review the jPlayer options: support and supplied.", NO_SUPPORT: "Video or audio formats defined in the supplied option are missing.", URL: "Check media URL is valid.", URL_NOT_SET: "Use setMedia() to set the media URL.", VERSION: "Update jPlayer files." }; $.jPlayer.warning = { CSS_SELECTOR_COUNT: "e_css_selector_count", CSS_SELECTOR_METHOD: "e_css_selector_method", CSS_SELECTOR_STRING: "e_css_selector_string", OPTION_KEY: "e_option_key" }; $.jPlayer.warningMsg = { CSS_SELECTOR_COUNT: "The number of css selectors found did not equal one: ", CSS_SELECTOR_METHOD: "The methodName given in jPlayer('cssSelector') is not a valid jPlayer method.", CSS_SELECTOR_STRING: "The methodCssSelector given in jPlayer('cssSelector') is not a String or is empty.", OPTION_KEY: "The option requested in jPlayer('option') is undefined." }; $.jPlayer.warningHint = { CSS_SELECTOR_COUNT: "Check your css selector and the ancestor.", CSS_SELECTOR_METHOD: "Check your method name.", CSS_SELECTOR_STRING: "Check your css selector is a string.", OPTION_KEY: "Check your option name." }; })(jQuery);