Created
July 28, 2011 20:35
-
-
Save cadecairos/1112489 to your computer and use it in GitHub Desktop.
popcorn complete
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
/* | |
* popcorn.js version abd0127 | |
* http://popcornjs.org | |
* | |
* Copyright 2011, Mozilla Foundation | |
* Licensed under the MIT license | |
*/ | |
(function(global, document) { | |
// Cache refs to speed up calls to native utils | |
var | |
AP = Array.prototype, | |
OP = Object.prototype, | |
forEach = AP.forEach, | |
slice = AP.slice, | |
hasOwn = OP.hasOwnProperty, | |
toString = OP.toString, | |
// ID string matching | |
rIdExp = /^(#([\w\-\_\.]+))$/, | |
// Ready fn cache | |
readyStack = [], | |
readyBound = false, | |
readyFired = false, | |
// Non-public internal data object | |
internal = { | |
events: { | |
hash: {}, | |
apis: {} | |
} | |
}, | |
// Declare constructor | |
// Returns an instance object. | |
Popcorn = function( entity, options ) { | |
// Return new Popcorn object | |
return new Popcorn.p.init( entity, options || null ); | |
}; | |
// Instance caching | |
Popcorn.instances = []; | |
Popcorn.instanceIds = {}; | |
Popcorn.removeInstance = function( instance ) { | |
// If called prior to any instances being created | |
// Return early to avoid splicing on nothing | |
if ( !Popcorn.instances.length ) { | |
return; | |
} | |
// Remove instance from Popcorn.instances | |
Popcorn.instances.splice( Popcorn.instanceIds[ instance.id ], 1 ); | |
// Delete the instance id key | |
delete Popcorn.instanceIds[ instance.id ]; | |
// Return current modified instances | |
return Popcorn.instances; | |
}; | |
// Addes a Popcorn instance to the Popcorn instance array | |
Popcorn.addInstance = function( instance ) { | |
var instanceLen = Popcorn.instances.length, | |
instanceId = instance.media.id && instance.media.id; | |
// If the media element has its own `id` use it, otherwise provide one | |
// Ensure that instances have unique ids and unique entries | |
// Uses `in` operator to avoid false positives on 0 | |
instance.id = !( instanceId in Popcorn.instanceIds ) && instanceId || | |
"__popcorn" + instanceLen; | |
// Create a reference entry for this instance | |
Popcorn.instanceIds[ instance.id ] = instanceLen; | |
// Add this instance to the cache | |
Popcorn.instances.push( instance ); | |
// Return the current modified instances | |
return Popcorn.instances; | |
}; | |
// Request Popcorn object instance by id | |
Popcorn.getInstanceById = function( id ) { | |
return Popcorn.instances[ Popcorn.instanceIds[ id ] ]; | |
}; | |
// Remove Popcorn object instance by id | |
Popcorn.removeInstanceById = function( id ) { | |
return Popcorn.removeInstance( Popcorn.instances[ Popcorn.instanceIds[ id ] ] ); | |
}; | |
// Declare a shortcut (Popcorn.p) to and a definition of | |
// the new prototype for our Popcorn constructor | |
Popcorn.p = Popcorn.prototype = { | |
init: function( entity, options ) { | |
var matches; | |
// Supports Popcorn(function () { /../ }) | |
// Originally proposed by Daniel Brooks | |
if ( typeof entity === "function" ) { | |
// If document ready has already fired | |
if ( document.readyState === "interactive" || document.readyState === "complete" ) { | |
entity( document, Popcorn ); | |
return; | |
} | |
// Add `entity` fn to ready stack | |
readyStack.push( entity ); | |
// This process should happen once per page load | |
if ( !readyBound ) { | |
// set readyBound flag | |
readyBound = true; | |
var DOMContentLoaded = function() { | |
readyFired = true; | |
// Remove global DOM ready listener | |
document.removeEventListener( "DOMContentLoaded", DOMContentLoaded, false ); | |
// Execute all ready function in the stack | |
for ( var i = 0, readyStackLength = readyStack.length; i < readyStackLength; i++ ) { | |
readyStack[ i ].call( document, Popcorn ); | |
} | |
// GC readyStack | |
readyStack = null; | |
}; | |
// Register global DOM ready listener | |
document.addEventListener( "DOMContentLoaded", DOMContentLoaded, false ); | |
} | |
return; | |
} | |
// Check if entity is a valid string id | |
matches = rIdExp.exec( entity ); | |
// Get media element by id or object reference | |
this.media = matches && matches.length && matches[ 2 ] ? | |
document.getElementById( matches[ 2 ] ) : | |
entity; | |
// Create an audio or video element property reference | |
this[ ( this.media.nodeName && this.media.nodeName.toLowerCase() ) || "video" ] = this.media; | |
// Register new instance | |
Popcorn.addInstance( this ); | |
this.options = options || {}; | |
this.data = { | |
// Allows disabling a plugin per instance | |
disabled: [], | |
// Stores DOM event queues by type | |
events: {}, | |
// Stores Special event hooks data | |
hooks: {}, | |
// Store track event history data | |
history: [], | |
// Store track event object references by trackId | |
trackRefs: {}, | |
// Playback track event queues | |
trackEvents: { | |
byStart: [{ | |
start: -1, | |
end: -1 | |
}], | |
byEnd: [{ | |
start: -1, | |
end: -1 | |
}], | |
startIndex: 0, | |
endIndex: 0, | |
previousUpdateTime: -1 | |
} | |
}; | |
// Wrap true ready check | |
var isReady = function( that ) { | |
if ( that.media.readyState >= 2 ) { | |
// Adding padding to the front and end of the arrays | |
// this is so we do not fall off either end | |
var duration = that.media.duration, | |
// Check for no duration info (NaN) | |
videoDurationPlus = duration != duration ? Number.MAX_VALUE : duration + 1; | |
Popcorn.addTrackEvent( that, { | |
start: videoDurationPlus, | |
end: videoDurationPlus | |
}); | |
that.media.addEventListener( "timeupdate", function( event ) { | |
var currentTime = this.currentTime, | |
previousTime = that.data.trackEvents.previousUpdateTime, | |
tracks = that.data.trackEvents, | |
tracksByEnd = tracks.byEnd, | |
tracksByStart = tracks.byStart; | |
// Playbar advancing | |
if ( previousTime < currentTime ) { | |
while ( tracksByEnd[ tracks.endIndex ] && tracksByEnd[ tracks.endIndex ].end <= currentTime ) { | |
// If plugin does not exist on this instance, remove it | |
if ( !tracksByEnd[ tracks.endIndex ]._natives || !!that[ tracksByEnd[ tracks.endIndex ]._natives.type ] ) { | |
if ( tracksByEnd[ tracks.endIndex ]._running === true ) { | |
tracksByEnd[ tracks.endIndex ]._running = false; | |
tracksByEnd[ tracks.endIndex ]._natives.end.call( that, event, tracksByEnd[ tracks.endIndex ] ); | |
} | |
tracks.endIndex++; | |
} else { | |
// remove track event | |
Popcorn.removeTrackEvent( that, tracksByEnd[ tracks.endIndex ]._id ); | |
return; | |
} | |
} | |
while ( tracksByStart[ tracks.startIndex ] && tracksByStart[ tracks.startIndex ].start <= currentTime ) { | |
// If plugin does not exist on this instance, remove it | |
if ( !tracksByStart[ tracks.startIndex ]._natives || !!that[ tracksByStart[ tracks.startIndex ]._natives.type ] ) { | |
if ( tracksByStart[ tracks.startIndex ].end > currentTime && | |
tracksByStart[ tracks.startIndex ]._running === false && | |
that.data.disabled.indexOf( tracksByStart[ tracks.startIndex ]._natives.type ) === -1 ) { | |
tracksByStart[ tracks.startIndex ]._running = true; | |
tracksByStart[ tracks.startIndex ]._natives.start.call( that, event, tracksByStart[ tracks.startIndex ] ); | |
} | |
tracks.startIndex++; | |
} else { | |
// remove track event | |
Popcorn.removeTrackEvent( that, tracksByStart[ tracks.startIndex ]._id ); | |
return; | |
} | |
} | |
// Playbar receding | |
} else if ( previousTime > currentTime ) { | |
while ( tracksByStart[ tracks.startIndex ] && tracksByStart[ tracks.startIndex ].start > currentTime ) { | |
// if plugin does not exist on this instance, remove it | |
if ( !tracksByStart[ tracks.startIndex ]._natives || !!that[ tracksByStart[ tracks.startIndex ]._natives.type ] ) { | |
if ( tracksByStart[ tracks.startIndex ]._running === true ) { | |
tracksByStart[ tracks.startIndex ]._running = false; | |
tracksByStart[ tracks.startIndex ]._natives.end.call( that, event, tracksByStart[ tracks.startIndex ] ); | |
} | |
tracks.startIndex--; | |
} else { | |
// remove track event | |
Popcorn.removeTrackEvent( that, tracksByStart[ tracks.startIndex ]._id ); | |
return; | |
} | |
} | |
while ( tracksByEnd[ tracks.endIndex ] && tracksByEnd[ tracks.endIndex ].end > currentTime ) { | |
// if plugin does not exist on this instance, remove it | |
if ( !tracksByEnd[ tracks.endIndex ]._natives || !!that[ tracksByEnd[ tracks.endIndex ]._natives.type ] ) { | |
if ( tracksByEnd[ tracks.endIndex ].start <= currentTime && | |
tracksByEnd[ tracks.endIndex ]._running === false && | |
that.data.disabled.indexOf( tracksByEnd[ tracks.endIndex ]._natives.type ) === -1 ) { | |
tracksByEnd[ tracks.endIndex ]._running = true; | |
tracksByEnd[ tracks.endIndex ]._natives.start.call( that, event, tracksByEnd[tracks.endIndex] ); | |
} | |
tracks.endIndex--; | |
} else { | |
// remove track event | |
Popcorn.removeTrackEvent( that, tracksByEnd[ tracks.endIndex ]._id ); | |
return; | |
} | |
} | |
} | |
tracks.previousUpdateTime = currentTime; | |
}, false ); | |
} else { | |
global.setTimeout(function() { | |
isReady( that ); | |
}, 1 ); | |
} | |
}; | |
isReady( this ); | |
return this; | |
} | |
}; | |
// Extend constructor prototype to instance prototype | |
// Allows chaining methods to instances | |
Popcorn.p.init.prototype = Popcorn.p; | |
Popcorn.forEach = function( obj, fn, context ) { | |
if ( !obj || !fn ) { | |
return {}; | |
} | |
context = context || this; | |
var key, len; | |
// Use native whenever possible | |
if ( forEach && obj.forEach === forEach ) { | |
return obj.forEach( fn, context ); | |
} | |
if ( toString.call( obj ) === "[object NodeList]" ) { | |
for ( key = 0, len = obj.length; key < len; key++ ) { | |
fn.call( context, obj[ key ], key, obj ); | |
} | |
return obj; | |
} | |
for ( key in obj ) { | |
if ( hasOwn.call( obj, key ) ) { | |
fn.call( context, obj[ key ], key, obj ); | |
} | |
} | |
return obj; | |
}; | |
Popcorn.extend = function( obj ) { | |
var dest = obj, src = slice.call( arguments, 1 ); | |
Popcorn.forEach( src, function( copy ) { | |
for ( var prop in copy ) { | |
dest[ prop ] = copy[ prop ]; | |
} | |
}); | |
return dest; | |
}; | |
// A Few reusable utils, memoized onto Popcorn | |
Popcorn.extend( Popcorn, { | |
error: function( msg ) { | |
throw new Error( msg ); | |
}, | |
guid: function( prefix ) { | |
Popcorn.guid.counter++; | |
return ( prefix ? prefix : "" ) + ( +new Date() + Popcorn.guid.counter ); | |
}, | |
sizeOf: function( obj ) { | |
var size = 0; | |
for ( var prop in obj ) { | |
size++; | |
} | |
return size; | |
}, | |
isArray: Array.isArray || function( array ) { | |
return toString.call( array ) === "[object Array]"; | |
}, | |
nop: function() {}, | |
position: function( elem ) { | |
var clientRect = elem.getBoundingClientRect(), | |
bounds = {}, | |
doc = elem.ownerDocument, | |
docElem = document.documentElement, | |
body = document.body, | |
clientTop, clientLeft, scrollTop, scrollLeft, top, left; | |
// Determine correct clientTop/Left | |
clientTop = docElem.clientTop || body.clientTop || 0; | |
clientLeft = docElem.clientLeft || body.clientLeft || 0; | |
// Determine correct scrollTop/Left | |
scrollTop = ( global.pageYOffset && docElem.scrollTop || body.scrollTop ); | |
scrollLeft = ( global.pageXOffset && docElem.scrollLeft || body.scrollLeft ); | |
// Temp top/left | |
top = Math.ceil( clientRect.top + scrollTop - clientTop ); | |
left = Math.ceil( clientRect.left + scrollLeft - clientLeft ); | |
for ( var p in clientRect ) { | |
bounds[ p ] = Math.round( clientRect[ p ] ); | |
} | |
return Popcorn.extend({}, bounds, { top: top, left: left }); | |
}, | |
disable: function( instance, plugin ) { | |
var disabled = instance.data.disabled; | |
if ( disabled.indexOf( plugin ) === -1 ) { | |
disabled.push( plugin ); | |
} | |
return instance; | |
}, | |
enable: function( instance, plugin ) { | |
var disabled = instance.data.disabled, | |
index = disabled.indexOf( plugin ); | |
if ( index > -1 ) { | |
disabled.splice( index, 1 ); | |
} | |
return instance; | |
} | |
}); | |
// Memoized GUID Counter | |
Popcorn.guid.counter = 1; | |
// Factory to implement getters, setters and controllers | |
// as Popcorn instance methods. The IIFE will create and return | |
// an object with defined methods | |
Popcorn.extend(Popcorn.p, (function() { | |
var methods = "load play pause currentTime playbackRate mute volume duration", | |
ret = {}; | |
// Build methods, store in object that is returned and passed to extend | |
Popcorn.forEach( methods.split( /\s+/g ), function( name ) { | |
ret[ name ] = function( arg ) { | |
if ( typeof this.media[ name ] === "function" ) { | |
this.media[ name ](); | |
return this; | |
} | |
if ( arg !== false && arg !== null && typeof arg !== "undefined" ) { | |
this.media[ name ] = arg; | |
return this; | |
} | |
return this.media[ name ]; | |
}; | |
}); | |
return ret; | |
})() | |
); | |
Popcorn.forEach( "enable disable".split(" "), function( method ) { | |
Popcorn.p[ method ] = function( plugin ) { | |
return Popcorn[ method ]( this, plugin ); | |
}; | |
}); | |
Popcorn.extend(Popcorn.p, { | |
// Rounded currentTime | |
roundTime: function() { | |
return -~this.media.currentTime; | |
}, | |
// Attach an event to a single point in time | |
exec: function( time, fn ) { | |
// Creating a one second track event with an empty end | |
Popcorn.addTrackEvent( this, { | |
start: time, | |
end: time + 1, | |
_running: false, | |
_natives: { | |
start: fn || Popcorn.nop, | |
end: Popcorn.nop, | |
type: "exec" | |
} | |
}); | |
return this; | |
}, | |
// Get the client bounding box of an instance element | |
position: function() { | |
return Popcorn.position( this.media ); | |
}, | |
// Toggle a plugin's playback behaviour (on or off) per instance | |
toggle: function( plugin ) { | |
return Popcorn[ this.data.disabled.indexOf( plugin ) > -1 ? "enable" : "disable" ]( this, plugin ); | |
}, | |
// Set default values for plugin options objects per instance | |
defaults: function( plugin, defaults ) { | |
// If an array of default configurations is provided, | |
// iterate and apply each to this instance | |
if ( Popcorn.isArray( plugin ) ) { | |
Popcorn.forEach( plugin, function( obj ) { | |
for ( var name in obj ) { | |
this.defaults( name, obj[ name ] ); | |
} | |
}, this ); | |
return this; | |
} | |
if ( !this.options.defaults ) { | |
this.options.defaults = {}; | |
} | |
if ( !this.options.defaults[ plugin ] ) { | |
this.options.defaults[ plugin ] = {}; | |
} | |
Popcorn.extend( this.options.defaults[ plugin ], defaults ); | |
return this; | |
} | |
}); | |
Popcorn.Events = { | |
UIEvents: "blur focus focusin focusout load resize scroll unload", | |
MouseEvents: "mousedown mouseup mousemove mouseover mouseout mouseenter mouseleave click dblclick", | |
Events: "loadstart progress suspend emptied stalled play pause " + | |
"loadedmetadata loadeddata waiting playing canplay canplaythrough " + | |
"seeking seeked timeupdate ended ratechange durationchange volumechange" | |
}; | |
Popcorn.Events.Natives = Popcorn.Events.UIEvents + " " + | |
Popcorn.Events.MouseEvents + " " + | |
Popcorn.Events.Events; | |
internal.events.apiTypes = [ "UIEvents", "MouseEvents", "Events" ]; | |
// Privately compile events table at load time | |
(function( events, data ) { | |
var apis = internal.events.apiTypes, | |
eventsList = events.Natives.split( /\s+/g ), | |
idx = 0, len = eventsList.length, prop; | |
for( ; idx < len; idx++ ) { | |
data.hash[ eventsList[idx] ] = true; | |
} | |
apis.forEach(function( val, idx ) { | |
data.apis[ val ] = {}; | |
var apiEvents = events[ val ].split( /\s+/g ), | |
len = apiEvents.length, | |
k = 0; | |
for ( ; k < len; k++ ) { | |
data.apis[ val ][ apiEvents[ k ] ] = true; | |
} | |
}); | |
})( Popcorn.Events, internal.events ); | |
Popcorn.events = { | |
isNative: function( type ) { | |
return !!internal.events.hash[ type ]; | |
}, | |
getInterface: function( type ) { | |
if ( !Popcorn.events.isNative( type ) ) { | |
return false; | |
} | |
var eventApi = internal.events, | |
apis = eventApi.apiTypes, | |
apihash = eventApi.apis, | |
idx = 0, len = apis.length, api, tmp; | |
for ( ; idx < len; idx++ ) { | |
tmp = apis[ idx ]; | |
if ( apihash[ tmp ][ type ] ) { | |
api = tmp; | |
break; | |
} | |
} | |
return api; | |
}, | |
// Compile all native events to single array | |
all: Popcorn.Events.Natives.split( /\s+/g ), | |
// Defines all Event handling static functions | |
fn: { | |
trigger: function( type, data ) { | |
var eventInterface, evt; | |
// setup checks for custom event system | |
if ( this.data.events[ type ] && Popcorn.sizeOf( this.data.events[ type ] ) ) { | |
eventInterface = Popcorn.events.getInterface( type ); | |
if ( eventInterface ) { | |
evt = document.createEvent( eventInterface ); | |
evt.initEvent( type, true, true, global, 1 ); | |
this.media.dispatchEvent( evt ); | |
return this; | |
} | |
// Custom events | |
Popcorn.forEach( this.data.events[ type ], function( obj, key ) { | |
obj.call( this, data ); | |
}, this ); | |
} | |
return this; | |
}, | |
listen: function( type, fn ) { | |
var self = this, | |
hasEvents = true, | |
eventHook = Popcorn.events.hooks[ type ], | |
origType = type, | |
tmp; | |
if ( !this.data.events[ type ] ) { | |
this.data.events[ type ] = {}; | |
hasEvents = false; | |
} | |
// Check and setup event hooks | |
if ( eventHook ) { | |
// Execute hook add method if defined | |
if ( eventHook.add ) { | |
eventHook.add.call( this, {}, fn ); | |
} | |
// Reassign event type to our piggyback event type if defined | |
if ( eventHook.bind ) { | |
type = eventHook.bind; | |
} | |
// Reassign handler if defined | |
if ( eventHook.handler ) { | |
tmp = fn; | |
fn = function wrapper( event ) { | |
eventHook.handler.call( self, event, tmp ); | |
}; | |
} | |
// assume the piggy back event is registered | |
hasEvents = true; | |
// Setup event registry entry | |
if ( !this.data.events[ type ] ) { | |
this.data.events[ type ] = {}; | |
// Toggle if the previous assumption was untrue | |
hasEvents = false; | |
} | |
} | |
// Register event and handler | |
this.data.events[ type ][ fn.name || ( fn.toString() + Popcorn.guid() ) ] = fn; | |
// only attach one event of any type | |
if ( !hasEvents && Popcorn.events.all.indexOf( type ) > -1 ) { | |
this.media.addEventListener( type, function( event ) { | |
Popcorn.forEach( self.data.events[ type ], function( obj, key ) { | |
if ( typeof obj === "function" ) { | |
obj.call( self, event ); | |
} | |
}); | |
}, false); | |
} | |
return this; | |
}, | |
unlisten: function( type, fn ) { | |
if ( this.data.events[ type ] && this.data.events[ type ][ fn ] ) { | |
delete this.data.events[ type ][ fn ]; | |
return this; | |
} | |
this.data.events[ type ] = null; | |
return this; | |
} | |
}, | |
hooks: { | |
canplayall: { | |
bind: "canplaythrough", | |
add: function( event, callback ) { | |
var state = false; | |
if ( this.media.readyState ) { | |
callback.call( this, event ); | |
state = true; | |
} | |
this.data.hooks.canplayall = { | |
fired: state | |
}; | |
}, | |
// declare special handling instructions | |
handler: function canplayall( event, callback ) { | |
if ( !this.data.hooks.canplayall.fired ) { | |
// trigger original user callback once | |
callback.call( this, event ); | |
this.data.hooks.canplayall.fired = true; | |
} | |
} | |
} | |
} | |
}; | |
// Extend Popcorn.events.fns (listen, unlisten, trigger) to all Popcorn instances | |
Popcorn.forEach( [ "trigger", "listen", "unlisten" ], function( key ) { | |
Popcorn.p[ key ] = Popcorn.events.fn[ key ]; | |
}); | |
// Protected API methods | |
Popcorn.protect = { | |
natives: "load play pause currentTime playbackRate mute volume duration removePlugin roundTime trigger listen unlisten exec".toLowerCase().split( /\s+/ ) | |
}; | |
// Internal Only - Adds track events to the instance object | |
Popcorn.addTrackEvent = function( obj, track ) { | |
// Determine if this track has default options set for it | |
// If so, apply them to the track object | |
if ( track && track._natives && track._natives.type && | |
( obj.options.defaults && obj.options.defaults[ track._natives.type ] ) ) { | |
track = Popcorn.extend( {}, obj.options.defaults[ track._natives.type ], track ); | |
} | |
if ( track._natives ) { | |
// Supports user defined track event id | |
track._id = !track.id ? Popcorn.guid( track._natives.type ) : track.id; | |
// Push track event ids into the history | |
obj.data.history.push( track._id ); | |
} | |
track.start = Popcorn.util.toSeconds( track.start, obj.options.framerate ); | |
track.end = Popcorn.util.toSeconds( track.end, obj.options.framerate ); | |
// Store this definition in an array sorted by times | |
var byStart = obj.data.trackEvents.byStart, | |
byEnd = obj.data.trackEvents.byEnd, | |
idx; | |
for ( idx = byStart.length - 1; idx >= 0; idx-- ) { | |
if ( track.start >= byStart[ idx ].start ) { | |
byStart.splice( idx + 1, 0, track ); | |
break; | |
} | |
} | |
for ( idx = byEnd.length - 1; idx >= 0; idx-- ) { | |
if ( track.end > byEnd[ idx ].end ) { | |
byEnd.splice( idx + 1, 0, track ); | |
break; | |
} | |
} | |
// Store references to user added trackevents in ref table | |
if ( track._id ) { | |
Popcorn.addTrackEvent.ref( obj, track ); | |
} | |
}; | |
// Internal Only - Adds track event references to the instance object's trackRefs hash table | |
Popcorn.addTrackEvent.ref = function( obj, track ) { | |
obj.data.trackRefs[ track._id ] = track; | |
return obj; | |
}; | |
Popcorn.removeTrackEvent = function( obj, trackId ) { | |
var historyLen = obj.data.history.length, | |
indexWasAt = 0, | |
byStart = [], | |
byEnd = [], | |
history = []; | |
Popcorn.forEach( obj.data.trackEvents.byStart, function( o, i, context ) { | |
// Preserve the original start/end trackEvents | |
if ( !o._id ) { | |
byStart.push( obj.data.trackEvents.byStart[i] ); | |
byEnd.push( obj.data.trackEvents.byEnd[i] ); | |
} | |
// Filter for user track events (vs system track events) | |
if ( o._id ) { | |
// Filter for the trackevent to remove | |
if ( o._id !== trackId ) { | |
byStart.push( obj.data.trackEvents.byStart[i] ); | |
byEnd.push( obj.data.trackEvents.byEnd[i] ); | |
} | |
// Capture the position of the track being removed. | |
if ( o._id === trackId ) { | |
indexWasAt = i; | |
o._natives._teardown && o._natives._teardown.call( obj, o ); | |
} | |
} | |
}); | |
// Update | |
if ( indexWasAt <= obj.data.trackEvents.startIndex ) { | |
obj.data.trackEvents.startIndex--; | |
} | |
if ( indexWasAt <= obj.data.trackEvents.endIndex ) { | |
obj.data.trackEvents.endIndex--; | |
} | |
obj.data.trackEvents.byStart = byStart; | |
obj.data.trackEvents.byEnd = byEnd; | |
for ( var i = 0; i < historyLen; i++ ) { | |
if ( obj.data.history[ i ] !== trackId ) { | |
history.push( obj.data.history[ i ] ); | |
} | |
} | |
// Update ordered history array | |
obj.data.history = history; | |
// Update track event references | |
Popcorn.removeTrackEvent.ref( obj, trackId ); | |
}; | |
// Internal Only - Removes track event references from instance object's trackRefs hash table | |
Popcorn.removeTrackEvent.ref = function( obj, trackId ) { | |
delete obj.data.trackRefs[ trackId ]; | |
return obj; | |
}; | |
// Return an array of track events bound to this instance object | |
Popcorn.getTrackEvents = function( obj ) { | |
var trackevents = [], | |
refs = obj.data.trackEvents.byStart, | |
length = refs.length, | |
idx = 0, | |
ref; | |
for ( ; idx < length; idx++ ) { | |
ref = refs[ idx ]; | |
// Return only user attributed track event references | |
if ( ref._id ) { | |
trackevents.push( ref ); | |
} | |
} | |
return trackevents; | |
}; | |
// Internal Only - Returns an instance object's trackRefs hash table | |
Popcorn.getTrackEvents.ref = function( obj ) { | |
return obj.data.trackRefs; | |
}; | |
// Return a single track event bound to this instance object | |
Popcorn.getTrackEvent = function( obj, trackId ) { | |
return obj.data.trackRefs[ trackId ]; | |
}; | |
// Internal Only - Returns an instance object's track reference by track id | |
Popcorn.getTrackEvent.ref = function( obj, trackId ) { | |
return obj.data.trackRefs[ trackId ]; | |
}; | |
Popcorn.getLastTrackEventId = function( obj ) { | |
return obj.data.history[ obj.data.history.length - 1 ]; | |
}; | |
// Map and Extend TrackEvent functions to all Popcorn instances | |
Popcorn.extend( Popcorn.p, { | |
getTrackEvents: function() { | |
return Popcorn.getTrackEvents.call( null, this ); | |
}, | |
getTrackEvent: function( id ) { | |
return Popcorn.getTrackEvent.call( null, this, id ); | |
}, | |
getLastTrackEventId: function() { | |
return Popcorn.getLastTrackEventId.call( null, this ); | |
}, | |
removeTrackEvent: function( id ) { | |
Popcorn.removeTrackEvent.call( null, this, id ); | |
return this; | |
}, | |
removePlugin: function( name ) { | |
Popcorn.removePlugin.call( null, this, name ); | |
return this; | |
} | |
}); | |
// Plugin manifests | |
Popcorn.manifest = {}; | |
// Plugins are registered | |
Popcorn.registry = []; | |
Popcorn.registryByName = {}; | |
// An interface for extending Popcorn | |
// with plugin functionality | |
Popcorn.plugin = function( name, definition, manifest ) { | |
if ( Popcorn.protect.natives.indexOf( name.toLowerCase() ) >= 0 ) { | |
Popcorn.error( "'" + name + "' is a protected function name" ); | |
return; | |
} | |
// Provides some sugar, but ultimately extends | |
// the definition into Popcorn.p | |
var reserved = [ "start", "end" ], | |
plugin = {}, | |
setup, | |
isfn = typeof definition === "function", | |
methods = [ "_setup", "_teardown", "start", "end" ]; | |
// combines calls of two function calls into one | |
var combineFn = function( first, second ) { | |
first = first || Popcorn.nop; | |
second = second || Popcorn.nop; | |
return function() { | |
first.apply( this, arguments ); | |
second.apply( this, arguments ); | |
}; | |
}; | |
// If `manifest` arg is undefined, check for manifest within the `definition` object | |
// If no `definition.manifest`, an empty object is a sufficient fallback | |
if ( !manifest ) { | |
manifest = definition.manifest || {}; | |
} | |
// apply safe, and empty default functions | |
methods.forEach(function( method ) { | |
definition[ method ] = definition[ method ] || Popcorn.nop; | |
}); | |
var pluginFn = function( setup, options ) { | |
if ( !options ) { | |
return this; | |
} | |
// Storing the plugin natives | |
var natives = options._natives = {}, | |
compose = "", | |
defaults, originalOpts, manifestOpts, mergedSetupOpts; | |
Popcorn.extend( natives, setup ); | |
options._natives.type = name; | |
options._running = false; | |
// Check for previously set default options | |
defaults = this.options.defaults && this.options.defaults[ options._natives && options._natives.type ]; | |
// default to an empty string if no effect exists | |
// split string into an array of effects | |
options.compose = options.compose && options.compose.split( " " ) || []; | |
options.effect = options.effect && options.effect.split( " " ) || []; | |
// join the two arrays together | |
options.compose = options.compose.concat( options.effect ); | |
options.compose.forEach(function( composeOption ) { | |
// if the requested compose is garbage, throw it away | |
compose = Popcorn.compositions[ composeOption ] || {}; | |
// extends previous functions with compose function | |
methods.forEach(function( method ) { | |
natives[ method ] = combineFn( natives[ method ], compose[ method ] ); | |
}); | |
}); | |
// Ensure a manifest object, an empty object is a sufficient fallback | |
options._natives.manifest = manifest; | |
// Checks for expected properties | |
if ( !( "start" in options ) ) { | |
options.start = 0; | |
} | |
if ( !( "end" in options ) ) { | |
options.end = this.duration() || Number.MAX_VALUE; | |
} | |
// Merge with defaults if they exist, make sure per call is prioritized | |
mergedSetupOpts = defaults ? Popcorn.extend( {}, defaults, options ) : | |
options; | |
// Resolves 239, 241, 242 | |
if ( !mergedSetupOpts.target ) { | |
// Sometimes the manifest may be missing entirely | |
// or it has an options object that doesn't have a `target` property | |
manifestOpts = "options" in manifest && manifest.options; | |
mergedSetupOpts.target = manifestOpts && "target" in manifestOpts && manifestOpts.target; | |
} | |
// Trigger _setup method if exists | |
options._natives._setup && options._natives._setup.call( this, mergedSetupOpts ); | |
// Create new track event for this instance | |
Popcorn.addTrackEvent( this, Popcorn.extend( mergedSetupOpts, options ) ); | |
// Future support for plugin event definitions | |
// for all of the native events | |
Popcorn.forEach( setup, function( callback, type ) { | |
if ( type !== "type" ) { | |
if ( reserved.indexOf( type ) === -1 ) { | |
this.listen( type, callback ); | |
} | |
} | |
}, this ); | |
return this; | |
}; | |
// Augment the manifest object | |
if ( manifest || ( "manifest" in definition ) ) { | |
Popcorn.manifest[ name ] = manifest || definition.manifest; | |
} | |
// Assign new named definition | |
plugin[ name ] = function( options ) { | |
return pluginFn.call( this, isfn ? definition.call( this, options ) : definition, | |
options ); | |
}; | |
// Extend Popcorn.p with new named definition | |
Popcorn.extend( Popcorn.p, plugin ); | |
// Push into the registry | |
var entry = { | |
fn: plugin[ name ], | |
definition: definition, | |
base: definition, | |
parents: [], | |
name: name | |
}; | |
Popcorn.registry.push( | |
Popcorn.extend( plugin, entry, { | |
type: name | |
}) | |
); | |
Popcorn.registryByName[ name ] = entry; | |
return plugin; | |
}; | |
Popcorn.plugin.debug = false; | |
// removePlugin( type ) removes all tracks of that from all instances of popcorn | |
// removePlugin( obj, type ) removes all tracks of type from obj, where obj is a single instance of popcorn | |
Popcorn.removePlugin = function( obj, name ) { | |
// Check if we are removing plugin from an instance or from all of Popcorn | |
if ( !name ) { | |
// Fix the order | |
name = obj; | |
obj = Popcorn.p; | |
if ( Popcorn.protect.natives.indexOf( name.toLowerCase() ) >= 0 ) { | |
Popcorn.error( "'" + name + "' is a protected function name" ); | |
return; | |
} | |
var registryLen = Popcorn.registry.length, | |
registryIdx; | |
// remove plugin reference from registry | |
for ( registryIdx = 0; registryIdx < registryLen; registryIdx++ ) { | |
if ( Popcorn.registry[ registryIdx ].name === name ) { | |
Popcorn.registry.splice( registryIdx, 1 ); | |
delete Popcorn.registryByName[ name ]; | |
// delete the plugin | |
delete obj[ name ]; | |
// plugin found and removed, stop checking, we are done | |
return; | |
} | |
} | |
} | |
var byStart = obj.data.trackEvents.byStart, | |
byEnd = obj.data.trackEvents.byEnd, | |
idx, sl; | |
// remove all trackEvents | |
for ( idx = 0, sl = byStart.length; idx < sl; idx++ ) { | |
if ( ( byStart[ idx ] && byStart[ idx ]._natives && byStart[ idx ]._natives.type === name ) && | |
( byEnd[ idx ] && byEnd[ idx ]._natives && byEnd[ idx ]._natives.type === name ) ) { | |
byStart[ idx ]._natives._teardown && byStart[ idx ]._natives._teardown.call( obj, byStart[ idx ] ); | |
byStart.splice( idx, 1 ); | |
byEnd.splice( idx, 1 ); | |
// update for loop if something removed, but keep checking | |
idx--; sl--; | |
if ( obj.data.trackEvents.startIndex <= idx ) { | |
obj.data.trackEvents.startIndex--; | |
obj.data.trackEvents.endIndex--; | |
} | |
} | |
} | |
}; | |
Popcorn.compositions = {}; | |
// Plugin inheritance | |
Popcorn.compose = function( name, definition, manifest ) { | |
// If `manifest` arg is undefined, check for manifest within the `definition` object | |
// If no `definition.manifest`, an empty object is a sufficient fallback | |
Popcorn.manifest[ name ] = manifest = manifest || definition.manifest || {}; | |
// register the effect by name | |
Popcorn.compositions[ name ] = definition; | |
}; | |
Popcorn.plugin.effect = Popcorn.effect = Popcorn.compose; | |
// stores parsers keyed on filetype | |
Popcorn.parsers = {}; | |
// An interface for extending Popcorn | |
// with parser functionality | |
Popcorn.parser = function( name, type, definition ) { | |
if ( Popcorn.protect.natives.indexOf( name.toLowerCase() ) >= 0 ) { | |
Popcorn.error( "'" + name + "' is a protected function name" ); | |
return; | |
} | |
// fixes parameters for overloaded function call | |
if ( typeof type === "function" && !definition ) { | |
definition = type; | |
type = ""; | |
} | |
if ( typeof definition !== "function" || typeof type !== "string" ) { | |
return; | |
} | |
// Provides some sugar, but ultimately extends | |
// the definition into Popcorn.p | |
var natives = Popcorn.events.all, | |
parseFn, | |
parser = {}; | |
parseFn = function( filename, callback ) { | |
if ( !filename ) { | |
return this; | |
} | |
var that = this; | |
Popcorn.xhr({ | |
url: filename, | |
dataType: type, | |
success: function( data ) { | |
var tracksObject = definition( data ), | |
tracksData, | |
tracksDataLen, | |
tracksDef, | |
idx = 0; | |
tracksData = tracksObject.data || []; | |
tracksDataLen = tracksData.length; | |
tracksDef = null; | |
// If no tracks to process, return immediately | |
if ( !tracksDataLen ) { | |
return; | |
} | |
// Create tracks out of parsed object | |
for ( ; idx < tracksDataLen; idx++ ) { | |
tracksDef = tracksData[ idx ]; | |
for ( var key in tracksDef ) { | |
if ( hasOwn.call( tracksDef, key ) && !!that[ key ] ) { | |
that[ key ]( tracksDef[ key ] ); | |
} | |
} | |
} | |
if ( callback ) { | |
callback(); | |
} | |
} | |
}); | |
return this; | |
}; | |
// Assign new named definition | |
parser[ name ] = parseFn; | |
// Extend Popcorn.p with new named definition | |
Popcorn.extend( Popcorn.p, parser ); | |
// keys the function name by filetype extension | |
//Popcorn.parsers[ name ] = true; | |
return parser; | |
}; | |
// Cache references to reused RegExps | |
var rparams = /\?/, | |
// XHR Setup object | |
setup = { | |
url: "", | |
data: "", | |
dataType: "", | |
success: Popcorn.nop, | |
type: "GET", | |
async: true, | |
xhr: function() { | |
return new global.XMLHttpRequest(); | |
} | |
}; | |
Popcorn.xhr = function( options ) { | |
options.dataType = options.dataType && options.dataType.toLowerCase() || null; | |
if ( options.dataType && | |
( options.dataType === "jsonp" || options.dataType === "script" ) ) { | |
Popcorn.xhr.getJSONP( | |
options.url, | |
options.success, | |
options.dataType === "script" | |
); | |
return; | |
} | |
var settings = Popcorn.extend( {}, setup, options ); | |
// Create new XMLHttpRequest object | |
settings.ajax = settings.xhr(); | |
if ( settings.ajax ) { | |
if ( settings.type === "GET" && settings.data ) { | |
// append query string | |
settings.url += ( rparams.test( settings.url ) ? "&" : "?" ) + settings.data; | |
// Garbage collect and reset settings.data | |
settings.data = null; | |
} | |
settings.ajax.open( settings.type, settings.url, settings.async ); | |
settings.ajax.send( settings.data || null ); | |
return Popcorn.xhr.httpData( settings ); | |
} | |
}; | |
Popcorn.xhr.httpData = function( settings ) { | |
var data, json = null; | |
settings.ajax.onreadystatechange = function() { | |
if ( settings.ajax.readyState === 4 ) { | |
try { | |
json = JSON.parse( settings.ajax.responseText ); | |
} catch( e ) { | |
//suppress | |
} | |
data = { | |
xml: settings.ajax.responseXML, | |
text: settings.ajax.responseText, | |
json: json | |
}; | |
// If a dataType was specified, return that type of data | |
if ( settings.dataType ) { | |
data = data[ settings.dataType ]; | |
} | |
settings.success.call( settings.ajax, data ); | |
} | |
}; | |
return data; | |
}; | |
Popcorn.xhr.getJSONP = function( url, success, isScript ) { | |
// If this is a script request, ensure that we do not call something that has already been loaded | |
if ( isScript ) { | |
var scripts = document.querySelectorAll( "script[src=\"" + url + "\"]" ); | |
// If there are scripts with this url loaded, early return | |
if ( scripts.length ) { | |
// Execute success callback and pass "exists" flag | |
success && success( true ); | |
return; | |
} | |
} | |
var head = document.head || document.getElementsByTagName( "head" )[ 0 ] || document.documentElement, | |
script = document.createElement( "script" ), | |
paramStr = url.split( "?" )[ 1 ], | |
isFired = false, | |
params = [], | |
callback, parts, callparam; | |
if ( paramStr && !isScript ) { | |
params = paramStr.split( "&" ); | |
} | |
if ( params.length ) { | |
parts = params[ params.length - 1 ].split( "=" ); | |
} | |
callback = params.length ? ( parts[ 1 ] ? parts[ 1 ] : parts[ 0 ] ) : "jsonp"; | |
if ( !paramStr && !isScript ) { | |
url += "?callback=" + callback; | |
} | |
if ( callback && !isScript ) { | |
// If a callback name already exists | |
if ( !!window[ callback ] ) { | |
// Create a new unique callback name | |
callback = Popcorn.guid( callback ); | |
} | |
// Define the JSONP success callback globally | |
window[ callback ] = function( data ) { | |
success && success( data ); | |
isFired = true; | |
}; | |
// Replace callback param and callback name | |
url = url.replace( parts.join( "=" ), parts[ 0 ] + "=" + callback ); | |
} | |
script.onload = script.onreadystatechange = function() { | |
if ( !script.readyState || /loaded|complete/.test( script.readyState ) ) { | |
// Handling remote script loading callbacks | |
if ( isScript ) { | |
// getScript | |
success && success(); | |
} | |
// Executing for JSONP requests | |
if ( isFired ) { | |
// Garbage collect the callback | |
delete window[ callback ]; | |
// Garbage collect the script resource | |
head.removeChild( script ); | |
} | |
} | |
}; | |
script.src = url; | |
head.insertBefore( script, head.firstChild ); | |
return; | |
}; | |
Popcorn.getJSONP = Popcorn.xhr.getJSONP; | |
Popcorn.getScript = Popcorn.xhr.getScript = function( url, success ) { | |
return Popcorn.xhr.getJSONP( url, success, true ); | |
}; | |
Popcorn.util = { | |
// Simple function to parse a timestamp into seconds | |
// Acceptable formats are: | |
// HH:MM:SS.MMM | |
// HH:MM:SS;FF | |
// Hours and minutes are optional. They default to 0 | |
toSeconds: function( timeStr, framerate ) { | |
//Hours and minutes are optional | |
//Seconds must be specified | |
//Seconds can be followed by milliseconds OR by the frame information | |
var validTimeFormat = /^([0-9]+:){0,2}[0-9]+([.;][0-9]+)?$/, | |
errorMessage = "Invalid time format"; | |
if ( typeof timeStr === "number" ) { | |
return timeStr; | |
} else if ( typeof timeStr === "string" ) { | |
if ( ! validTimeFormat.test( timeStr ) ) { | |
Popcorn.error( errorMessage ); | |
} | |
} else { | |
Popcorn.error( errorMessage ); | |
} | |
var t = timeStr.split( ":" ), | |
lastIndex = t.length - 1, | |
lastElement = t[ lastIndex ]; | |
//Fix last element: | |
if ( lastElement.indexOf( ";" ) > -1 ) { | |
var frameInfo = lastElement.split( ";" ), | |
frameTime = 0; | |
if ( framerate && ( typeof framerate === "number" ) ) { | |
frameTime = parseFloat( frameInfo[ 1 ], 10 ) / framerate; | |
} | |
t[ lastIndex ] = | |
parseInt( frameInfo[ 0 ], 10 ) + frameTime; | |
} | |
if ( t.length === 1 ) { | |
return parseFloat( t[ 0 ], 10 ); | |
} else if ( t.length === 2 ) { | |
return ( parseInt( t[ 0 ], 10 ) * 60 ) + parseFloat( t[ 1 ], 10 ); | |
} else if ( t.length === 3 ) { | |
return ( parseInt( t[ 0 ], 10 ) * 3600 ) + | |
( parseInt( t[ 1 ], 10 ) * 60 ) + | |
parseFloat( t[ 2 ], 10 ); | |
} | |
} | |
}; | |
// Exposes Popcorn to global context | |
global.Popcorn = Popcorn; | |
document.addEventListener( "DOMContentLoaded", function() { | |
// Supports non-specific elements | |
var dataAttr = "data-timeline-sources", | |
medias = document.querySelectorAll( "[" + dataAttr + "]" ); | |
Popcorn.forEach( medias, function( idx, key ) { | |
var media = medias[ key ], | |
hasDataSources = false, | |
dataSources, data, popcornMedia; | |
// Ensure that the DOM has an id | |
if ( !media.id ) { | |
media.id = Popcorn.guid( "__popcorn" ); | |
} | |
// Ensure we're looking at a dom node | |
if ( media.nodeType && media.nodeType === 1 ) { | |
popcornMedia = Popcorn( "#" + media.id ); | |
dataSources = ( media.getAttribute( dataAttr ) || "" ).split( "," ); | |
if ( dataSources[ 0 ] ) { | |
Popcorn.forEach( dataSources, function( source ) { | |
// split the parser and data as parser!file | |
data = source.split( "!" ); | |
// if no parser is defined for the file, assume "parse" + file extension | |
if ( data.length === 1 ) { | |
data = source.split( "." ); | |
data[ 0 ] = "parse" + data[ data.length - 1 ].toUpperCase(); | |
data[ 1 ] = source; | |
} | |
// If the media has data sources and the correct parser is registered, continue to load | |
if ( dataSources[ 0 ] && popcornMedia[ data[ 0 ] ] ) { | |
// Set up the media and load in the datasources | |
popcornMedia[ data[ 0 ] ]( data[ 1 ] ); | |
} | |
}); | |
} | |
// Only play the media if it was specified to do so | |
if ( !!popcornMedia.autoplay ) { | |
popcornMedia.play(); | |
} | |
} | |
}); | |
}, false ); | |
})(window, window.document); | |
// PLUGIN: Mustache | |
(function (Popcorn) { | |
/** | |
* Mustache Popcorn Plug-in | |
* | |
* Adds the ability to render JSON using templates via the Mustache templating library. | |
* | |
* @param {Object} options | |
* | |
* Required parameters: start, end, template, data, and target. | |
* Optional parameter: static. | |
* | |
* start: the time in seconds when the mustache template should be rendered | |
* in the target div. | |
* | |
* end: the time in seconds when the rendered mustache template should be | |
* removed from the target div. | |
* | |
* target: a String -- the target div's id. | |
* | |
* template: the mustache template for the plugin to use when rendering. This can be | |
* a String containing the template, or a Function that returns the template's | |
* String. | |
* | |
* data: the data to be rendered using the mustache template. This can be a JSON String, | |
* a JavaScript Object literal, or a Function returning a String or Literal. | |
* | |
* dynamic: an optional argument indicating that the template and json data are dynamic | |
* and need to be loaded dynamically on every use. Defaults to True. | |
* | |
* Example: | |
var p = Popcorn('#video') | |
// Example using template and JSON strings. | |
.mustache({ | |
start: 5, // seconds | |
end: 15, // seconds | |
target: 'mustache', | |
template: '<h1>{{header}}</h1>' + | |
'{{#bug}}' + | |
'{{/bug}}' + | |
'' + | |
'{{#items}}' + | |
' {{#first}}' + | |
' <li><strong>{{name}}</strong></li>' + | |
' {{/first}}' + | |
' {{#link}}' + | |
' <li><a href="{{url}}">{{name}}</a></li>' + | |
' {{/link}}' + | |
'{{/items}}' + | |
'' + | |
'{{#empty}}' + | |
' <p>The list is empty.</p>' + | |
'{{/empty}}' , | |
data: '{' + | |
' "header": "Colors", ' + | |
' "items": [ ' + | |
' {"name": "red", "first": true, "url": "#Red"}, ' + | |
' {"name": "green", "link": true, "url": "#Green"}, ' + | |
' {"name": "blue", "link": true, "url": "#Blue"} ' + | |
' ],' + | |
' 'empty': false' + | |
'}', | |
dynamic: false // The json is not going to change, load it early. | |
} ) | |
// Example showing Functions instead of Strings. | |
.mustache({ | |
start: 20, // seconds | |
end: 25, // seconds | |
target: 'mustache', | |
template: function(instance, options) { | |
var template = // load your template file here... | |
return template; | |
}, | |
data: function(instance, options) { | |
var json = // load your json here... | |
return json; | |
} | |
} ); | |
* | |
*/ | |
Popcorn.plugin( 'mustache' , function( options ) { | |
var getData, data, getTemplate, template; | |
Popcorn.getScript('https://github.com/janl/mustache.js/raw/master/mustache.js'); | |
var shouldReload = !!options.dynamic, | |
typeOfTemplate = typeof options.template, | |
typeOfData = typeof options.data, | |
target = document.getElementById( options.target ); | |
if ( !target && Popcorn.plugin.debug ) { | |
throw new Error( "target container doesn't exist" ); | |
} | |
options.container = target || document.createElement( "div" ); | |
if ( typeOfTemplate === 'function' ) { | |
if ( !shouldReload ) { | |
template = options.template( options ); | |
} else { | |
getTemplate = options.template; | |
} | |
} else if ( typeOfTemplate === 'string' ) { | |
template = options.template; | |
} else if ( Popcorn.plugin.debug ) { | |
throw new Error( 'Mustache Plugin Error: options.template must be a String or a Function.' ); | |
} else { | |
template = ""; | |
} | |
if ( typeOfData === 'function' ) { | |
if ( !shouldReload ) { | |
data = options.data(options); | |
} else { | |
getData = options.data; | |
} | |
} else if ( typeOfData === 'string' ) { | |
data = JSON.parse( options.data ); | |
} else if ( typeOfData === 'object' ) { | |
data = options.data; | |
} else if ( Popcorn.plugin.debug ) { | |
throw new Error( 'Mustache Plugin Error: options.data must be a String, Object, or Function.' ); | |
} else { | |
data = ""; | |
} | |
return { | |
start: function( event, options ) { | |
var interval = function() { | |
if( !window.Mustache ) { | |
setTimeout( function() { | |
interval(); | |
}, 10 ); | |
} else { | |
// if dynamic, freshen json data on every call to start, just in case. | |
if ( getData ) { | |
data = getData( options ); | |
} | |
if ( getTemplate ) { | |
template = getTemplate( options ); | |
} | |
var html = Mustache.to_html( template, | |
data | |
).replace( /^\s*/mg, '' ); | |
options.container.innerHTML = html; | |
} | |
}; | |
interval(); | |
}, | |
end: function( event, options ) { | |
options.container.innerHTML = ''; | |
}, | |
_teardown: function( options ) { | |
getData = data = getTemplate = template = null; | |
} | |
}; | |
}, | |
{ | |
about: { | |
name: 'Popcorn Mustache Plugin', | |
version: '0.1', | |
author: 'David Humphrey (@humphd)', | |
website: 'http://vocamus.net/dave' | |
}, | |
options: { | |
start: {elem:'input', type:'text', label:'In'}, | |
end: {elem:'input', type:'text', label:'Out'}, | |
target: 'mustache-container', | |
template: {elem:'input', type:'text', label:'Template'}, | |
data: {elem:'input', type:'text', label:'Data'}, | |
/* TODO: how to show a checkbox/boolean? */ | |
dynamic: {elem:'input', type:'text', label:'Dynamic'} | |
} | |
}); | |
})( Popcorn ); | |
//PLUGIN: facebook | |
(function(Popcorn, global ) { | |
/** | |
* Facebook Popcorn plug-in | |
* Places Facebook's "social plugins" inside a div ( http://developers.facebook.com/docs/plugins/ ) | |
* Sets options according to user input or default values | |
* Options parameter will need a target, type, start and end time | |
* Type is the name of the plugin in fbxml format. Options: LIKE (default), LIKE-BOX, ACTIVITY, FACEPILE | |
* Target is the id of the document element that the text needs to be attached to. This target element must exist on the DOM | |
* Start is the time that you want this plug-in to execute | |
* End is the time that you want this plug-in to stop executing | |
* | |
* Other than the mandatory four parameters, there are several optional parameters (Some options are only applicable to certain plugins) | |
* Action - like button will either "Like" or "Recommend". Options: recommend / like(default) | |
* Always_post_to_friends - live-stream posts will be always be posted on your facebook wall if true. Options: true / false(default) | |
* Border_color - border color of the activity feed. Names (i.e: "white") and html color codes are valid | |
* Colorscheme - changes the color of almost all plugins. Options: light(default) / dark | |
* Event_app_id - an app_id is required for the live-stream plugin | |
* Font - the font of the text contained in the plugin. Options: arial / segoe ui / tahoma / trebuchet ms / verdana / lucida grande | |
* Header - displays the title of like-box or activity feed. Options: true / false(default) | |
* Href - url to apply to the plugin. Default is current page | |
* Layout - changes the format of the 'like' count (written in english or a number in a callout). | |
* Options: box_count / button_count / standard(default) | |
* Max_rows - number of rows to disperse pictures in facepile. Default is 1 | |
* Recommendations - shows recommendations, if any, in the bottom half of activity feed. Options: true / false(default) | |
* Show_faces - show pictures beside like button and like-box. Options: true / false(default) | |
* Site - href for activity feed. No idea why it must be "site". Default is current page | |
* Stream - displays a the latest posts from the specified page's wall. Options: true / false(default) | |
* Type - determines which plugin to create. Case insensitive | |
* Xid - unique identifier if more than one live-streams are on one page | |
* | |
* @param {Object} options | |
* | |
* Example: | |
var p = Popcorn('#video') | |
.facebook({ | |
type : "LIKE-BOX", | |
target: "likeboxdiv", | |
start : 3, | |
end : 10, | |
href : "http://www.facebook.com/senecacollege", | |
show_faces: "true", | |
header: "false" | |
} ) | |
* This will show how many people "like" Seneca College's Facebook page, and show their profile pictures | |
*/ | |
var ranOnce = false; | |
function toggle( container, display ) { | |
if ( container ) { | |
container.style.display = display; | |
return; | |
} | |
setTimeout(function() { | |
toggle( container, display ); | |
}, 10 ); | |
} | |
Popcorn.plugin( "facebook" , { | |
manifest:{ | |
about:{ | |
name : "Popcorn Facebook Plugin", | |
version: "0.1", | |
author : "Dan Ventura", | |
website: "dsventura.blogspot.com" | |
}, | |
options:{ | |
type : {elem:"select", options:["LIKE", "LIKE-BOX", "ACTIVITY", "FACEPILE", "LIVE-STREAM", "SEND"], label:"Type"}, | |
target : "facebook-container", | |
start : {elem:'input', type:'number', label:'In'}, | |
end : {elem:'input', type:'number', label:'Out'}, | |
// optional parameters: | |
font : {elem:"input", type:"text", label:"font"}, | |
xid : {elem:"input", type:"text", label:"Xid"}, | |
href : {elem:"input", type:"text", label:"Href"}, | |
site : {elem:"input", type:"text", label:"Site"}, | |
height : {elem:"input", type:"text", label:"Height"}, | |
width : {elem:"input", type:"text", label:"Width"}, | |
action : {elem:"select", options:["like", "recommend"], label:"Action"}, | |
stream : {elem:"select", options:["false", "true"], label:"Stream"}, | |
header : {elem:"select", options:["false", "true"], label:"Header"}, | |
layout : {elem:"select", options:["standard", "button_count", "box_count"], label:"Layout"}, | |
max_rows : {elem:"input", type:"text", label:"Max_rows"}, | |
border_color : {elem:"input", type:"text", label:"Border_color"}, | |
event_app_id : {elem:"input", type:"text", label:"Event_app_id"}, | |
colorscheme : {elem:"select", options:["light", "dark"], label:"Colorscheme"}, | |
show_faces : {elem:"select", options:["false", "true"], label:"Showfaces"}, | |
recommendations : {elem:"select", options:["false", "true"], label:"Recommendations"}, | |
always_post_to_friends : {elem:"input", options:["false", "true"], label:"Always_post_to_friends"} | |
} | |
}, | |
_setup: function( options ) { | |
var target = document.getElementById( options.target ); | |
// facebook script requires a div named fb-root | |
if ( !document.getElementById( "fb-root" ) ) { | |
var fbRoot = document.createElement( "div" ); | |
fbRoot.setAttribute( "id", "fb-root" ); | |
document.body.appendChild( fbRoot ); | |
} | |
if ( !ranOnce || options.event_app_id ) { | |
ranOnce = true; | |
// initialize facebook JS SDK | |
Popcorn.getScript("http://connect.facebook.net/en_US/all.js"); | |
global.fbAsyncInit = function() { | |
FB.init({ | |
appId : ( options.event_app_id || "" ), | |
status : true, | |
cookie : true, | |
xfbml : true | |
}); | |
}; | |
} | |
var validType = function( type ) { | |
return ( [ "like", "like-box", "activity", "facepile", "comments", "live-stream", "send" ].indexOf( type ) > -1 ); | |
}; | |
// default plugin is like button | |
options.type = ( options.type || "like" ).toLowerCase(); | |
// default plugin is like button | |
if ( !validType( options.type ) ) { | |
options.type = "like"; | |
} | |
options._container = document.createElement( "fb:" + options.type ); | |
var setOptions = (function( options ) { | |
options._container.style.display = "none"; | |
// activity feed uses 'site' rather than 'href' | |
var attr = options.type === "activity" ? "site" : "href"; | |
options._container.setAttribute( attr, ( options[ attr ] || document.URL ) ); | |
return { | |
"like": function () { | |
options._container.setAttribute( "send", ( options.send || false ) ); | |
options._container.setAttribute( "width", options.width ); | |
options._container.setAttribute( "show_faces", options.show_faces ); | |
options._container.setAttribute( "layout", options.layout ); | |
options._container.setAttribute( "font", options.font ); | |
options._container.setAttribute( "colorscheme", options.colorscheme ); | |
}, | |
"like-box": function () { | |
options._container.setAttribute( "height", ( options.height || 250 ) ); | |
options._container.setAttribute( "width", options.width ); | |
options._container.setAttribute( "show_faces", options.show_faces ); | |
options._container.setAttribute( "stream", options.stream ); | |
options._container.setAttribute( "header", options.header ); | |
options._container.setAttribute( "colorscheme", options.colorscheme ); | |
}, | |
"facepile": function () { | |
options._container.setAttribute( "height", options.height ); | |
options._container.setAttribute( "width", options.width ); | |
options._container.setAttribute( "max_rows", ( options.max_rows || 1 ) ); | |
}, | |
"activity": function () { | |
options._container.setAttribute( "width", options.width ); | |
options._container.setAttribute( "height", options.height ); | |
options._container.setAttribute( "header", options.header ); | |
options._container.setAttribute( "border_color", options.border_color ); | |
options._container.setAttribute( "recommendations", options.recommendations ); | |
options._container.setAttribute( "font", options.font ); | |
options._container.setAttribute( "colorscheme", options.colorscheme ); | |
}, | |
"live-stream": function() { | |
options._container.setAttribute( "width", ( options.width || 400 ) ); | |
options._container.setAttribute( "height", ( options.height || 500 ) ); | |
options._container.setAttribute( "always_post_to_friends", ( options.always_post_to_friends || false ) ); | |
options._container.setAttribute( "event_app_id", options.event_app_id ); | |
options._container.setAttribute( "xid", options.xid ); | |
}, | |
"send": function() { | |
options._container.setAttribute( "font", options.font ); | |
options._container.setAttribute( "colorscheme", options.colorscheme ); | |
} | |
}; | |
})( options ); | |
setOptions[ options.type ](); | |
if ( !target && Popcorn.plugin.debug ) { | |
throw new Error( "flickr target container doesn't exist" ); | |
} | |
target && target.appendChild( options._container ); | |
}, | |
/** | |
* @member facebook | |
* The start function will be executed when the currentTime | |
* of the video reaches the start time provided by the | |
* options variable | |
*/ | |
start: function( event, options ){ | |
toggle( options._container, "inline" ); | |
}, | |
/** | |
* @member facebook | |
* The end function will be executed when the currentTime | |
* of the video reaches the end time provided by the | |
* options variable | |
*/ | |
end: function( event, options ){ | |
toggle ( options._container, "none" ); | |
} | |
}); | |
})( Popcorn, this ); | |
// PLUGIN: IMAGE | |
(function (Popcorn) { | |
/** | |
* Images popcorn plug-in | |
* Shows an image element | |
* Options parameter will need a start, end, href, target and src. | |
* Start is the time that you want this plug-in to execute | |
* End is the time that you want this plug-in to stop executing | |
* href is the url of the destination of a link - optional | |
* Target is the id of the document element that the iframe needs to be attached to, | |
* this target element must exist on the DOM | |
* Src is the url of the image that you want to display | |
* text is the overlayed text on the image - optional | |
* | |
* @param {Object} options | |
* | |
* Example: | |
var p = Popcorn('#video') | |
.image({ | |
start: 5, // seconds | |
end: 15, // seconds | |
href: 'http://www.drumbeat.org/', | |
src: 'http://www.drumbeat.org/sites/default/files/domain-2/drumbeat_logo.png', | |
text: 'DRUMBEAT', | |
target: 'imagediv' | |
} ) | |
* | |
*/ | |
Popcorn.plugin( "image", { | |
manifest: { | |
about:{ | |
name: "Popcorn image Plugin", | |
version: "0.1", | |
author: "Scott Downe", | |
website: "http://scottdowne.wordpress.com/" | |
}, | |
options: { | |
start: { | |
elem: "input", | |
type: "number", | |
label: "In" | |
}, | |
end: { | |
elem: "input", | |
type: "number", | |
label: "Out" | |
}, | |
href: { | |
elem: "input", | |
type: "text", | |
label: "Link URL" | |
}, | |
target: "image-container", | |
src: { | |
elem: "input", | |
type: "text", | |
label: "Source URL" | |
}, | |
text: { | |
elem: "input", | |
type: "text", | |
label: "TEXT" | |
} | |
} | |
}, | |
_setup: function( options ) { | |
var img = document.createElement( "img" ), | |
target = document.getElementById( options.target ); | |
options.link = document.createElement( "a" ); | |
options.link.style.position = "relative"; | |
options.link.style.textDecoration = "none"; | |
if ( !target && Popcorn.plugin.debug ) { | |
throw new Error( "target container doesn't exist" ); | |
} | |
// add the widget's div to the target div | |
target && target.appendChild( options.link ); | |
img.addEventListener( "load", function() { | |
// borders look really bad, if someone wants it they can put it on their div target | |
img.style.borderStyle = "none"; | |
if ( options.href ) { | |
options.link.href = options.href; | |
} | |
options.link.target = "_blank"; | |
var fontHeight = ( img.height / 12 ) + "px", | |
divText = document.createElement( "div" ); | |
Popcorn.extend( divText.style, { | |
color: "black", | |
fontSize: fontHeight, | |
fontWeight: "bold", | |
position: "relative", | |
textAlign: "center", | |
width: img.width + "px", | |
zIndex: "10" | |
}); | |
divText.innerHTML = options.text || ""; | |
options.link.appendChild( divText ); | |
options.link.appendChild( img ); | |
divText.style.top = ( img.height / 2 ) - ( divText.offsetHeight / 2 ) + "px"; | |
options.link.style.display = "none"; | |
}, false ); | |
img.src = options.src; | |
}, | |
/** | |
* @member image | |
* The start function will be executed when the currentTime | |
* of the video reaches the start time provided by the | |
* options variable | |
*/ | |
start: function( event, options ) { | |
options.link.style.display = "block"; | |
}, | |
/** | |
* @member image | |
* The end function will be executed when the currentTime | |
* of the video reaches the end time provided by the | |
* options variable | |
*/ | |
end: function( event, options ) { | |
options.link.style.display = "none"; | |
}, | |
_teardown: function( options ) { | |
document.getElementById( options.target ) && document.getElementById( options.target ).removeChild( options.link ); | |
} | |
}); | |
})( Popcorn ); | |
// PLUGIN: lowerthird | |
(function (Popcorn) { | |
/** | |
* Lower Third popcorn plug-in | |
* Displays information about a speaker over the video, or in the target div | |
* Options parameter will need a start, and end. | |
* Optional parameters are target, salutation, name and role. | |
* Start is the time that you want this plug-in to execute | |
* End is the time that you want this plug-in to stop executing | |
* Target is the id of the document element that the content is | |
* appended to, this target element must exist on the DOM | |
* salutation is the speaker's Mr. Ms. Dr. etc. | |
* name is the speaker's name. | |
* role is information about the speaker, example Engineer. | |
* | |
* @param {Object} options | |
* | |
* Example: | |
var p = Popcorn('#video') | |
.lowerthird({ | |
start: 5, // seconds, mandatory | |
end: 15, // seconds, mandatory | |
salutation: 'Mr', // optional | |
name: 'Scott Downe', // optional | |
role: 'Programmer', // optional | |
target: 'subtitlediv' // optional | |
} ) | |
* | |
*/ | |
Popcorn.plugin( "lowerthird" , { | |
manifest: { | |
about:{ | |
name: "Popcorn lowerthird Plugin", | |
version: "0.1", | |
author: "Scott Downe", | |
website: "http://scottdowne.wordpress.com/" | |
}, | |
options:{ | |
start : {elem:'input', type:'text', label:'In'}, | |
end : {elem:'input', type:'text', label:'Out'}, | |
target : 'lowerthird-container', | |
salutation : {elem:'input', type:'text', label:'Text'}, | |
name : {elem:'input', type:'text', label:'Text'}, | |
role : {elem:'input', type:'text', label:'Text'} | |
} | |
}, | |
_setup: function( options ) { | |
var target = document.getElementById( options.target ); | |
// Creates a div for all Lower Thirds to use | |
if ( !this.container ) { | |
this.container = document.createElement('div'); | |
this.container.style.position = "absolute"; | |
this.container.style.color = "white"; | |
this.container.style.textShadow = "black 2px 2px 6px"; | |
this.container.style.fontSize = "24px"; | |
this.container.style.fontWeight = "bold"; | |
this.container.style.paddingLeft = "40px"; | |
// the video element must have height and width defined | |
this.container.style.width = this.video.offsetWidth + "px"; | |
this.container.style.left = this.position().left + "px"; | |
this.video.parentNode.appendChild( this.container ); | |
} | |
// if a target is specified, use that | |
if ( options.target && options.target !== "lowerthird-container") { | |
options.container = document.createElement( "div" ); | |
if ( !target && Popcorn.plugin.debug ) { | |
throw new Error( "target container doesn't exist" ); | |
} | |
target && target.appendChild( options.container ); | |
} else { // use shared default container | |
options.container = this.container; | |
} | |
}, | |
/** | |
* @member lowerthird | |
* The start function will be executed when the currentTime | |
* of the video reaches the start time provided by the | |
* options variable | |
*/ | |
start: function(event, options){ | |
options.container.innerHTML = ( options.salutation ? options.salutation + " " : "" ) + options.name + ( options.role ? "<br />" + options.role : "" ); | |
this.container.style.top = this.position().top + this.video.offsetHeight - ( 40 + this.container.offsetHeight ) + "px"; | |
}, | |
/** | |
* @member lowerthird | |
* The end function will be executed when the currentTime | |
* of the video reaches the end time provided by the | |
* options variable | |
*/ | |
end: function(event, options){ | |
options.container.innerHTML = ""; | |
} | |
} ); | |
})( Popcorn ); | |
// PLUGIN: WEBPAGE | |
(function (Popcorn) { | |
/** | |
* Webpages popcorn plug-in | |
* Creates an iframe showing a website specified by the user | |
* Options parameter will need a start, end, id, target and src. | |
* Start is the time that you want this plug-in to execute | |
* End is the time that you want this plug-in to stop executing | |
* Id is the id that you want assigned to the iframe | |
* Target is the id of the document element that the iframe needs to be attached to, | |
* this target element must exist on the DOM | |
* Src is the url of the website that you want the iframe to display | |
* | |
* @param {Object} options | |
* | |
* Example: | |
var p = Popcorn('#video') | |
.webpage({ | |
id: "webpages-a", | |
start: 5, // seconds | |
end: 15, // seconds | |
src: 'http://www.webmademovies.org', | |
target: 'webpagediv' | |
} ) | |
* | |
*/ | |
Popcorn.plugin( "webpage" , { | |
manifest: { | |
about:{ | |
name: "Popcorn Webpage Plugin", | |
version: "0.1", | |
author: "@annasob", | |
website: "annasob.wordpress.com" | |
}, | |
options:{ | |
id : {elem:'input', type:'text', label:'Id'}, | |
start : {elem:'input', type:'text', label:'In'}, | |
end : {elem:'input', type:'text', label:'Out'}, | |
src : {elem:'input', type:'text', label:'Src'}, | |
target : 'iframe-container' | |
} | |
}, | |
_setup : function( options ) { | |
var target = document.getElementById( options.target ); | |
// make an iframe | |
options._iframe = document.createElement( 'iframe' ); | |
options._iframe.setAttribute('width', "100%"); | |
options._iframe.setAttribute('height', "100%"); | |
options._iframe.id = options.id; | |
options._iframe.src = options.src; | |
options._iframe.style.display = 'none'; | |
if ( !target && Popcorn.plugin.debug ) { | |
throw new Error( "target container doesn't exist" ); | |
} | |
// add the hidden iframe to the DOM | |
target && target.appendChild(options._iframe); | |
}, | |
/** | |
* @member webpage | |
* The start function will be executed when the currentTime | |
* of the video reaches the start time provided by the | |
* options variable | |
*/ | |
start: function(event, options){ | |
// make the iframe visible | |
options._iframe.src = options.src; | |
options._iframe.style.display = 'inline'; | |
}, | |
/** | |
* @member webpage | |
* The end function will be executed when the currentTime | |
* of the video reaches the end time provided by the | |
* options variable | |
*/ | |
end: function(event, options){ | |
// make the iframe invisible | |
options._iframe.style.display = 'none'; | |
}, | |
_teardown: function( options ) { | |
document.getElementById( options.target ) && document.getElementById( options.target ).removeChild( options._iframe ); | |
} | |
}); | |
})( Popcorn ); | |
// PLUGIN: Timeline | |
(function ( Popcorn ) { | |
/** | |
* timeline popcorn plug-in | |
* Adds data associated with a certain time in the video, which creates a scrolling view of each item as the video progresses | |
* Options parameter will need a start, target, title, and text | |
* -Start is the time that you want this plug-in to execute | |
* -End is the time that you want this plug-in to stop executing, tho for this plugin an end time may not be needed ( optional ) | |
* -Target is the id of the DOM element that you want the map to appear in. This element must be in the DOM | |
* -Title is the title of the current timeline box | |
* -Text is text is simply related text that will be displayed | |
* -innerHTML gives the user the option to add things such as links, buttons and so on | |
* -direction specifies whether the timeline will grow from the top or the bottom, receives input as "UP" or "DOWN" | |
* @param {Object} options | |
* | |
* Example: | |
var p = Popcorn("#video") | |
.timeline( { | |
start: 5, // seconds | |
target: "timeline", | |
title: "Seneca", | |
text: "Welcome to seneca", | |
innerHTML: "Click this link <a href='http://www.google.ca'>Google</a>" | |
} ) | |
* | |
*/ | |
var i = 1, | |
head = document.getElementsByTagName( "head" )[ 0 ], | |
css = document.createElement( "link" ); | |
css.type = "text/css"; | |
css.rel = "stylesheet"; | |
css.href = "//popcornjs.org/code/plugins/timeline/popcorn.timeline.css"; | |
head.insertBefore( css, head.firstChild ); | |
Popcorn.plugin( "timeline" , function( options ) { | |
var target = document.getElementById( options.target ), | |
newdiv = document.createElement( "div" ); | |
newdiv.style.display = "none"; | |
newdiv.id = "timelineDiv" + i; | |
// Default to up if options.direction is non-existant or not up or down | |
options.direction = options.direction || "up"; | |
if ( options.direction.toLowerCase() !== "up" || options.direction.toLowerCase() !== "down" ) { | |
options.direction = "up"; | |
} | |
if ( target ) { | |
target.style.width = "400px"; | |
target.style.height = "200px"; | |
target.style.overflow = "auto"; | |
target.appendChild( newdiv ); | |
// if this isnt the first div added to the target div | |
if( options.direction.length === 2 ){ | |
// insert the current div before the previous div inserted | |
target.insertBefore( newdiv, document.getElementById( "timelineDiv" + ( i - 1 ) ) ); | |
} | |
} else if ( Popcorn.plugin.debug ) { | |
throw new Error( "target container doesn't exist" ); | |
} | |
i++; | |
// Default to empty if not used | |
//options.innerHTML = options.innerHTML || ""; | |
newdiv.innerHTML = "<p><span id='big'>" + options.title + "</span><br />" + | |
"<span id='mid'>" + options.text + "</span><br />" + options.innerHTML; | |
return { | |
start: function( event, options ) { | |
newdiv.style.display = "block"; | |
if( options.direction === "down" ) { | |
target.scrollTop = target.scrollHeight; | |
} | |
}, | |
end: function( event, options ) { | |
newdiv.style.display = "none"; | |
}, | |
_teardown: function( options ) { | |
while ( target.firstChild ) { | |
target.removeChild( target.firstChild ); | |
} | |
} | |
}; | |
}, | |
{ | |
about: { | |
name: "Popcorn Timeline Plugin", | |
version: "0.1", | |
author: "David Seifried @dcseifried", | |
website: "dseifried.wordpress.com" | |
}, | |
options: { | |
start: { | |
elem: "input", | |
type: "text", | |
label: "In" | |
}, | |
end: { | |
elem: "input", | |
type: "text", | |
label: "Out" | |
}, | |
target: "feed-container", | |
title: { | |
elem: "input", | |
type: "text", | |
label: "title" | |
}, | |
text: { | |
elem: "input", | |
type: "text", | |
label: "text" | |
}, | |
innerHTML: { | |
elem: "input", | |
type: "text", | |
label: "innerHTML" | |
}, | |
direction: { | |
elem: "input", | |
type: "text", | |
label: "direction" | |
} | |
} | |
}); | |
})( Popcorn ); | |
// PLUGIN: Flickr | |
(function (Popcorn) { | |
/** | |
* Flickr popcorn plug-in | |
* Appends a users Flickr images to an element on the page. | |
* Options parameter will need a start, end, target and userid or username and api_key. | |
* Optional parameters are numberofimages, height, width, padding, and border | |
* Start is the time that you want this plug-in to execute (in seconds) | |
* End is the time that you want this plug-in to stop executing (in seconds) | |
* Userid is the id of who's Flickr images you wish to show | |
* Tags is a mutually exclusive list of image descriptor tags | |
* Username is the username of who's Flickr images you wish to show | |
* using both userid and username is redundant | |
* an api_key is required when using username | |
* Apikey is your own api key provided by Flickr | |
* Target is the id of the document element that the images are | |
* appended to, this target element must exist on the DOM | |
* Numberofimages specify the number of images to retreive from flickr, defaults to 4 | |
* Height the height of the image, defaults to '50px' | |
* Width the width of the image, defaults to '50px' | |
* Padding number of pixels between images, defaults to '5px' | |
* Border border size in pixels around images, defaults to '0px' | |
* | |
* @param {Object} options | |
* | |
* Example: | |
var p = Popcorn('#video') | |
.flickr({ | |
start: 5, // seconds, mandatory | |
end: 15, // seconds, mandatory | |
userid: '35034346917@N01', // optional | |
tags: 'dogs,cats', // optional | |
numberofimages: '8', // optional | |
height: '50px', // optional | |
width: '50px', // optional | |
padding: '5px', // optional | |
border: '0px', // optional | |
target: 'flickrdiv' // mandatory | |
} ) | |
* | |
*/ | |
var idx = 0; | |
Popcorn.plugin( "flickr" , function( options ) { | |
var containerDiv, | |
target = document.getElementById( options.target ), | |
_userid, | |
_uri, | |
_link, | |
_image, | |
_count = options.numberofimages || 4 , | |
_height = options.height || "50px", | |
_width = options.width || "50px", | |
_padding = options.padding || "5px", | |
_border = options.border || "0px"; | |
// create a new div this way anything in the target div is left intact | |
// this is later populated with Flickr images | |
containerDiv = document.createElement( "div" ); | |
containerDiv.id = "flickr" + idx; | |
containerDiv.style.width = "100%"; | |
containerDiv.style.height = "100%"; | |
containerDiv.style.display = "none"; | |
idx++; | |
// ensure the target container the user chose exists | |
if ( !target && Popcorn.plugin.debug ) { | |
throw new Error( "flickr target container doesn't exist" ); | |
} | |
target && target.appendChild( containerDiv ); | |
// get the userid from Flickr API by using the username and apikey | |
var isUserIDReady = function() { | |
if ( !_userid ) { | |
_uri = "http://api.flickr.com/services/rest/?method=flickr.people.findByUsername&"; | |
_uri += "username=" + options.username + "&api_key=" + options.apikey + "&format=json&jsoncallback=flickr"; | |
Popcorn.getJSONP( _uri, function( data ) { | |
_userid = data.user.nsid; | |
getFlickrData(); | |
}); | |
} else { | |
setTimeout(function () { | |
isUserIDReady(); | |
}, 5 ); | |
} | |
}; | |
// get the photos from Flickr API by using the user_id and/or tags | |
var getFlickrData = function() { | |
_uri = "http://api.flickr.com/services/feeds/photos_public.gne?"; | |
if ( _userid ) { | |
_uri += "id=" + _userid + "&"; | |
} | |
if ( options.tags ) { | |
_uri += "tags=" + options.tags + "&"; | |
} | |
_uri += "lang=en-us&format=json&jsoncallback=flickr"; | |
Popcorn.xhr.getJSONP( _uri, function( data ) { | |
var fragment = document.createElement( "p" ); | |
fragment.innerHTML = "<p style='padding:" + _padding + ";'>" + data.title + "<p/>"; | |
Popcorn.forEach( data.items, function ( item, i ) { | |
if ( i < _count ) { | |
_link = document.createElement( "a" ); | |
_link.setAttribute( "href", item.link ); | |
_link.setAttribute( "target", "_blank" ); | |
_image = document.createElement( "img" ); | |
_image.setAttribute( "src", item.media.m ); | |
_image.setAttribute( "height",_height ); | |
_image.setAttribute( "width", _width ); | |
_image.setAttribute( "style", "border:" + _border + ";padding:" + _padding ); | |
_link.appendChild( _image ); | |
fragment.appendChild( _link ); | |
} else { | |
return false; | |
} | |
}); | |
containerDiv.appendChild( fragment ); | |
}); | |
}; | |
if ( options.username && options.apikey ) { | |
isUserIDReady(); | |
} | |
else { | |
_userid = options.userid; | |
getFlickrData(); | |
} | |
return { | |
/** | |
* @member flickr | |
* The start function will be executed when the currentTime | |
* of the video reaches the start time provided by the | |
* options variable | |
*/ | |
start: function( event, options ) { | |
containerDiv.style.display = "inline"; | |
}, | |
/** | |
* @member flickr | |
* The end function will be executed when the currentTime | |
* of the video reaches the end time provided by the | |
* options variable | |
*/ | |
end: function( event, options ) { | |
containerDiv.style.display = "none"; | |
}, | |
_teardown: function( options ) { | |
document.getElementById( options.target ) && document.getElementById( options.target ).removeChild( containerDiv ); | |
} | |
}; | |
}, | |
{ | |
about: { | |
name: "Popcorn Flickr Plugin", | |
version: "0.2", | |
author: "Scott Downe, Steven Weerdenburg, Annasob", | |
website: "http://scottdowne.wordpress.com/" | |
}, | |
options: { | |
start: { | |
elem: "input", | |
type: "number", | |
label: "In" | |
}, | |
end: { | |
elem: "input", | |
type: "number", | |
label: "Out" | |
}, | |
userid: { | |
elem: "input", | |
type: "text", | |
label: "UserID" | |
}, | |
tags: { | |
elem: "input", | |
type: "text", | |
label: "Tags" | |
}, | |
username: { | |
elem: "input", | |
type: "text", | |
label: "Username" | |
}, | |
apikey: { | |
elem: "input", | |
type: "text", | |
label: "Api_key" | |
}, | |
target: "flickr-container", | |
height: { | |
elem: "input", | |
type: "text", | |
label: "Height" | |
}, | |
width: { | |
elem: "input", | |
type: "text", | |
label: "Width" | |
}, | |
padding: { | |
elem: "input", | |
type: "text", | |
label: "Padding" | |
}, | |
border: { | |
elem: "input", | |
type: "text", | |
label: "Border" | |
}, | |
numberofimages: { | |
elem: "input", | |
type: "text", | |
label: "Number of Images" | |
} | |
} | |
}); | |
})( Popcorn ); | |
// PLUGIN: WIKIPEDIA | |
var wikiCallback; | |
(function ( Popcorn ) { | |
/** | |
* Wikipedia popcorn plug-in | |
* Displays a wikipedia aricle in the target specified by the user by using | |
* new DOM element instead overwriting them | |
* Options parameter will need a start, end, target, lang, src, title and numberofwords. | |
* -Start is the time that you want this plug-in to execute | |
* -End is the time that you want this plug-in to stop executing | |
* -Target is the id of the document element that the text from the article needs to be | |
* attached to, this target element must exist on the DOM | |
* -Lang (optional, defaults to english) is the language in which the article is in. | |
* -Src is the url of the article | |
* -Title (optional) is the title of the article | |
* -numberofwords (optional, defaults to 200) is the number of words you want displaid. | |
* | |
* @param {Object} options | |
* | |
* Example: | |
var p = Popcorn("#video") | |
.wikipedia({ | |
start: 5, // seconds | |
end: 15, // seconds | |
src: "http://en.wikipedia.org/wiki/Cape_Town", | |
target: "wikidiv" | |
} ) | |
* | |
*/ | |
Popcorn.plugin( "wikipedia" , { | |
manifest: { | |
about:{ | |
name: "Popcorn Wikipedia Plugin", | |
version: "0.1", | |
author: "@annasob", | |
website: "annasob.wordpress.com" | |
}, | |
options:{ | |
start: { | |
elem: "input", | |
type: "text", | |
label: "In" | |
}, | |
end: { | |
elem: "input", | |
type: "text", | |
label: "Out" | |
}, | |
lang: { | |
elem: "input", | |
type: "text", | |
label: "Language" | |
}, | |
src: { | |
elem: "input", | |
type: "text", | |
label: "Src" | |
}, | |
title: { | |
elem: "input", | |
type: "text", | |
label: "Title" | |
}, | |
numberofwords: { | |
elem: "input", | |
type: "text", | |
label: "Num Of Words" | |
}, | |
target: "wikipedia-container" | |
} | |
}, | |
/** | |
* @member wikipedia | |
* The setup function will get all of the needed | |
* items in place before the start function is called. | |
* This includes getting data from wikipedia, if the data | |
* is not received and processed before start is called start | |
* will not do anything | |
*/ | |
_setup : function( options ) { | |
// declare needed variables | |
// get a guid to use for the global wikicallback function | |
var _text, _guid = Popcorn.guid(); | |
// if the user didn't specify a language default to english | |
if ( !options.lang ) { | |
options.lang = "en"; | |
} | |
// if the user didn't specify number of words to use default to 200 | |
options.numberofwords = options.numberofwords || 200; | |
// wiki global callback function with a unique id | |
// function gets the needed information from wikipedia | |
// and stores it by appending values to the options object | |
window[ "wikiCallback" + _guid ] = function ( data ) { | |
options._link = document.createElement( "a" ); | |
options._link.setAttribute( "href", options.src ); | |
options._link.setAttribute( "target", "_blank" ); | |
// add the title of the article to the link | |
options._link.innerHTML = options.title || data.parse.displaytitle; | |
// get the content of the wiki article | |
options._desc = document.createElement( "p" ); | |
// get the article text and remove any special characters | |
_text = data.parse.text[ "*" ].substr( data.parse.text[ "*" ].indexOf( "<p>" ) ); | |
_text = _text.replace( /((<(.|\n)+?>)|(\((.*?)\) )|(\[(.*?)\]))/g, "" ); | |
options._desc.innerHTML = _text.substr( 0, options.numberofwords ) + " ..."; | |
options._fired = true; | |
}; | |
if ( options.src ) { | |
Popcorn.getScript( "http://" + options.lang + ".wikipedia.org/w/api.php?action=parse&props=text&page=" + | |
options.src.slice( options.src.lastIndexOf("/")+1) + "&format=json&callback=wikiCallback" + _guid); | |
} else if ( Popcorn.plugin.debug ) { | |
throw new Error( "Wikipedia plugin needs a 'src'" ); | |
} | |
}, | |
/** | |
* @member wikipedia | |
* The start function will be executed when the currentTime | |
* of the video reaches the start time provided by the | |
* options variable | |
*/ | |
start: function( event, options ){ | |
// dont do anything if the information didn't come back from wiki | |
var isReady = function () { | |
if ( !options._fired ) { | |
setTimeout( function () { | |
isReady(); | |
}, 13); | |
} else { | |
if ( options._link && options._desc ) { | |
if ( document.getElementById( options.target ) ) { | |
document.getElementById( options.target ).appendChild( options._link ); | |
document.getElementById( options.target ).appendChild( options._desc ); | |
options._added = true; | |
} | |
} | |
} | |
}; | |
isReady(); | |
}, | |
/** | |
* @member wikipedia | |
* The end function will be executed when the currentTime | |
* of the video reaches the end time provided by the | |
* options variable | |
*/ | |
end: function( event, options ){ | |
// ensure that the data was actually added to the | |
// DOM before removal | |
if ( options._added ) { | |
document.getElementById( options.target ).removeChild( options._link ); | |
document.getElementById( options.target ).removeChild( options._desc ); | |
} | |
}, | |
_teardown: function( options ){ | |
if ( options._added ) { | |
options._link.parentNode && document.getElementById( options.target ).removeChild( options._link ); | |
options._desc.parentNode && document.getElementById( options.target ).removeChild( options._desc ); | |
delete options.target; | |
} | |
} | |
}); | |
})( Popcorn ); | |
// PLUGIN: GML | |
(function (Popcorn) { | |
var gmlPlayer = function( $p ) { | |
var _stroke = 0, | |
onPt = 0, | |
onStroke = 0, | |
x = null, | |
y = null, | |
rotation = false, | |
strokes = 0, | |
play = function() {}, | |
reset = function() { | |
$p.background( 0 ); | |
onPt = onStroke = 0; | |
x = y = null; | |
}, | |
drawLine = function( x, y, x2, y2 ) { | |
var _x, _y, _x2, _y2; | |
if ( rotation ) { | |
_x = y * $p.height; | |
_y = $p.width - ( x * $p.width ); | |
_x2 = y2 * $p.height; | |
_y2 = $p.width - ( x2 * $p.width ); | |
} else { | |
_x = x * $p.width; | |
_y = y * $p.height; | |
_x2 = x2 * $p.width; | |
_y2 = y2 * $p.height; | |
} | |
$p.stroke( 0 ); | |
$p.strokeWeight( 13 ); | |
$p.strokeCap( $p.SQUARE ); | |
$p.line( _x, _y, _x2, _y2 ); | |
$p.stroke( 255 ); | |
$p.strokeWeight( 12 ); | |
$p.strokeCap( $p.ROUND ); | |
$p.line( _x, _y, _x2, _y2 ); | |
}, | |
seek = function( point ) { | |
( point < onPt ) && reset(); | |
while ( onPt <= point ) { | |
if ( !strokes ) { | |
return; | |
} | |
_stroke = strokes[ onStroke ] || strokes; | |
var pt = _stroke.pt[ onPt ], | |
p = onPt; | |
x != null && drawLine( x, y, pt.x, pt.y ); | |
x = pt.x; | |
y = pt.y; | |
( onPt === p ) && onPt++; | |
} | |
}; | |
$p.draw = function() { | |
play(); | |
}; | |
$p.setup = function() {}; | |
$p.construct = function( media, data, options ) { | |
var dataReady = function() { | |
if ( data ) { | |
strokes = data.gml.tag.drawing.stroke; | |
var drawingDur = ( options.end - options.start ) / ( strokes.pt || (function( strokes ) { | |
var rStrokes = []; | |
for ( var i = 0, sl = strokes.length; i < sl; i++ ) { | |
rStrokes = rStrokes.concat( strokes[ i ].pt ); | |
} | |
return rStrokes; | |
})( strokes ) ).length; | |
var tag = data.gml.tag, | |
app_name = tag.header && tag.header.client && tag.header.client.name; | |
rotation = app_name === 'Graffiti Analysis 2.0: DustTag' || | |
app_name === 'DustTag: Graffiti Analysis 2.0' || | |
app_name === 'Fat Tag - Katsu Edition'; | |
play = function() { | |
if ( media.currentTime < options.endDrawing ) { | |
seek( ( media.currentTime - options.start ) / drawingDur ); | |
} | |
}; | |
return; | |
} | |
setTimeout( dataReady, 5 ); | |
}; | |
$p.size( 640, 640 ); | |
$p.frameRate( 60 ); | |
$p.smooth(); | |
reset(); | |
$p.noLoop(); | |
dataReady(); | |
}; | |
}; | |
/** | |
* Grafiti markup Language (GML) popcorn plug-in | |
* Renders a GML tag inside an HTML element | |
* Options parameter will need a mandatory start, end, target, gmltag. | |
* Optional parameters: none. | |
* Start is the time that you want this plug-in to execute | |
* End is the time that you want this plug-in to stop executing | |
* Target is the id of the document element that you wish to render the grafiti in | |
* gmltag is the numerical reference to a gml tag via 000000book.com | |
* @param {Object} options | |
* | |
* Example: | |
var p = Popcorn('#video') | |
.gml({ | |
start: 0, // seconds | |
end: 5, // seconds | |
gmltag: '29582', | |
target: 'gmldiv' | |
}); | |
* | |
*/ | |
Popcorn.plugin( "gml" , { | |
_setup : function( options ) { | |
var self = this, | |
target = document.getElementById( options.target ); | |
options.endDrawing = options.endDrawing || options.end; | |
// create a canvas to put in the target div | |
options.container = document.createElement( "canvas" ); | |
options.container.style.display = "none"; | |
options.container.setAttribute( "id", "canvas" + options.gmltag ); | |
if ( !target && Popcorn.plugin.debug ) { | |
throw new Error( "target container doesn't exist" ); | |
} | |
target && target.appendChild( options.container ); | |
if ( !window.Processing ) { | |
Popcorn.getScript( "http://processingjs.org/content/download/processing-js-1.2.1/processing-1.2.1.min.js" ); | |
} | |
// makes sure both processing.js and the gml data are loaded | |
var readyCheck = function() { | |
if ( window.Processing ) { | |
Popcorn.getJSONP( "http://000000book.com/data/" + options.gmltag + ".json?callback=", function( data ) { | |
options.pjsInstance = new Processing( options.container, gmlPlayer ); | |
options.pjsInstance.construct( self.media, data, options ); | |
options._running && options.pjsInstance.loop(); | |
}, false ); | |
return; | |
} | |
setTimeout( readyCheck, 5 ); | |
}; | |
readyCheck(); | |
}, | |
/** | |
* @member gml | |
* The start function will be executed when the currentTime | |
* of the video reaches the start time provided by the | |
* options variable | |
*/ | |
start: function( event, options ) { | |
options.pjsInstance && options.pjsInstance.loop(); | |
options.container.style.display = "block"; | |
}, | |
/** | |
* @member gml | |
* The end function will be executed when the currentTime | |
* of the video reaches the end time provided by the | |
* options variable | |
*/ | |
end: function( event, options ) { | |
options.pjsInstance && options.pjsInstance.noLoop(); | |
options.container.style.display = "none"; | |
}, | |
_teardown: function( options ) { | |
options.pjsInstance && options.pjsInstance.exit(); | |
document.getElementById( options.target ) && document.getElementById( options.target ).removeChild( options.container ); | |
} | |
}); | |
})( Popcorn ); | |
// PLUGIN: Google Maps | |
var googleCallback; | |
(function (Popcorn) { | |
var i = 1, | |
_mapFired = false, | |
_mapLoaded = false, | |
geocoder, loadMaps; | |
//google api callback | |
googleCallback = function (data) { | |
// ensure all of the maps functions needed are loaded | |
// before setting _maploaded to true | |
if (typeof google !== "undefined" && google.maps && google.maps.Geocoder && google.maps.LatLng) { | |
geocoder = new google.maps.Geocoder(); | |
_mapLoaded = true; | |
} else { | |
setTimeout(function () { | |
googleCallback(data); | |
}, 1); | |
} | |
}; | |
// function that loads the google api | |
loadMaps = function () { | |
// for some reason the Google Map API adds content to the body | |
if (document.body) { | |
_mapFired = true; | |
Popcorn.getScript("http://maps.google.com/maps/api/js?sensor=false&callback=googleCallback"); | |
} else { | |
setTimeout(function () { | |
loadMaps(); | |
}, 1); | |
} | |
}; | |
/** | |
* googlemap popcorn plug-in | |
* Adds a map to the target div centered on the location specified by the user | |
* Options parameter will need a start, end, target, type, zoom, lat and lng, and location | |
* -Start is the time that you want this plug-in to execute | |
* -End is the time that you want this plug-in to stop executing | |
* -Target is the id of the DOM element that you want the map to appear in. This element must be in the DOM | |
* -Type [optional] either: HYBRID (default), ROADMAP, SATELLITE, TERRAIN, STREETVIEW | |
* -Zoom [optional] defaults to 0 | |
* -Heading [optional] STREETVIEW orientation of camera in degrees relative to true north (0 north, 90 true east, ect) | |
* -Pitch [optional] STREETVIEW vertical orientation of the camera (between 1 and 3 is recommended) | |
* -Lat and Lng: the coordinates of the map must be present if location is not specified. | |
* -Location: the adress you want the map to display, must be present if lat and lng are not specified. | |
* Note: using location requires extra loading time, also not specifying both lat/lng and location will | |
* cause and error. | |
* | |
* Tweening works using the following specifications: | |
* -location is the start point when using an auto generated route | |
* -tween when used in this context is a string which specifies the end location for your route | |
* Note that both location and tween must be present when using an auto generated route, or the map will not tween | |
* -interval is the speed in which the tween will be executed, a reasonable time is 1000 ( time in milliseconds ) | |
* Heading, Zoom, and Pitch streetview values are also used in tweening with the autogenerated route | |
* | |
* -tween is an array of objects, each containing data for one frame of a tween | |
* -position is an object with has two paramaters, lat and lng, both which are mandatory for a tween to work | |
* -pov is an object which houses heading, pitch, and zoom paramters, which are all optional, if undefined, these values default to 0 | |
* -interval is the speed in which the tween will be executed, a reasonable time is 1000 ( time in milliseconds ) | |
* | |
* @param {Object} options | |
* | |
* Example: | |
var p = Popcorn("#video") | |
.googlemap({ | |
start: 5, // seconds | |
end: 15, // seconds | |
type: "ROADMAP", | |
target: "map", | |
lat: 43.665429, | |
lng: -79.403323 | |
} ) | |
* | |
*/ | |
Popcorn.plugin("googlemap", function (options) { | |
var newdiv, map, location, | |
target = document.getElementById( options.target ); | |
// if this is the firest time running the plugins | |
// call the function that gets the sctipt | |
if (!_mapFired) { | |
loadMaps(); | |
} | |
// create a new div this way anything in the target div is left intact | |
// this is later passed on to the maps api | |
newdiv = document.createElement("div"); | |
newdiv.id = "actualmap" + i; | |
newdiv.style.width = "100%"; | |
newdiv.style.height = "100%"; | |
i++; | |
// ensure the target container the user chose exists | |
if ( !target && Popcorn.plugin.debug ) { | |
throw new Error( "target container doesn't exist" ); | |
} | |
target && target.appendChild( newdiv ); | |
// ensure that google maps and its functions are loaded | |
// before setting up the map parameters | |
var isMapReady = function () { | |
if (_mapLoaded) { | |
if (options.location) { | |
// calls an anonymous google function called on separate thread | |
geocoder.geocode({ | |
"address": options.location | |
}, function (results, status) { | |
if (status === google.maps.GeocoderStatus.OK) { | |
options.lat = results[0].geometry.location.lat(); | |
options.lng = results[0].geometry.location.lng(); | |
location = new google.maps.LatLng(options.lat, options.lng); | |
map = new google.maps.Map(newdiv, { | |
mapTypeId: google.maps.MapTypeId[options.type] || google.maps.MapTypeId.HYBRID | |
}); | |
map.getDiv().style.display = "none"; | |
} | |
}); | |
} else { | |
location = new google.maps.LatLng(options.lat, options.lng); | |
map = new google.maps.Map(newdiv, { | |
mapTypeId: google.maps.MapTypeId[options.type] || google.maps.MapTypeId.HYBRID | |
}); | |
map.getDiv().style.display = "none"; | |
} | |
} else { | |
setTimeout(function () { | |
isMapReady(); | |
}, 5); | |
} | |
}; | |
isMapReady(); | |
return { | |
/** | |
* @member webpage | |
* The start function will be executed when the currentTime | |
* of the video reaches the start time provided by the | |
* options variable | |
*/ | |
start: function( event, options ) { | |
var that = this, | |
sView; | |
// ensure the map has been initialized in the setup function above | |
var isMapSetup = function() { | |
if ( map ) { | |
map.getDiv().style.display = "block"; | |
// reset the location and zoom just in case the user plaid with the map | |
google.maps.event.trigger(map, 'resize'); | |
map.setCenter( location ); | |
// make sure options.zoom is a number | |
if ( options.zoom && typeof options.zoom !== "number" ) { | |
options.zoom = +options.zoom; | |
} | |
options.zoom = options.zoom || 8; // default to 8 | |
map.setZoom( options.zoom ); | |
//Make sure heading is a number | |
if ( options.heading && typeof options.heading !== "number" ) { | |
options.heading = +options.heading; | |
} | |
//Make sure pitch is a number | |
if ( options.pitch && typeof options.pitch !== "number" ) { | |
options.pitch = +options.pitch; | |
} | |
if ( options.type === "STREETVIEW" ) { | |
// Switch this map into streeview mode | |
map.setStreetView( | |
// Pass a new StreetViewPanorama instance into our map | |
sView = new google.maps.StreetViewPanorama( newdiv, { | |
position: location, | |
pov: { | |
heading: options.heading = options.heading || 0, | |
pitch: options.pitch = options.pitch || 0, | |
zoom: options.zoom | |
} | |
}) | |
); | |
// Function to handle tweening using a set timeout | |
var tween = function( rM, t ) { | |
setTimeout(function() { | |
// Checks whether this is a generated route or not | |
if( typeof options.tween === "object" ) { | |
for ( var i = 0; i < rM.length; i++ ) { | |
// Checks if this position along the tween should be displayed or not | |
if ( that.media.currentTime >= ( rM[ i ].interval * ( i + 1 ) ) / 1000 && | |
( that.media.currentTime <= ( rM[ i ].interval * ( i + 2 ) ) / 1000 || | |
that.media.currentTime >= rM[ i ].interval * ( rM.length ) / 1000 ) ) { | |
sView3.setPosition( new google.maps.LatLng( rM[ i ].position.lat, rM[ i ].position.lng ) ); | |
sView3.setPov({ | |
heading: rM[ i ].pov.heading || 0, | |
zoom: rM[ i ].pov.zoom || 0, | |
pitch: rM[ i ].pov.pitch || 0 | |
}); | |
} | |
} | |
// Calls the tween function again at the interval set by the user | |
tween( rM, rM[ 0 ].interval ); | |
} else { | |
for ( var k = 0; k < rM.length; k++ ) { | |
if( that.media.currentTime >= (options.interval * ( k + 1 ) ) / 1000 && | |
( that.media.currentTime <= (options.interval * ( k + 2 ) ) / 1000 || | |
that.media.currentTime >= options.interval * ( rM.length ) / 1000 ) ) { | |
sView2.setPosition( checkpoints[ k ] ); | |
sView2.setPov({ | |
heading: options.heading || 0, | |
zoom: options.zoom, | |
pitch: options.pitch || 0 | |
}); | |
} | |
} | |
tween( checkpoints, options.interval ); | |
} | |
}, t ); | |
}; | |
// Determines if we should use hardcoded values ( using options.tween ), | |
// or if we should use a start and end location and let google generate | |
// the route for us | |
if ( options.location && typeof options.tween === "string" ) { | |
// Creating another variable to hold the streetview map for tweening, | |
// Doing this because if there was more then one streetview map, the tweening would sometimes appear in other maps | |
var sView2 = sView; | |
// Create an array to store all the lat/lang values along our route | |
var checkpoints = []; | |
// Creates a new direction service, later used to create a route | |
var directionsService = new google.maps.DirectionsService(); | |
// Creates a new direction renderer using the current map | |
// This enables us to access all of the route data that is returned to us | |
var directionsDisplay = new google.maps.DirectionsRenderer( sView2 ); | |
var request = { | |
origin: options.location, | |
destination: options.tween, | |
travelMode: google.maps.TravelMode.DRIVING | |
}; | |
// Create the route using the direction service and renderer | |
directionsService.route( request, function( response, status ) { | |
if ( status == google.maps.DirectionsStatus.OK ) { | |
directionsDisplay.setDirections( response ); | |
showSteps( response, that ); | |
} | |
}); | |
var showSteps = function ( directionResult, that ) { | |
// Push new google map lat and lng values into an array from our list of lat and lng values | |
for ( var j = 0; j < directionResult.routes[ 0 ].overview_path.length; j++ ) { | |
checkpoints.push( new google.maps.LatLng( directionResult.routes[ 0 ].overview_path[ j ].lat(), directionResult.routes[ 0 ].overview_path[ j ].lng() ) ); | |
} | |
// Check to make sure the interval exists, if not, set to a default of 1000 | |
options.interval = options.interval || 1000; | |
tween( checkpoints, 10 ); | |
}; | |
} else if ( typeof options.tween === "object" ) { | |
// Same as the above to stop streetview maps from overflowing into one another | |
var sView3 = sView; | |
for ( var i = 0; i < options.tween.length; i++ ) { | |
// Make sure interval exists, if not, set to 1000 | |
options.tween[ i ].interval = options.tween[ i ].interval || 1000; | |
tween( options.tween, 10 ); | |
} | |
} | |
} | |
} else { | |
setTimeout(function () { | |
isMapSetup(); | |
}, 13); | |
} | |
}; | |
isMapSetup(); | |
}, | |
/** | |
* @member webpage | |
* The end function will be executed when the currentTime | |
* of the video reaches the end time provided by the | |
* options variable | |
*/ | |
end: function (event, options) { | |
// if the map exists hide it do not delete the map just in | |
// case the user seeks back to time b/w start and end | |
if (map) { | |
map.getDiv().style.display = "none"; | |
} | |
}, | |
_teardown: function (options) { | |
// the map must be manually removed | |
document.getElementById(options.target).removeChild(newdiv); | |
newdiv = map = location = null; | |
} | |
}; | |
}, { | |
about: { | |
name: "Popcorn Google Map Plugin", | |
version: "0.1", | |
author: "@annasob", | |
website: "annasob.wordpress.com" | |
}, | |
options: { | |
start: { | |
elem: "input", | |
type: "text", | |
label: "In" | |
}, | |
end: { | |
elem: "input", | |
type: "text", | |
label: "Out" | |
}, | |
target: "map-container", | |
type: { | |
elem: "select", | |
options: ["ROADMAP", "SATELLITE", "STREETVIEW", "HYBRID", "TERRAIN"], | |
label: "Type" | |
}, | |
zoom: { | |
elem: "input", | |
type: "text", | |
label: "Zoom" | |
}, | |
lat: { | |
elem: "input", | |
type: "text", | |
label: "Lat" | |
}, | |
lng: { | |
elem: "input", | |
type: "text", | |
label: "Lng" | |
}, | |
location: { | |
elem: "input", | |
type: "text", | |
label: "Location" | |
}, | |
heading: { | |
elem: "input", | |
type: "text", | |
label: "Heading" | |
}, | |
pitch: { | |
elem: "input", | |
type: "text", | |
label: "Pitch" | |
} | |
} | |
}); | |
})(Popcorn); | |
/** | |
* Pause Popcorn Plug-in | |
* | |
* When this plugin is used, links on the webpage, when clicked, will pause | |
* popcorn videos that especified 'pauseOnLinkClicked' as an option. Links may | |
* cause a new page to display on a new window, or may cause a new page to | |
* display in the current window, in which case the videos won't be available | |
* anymore. It only affects anchor tags. It does not affect objects with click | |
* events that act as anchors. | |
* | |
* Example: | |
var p = Popcorn('#video', { pauseOnLinkClicked : true } ) | |
.play(); | |
* | |
*/ | |
document.addEventListener( "click", function( event ) { | |
var targetElement = event.target; | |
//Some browsers use an element as the target, some use the text node inside | |
if ( targetElement.nodeName === "A" || targetElement.parentNode && targetElement.parentNode.nodeName === "A" ) { | |
Popcorn.instances.forEach( function( video ) { | |
if ( video.options.pauseOnLinkClicked ) { | |
video.pause(); | |
} | |
}); | |
} | |
}, false ); | |
// PLUGIN: TWITTER | |
(function (Popcorn) { | |
var scriptLoading = false; | |
/** | |
* Twitter popcorn plug-in | |
* Appends a Twitter widget to an element on the page. | |
* Options parameter will need a start, end, target and source. | |
* Optional parameters are height and width. | |
* Start is the time that you want this plug-in to execute | |
* End is the time that you want this plug-in to stop executing | |
* Src is the hash tag or twitter user to get tweets from | |
* Target is the id of the document element that the images are | |
* appended to, this target element must exist on the DOM | |
* Height is the height of the widget, defaults to 200 | |
* Width is the width of the widget, defaults to 250 | |
* | |
* @param {Object} options | |
* | |
* Example: | |
var p = Popcorn('#video') | |
.twitter({ | |
start: 5, // seconds, mandatory | |
end: 15, // seconds, mandatory | |
src: '@stevesong', // mandatory, also accepts hash tags | |
height: 200, // optional | |
width: 250, // optional | |
target: 'twitterdiv' // mandatory | |
} ) | |
* | |
*/ | |
Popcorn.plugin( "twitter" , { | |
manifest: { | |
about:{ | |
name: "Popcorn Twitter Plugin", | |
version: "0.1", | |
author: "Scott Downe", | |
website: "http://scottdowne.wordpress.com/" | |
}, | |
options:{ | |
start : {elem:'input', type:'number', label:'In'}, | |
end : {elem:'input', type:'number', label:'Out'}, | |
src : {elem:'input', type:'text', label:'Source'}, | |
target : 'twitter-container', | |
height : {elem:'input', type:'number', label:'Height'}, | |
width : {elem:'input', type:'number', label:'Width'} | |
} | |
}, | |
_setup: function( options ) { | |
if ( !window.TWTR && !scriptLoading ) { | |
scriptLoading = true; | |
Popcorn.getScript("http://widgets.twimg.com/j/2/widget.js"); | |
} | |
var target = document.getElementById( options.target ); | |
// setup widget div that is unique per track | |
options.container = document.createElement( 'div' ); // create the div to store the widget | |
options.container.setAttribute('id', Popcorn.guid()); // use this id to connect it to the widget | |
options.container.style.display = "none"; // display none by default | |
if ( !target && Popcorn.plugin.debug ) { | |
throw new Error( "target container doesn't exist" ); | |
} | |
// add the widget's div to the target div | |
target && target.appendChild( options.container ); | |
// setup info for the widget | |
var src = options.src || "", | |
width = options.width || 250, | |
height = options.height || 200, | |
profile = /^@/.test( src ), | |
hash = /^#/.test( src ), | |
widgetOptions = { | |
version: 2, | |
id: options.container.getAttribute( 'id' ), // use this id to connect it to the div | |
rpp: 30, | |
width: width, | |
height: height, | |
interval: 6000, | |
theme: { | |
shell: { | |
background: '#ffffff', | |
color: '#000000' | |
}, | |
tweets: { | |
background: '#ffffff', | |
color: '#444444', | |
links: '#1985b5' | |
} | |
}, | |
features: { | |
loop: true, | |
timestamp: true, | |
avatars: true, | |
hashtags: true, | |
toptweets: true, | |
live: true, | |
scrollbar: false, | |
behavior: 'default' | |
} | |
}; | |
// create widget | |
var isReady = function( that ) { | |
if ( window.TWTR ) { | |
if ( profile ) { | |
widgetOptions.type = "profile"; | |
new TWTR.Widget( widgetOptions ).render().setUser( src ).start(); | |
} else if ( hash ) { | |
widgetOptions.type = "search"; | |
widgetOptions.search = src; | |
widgetOptions.subject = src; | |
new TWTR.Widget( widgetOptions ).render().start(); | |
} | |
} else { | |
setTimeout( function() { | |
isReady( that ); | |
}, 1); | |
} | |
}; | |
isReady( this ); | |
}, | |
/** | |
* @member Twitter | |
* The start function will be executed when the currentTime | |
* of the video reaches the start time provided by the | |
* options variable | |
*/ | |
start: function( event, options ) { | |
options.container.style.display = "inline"; | |
}, | |
/** | |
* @member Twitter | |
* The end function will be executed when the currentTime | |
* of the video reaches the end time provided by the | |
* options variable | |
*/ | |
end: function( event, options ) { | |
options.container.style.display = "none"; | |
}, | |
_teardown: function( options ) { | |
document.getElementById( options.target ) && document.getElementById( options.target ).removeChild( options.container ); | |
} | |
}); | |
})( Popcorn ); | |
/** | |
* Processing Popcorn Plug-In | |
* | |
* This plugin adds a Processing.js sketch to be added to a target div or canvas. | |
* | |
* Options parameter needs to specify start, end, target and sketch attributes | |
* -Start is the time [in seconds] that you want the sketch to display and start looping. | |
* -End is the time [in seconds] you want the sketch to become hidden and stop looping. | |
* -Target is the id of the div or canvas you want the target sketch to be displayed in. ( a target that is a div will have a canvas created and placed inside of it. ) | |
* -Sketch specifies the filename of the Procesing code to be loaded into Processing.js | |
* -noLoop [optional] specifies whether a sketch should continue to loop when the video is paused or seeking. | |
* | |
* @param {Object} options | |
* | |
* Example: | |
var p = Popcorn( "#video" ) | |
.processing({ | |
start: 5, | |
end: 10, | |
target: "processing-div", | |
sketch: "processingSketch.pjs", | |
noLoop: true | |
}); | |
* | |
*/ | |
(function ( Popcorn ) { | |
Popcorn.plugin( "processing" , function ( options ) { | |
var init = function( context ) { | |
var initProcessing, | |
canvas; | |
if ( !window.Processing ) { | |
Popcorn.getScript( "http://processingjs.org/content/download/processing-js-1.2.1/processing-1.2.1.min.js" ); | |
} | |
options.parentTarget = document.getElementById( options.target ); | |
if ( !options.parentTarget && Popcorn.plugin.debug ) { | |
throw new Error( "target container doesn't exist" ); | |
} | |
initProcessing = function() { | |
var addListeners = function() { | |
context.listen( "pause", function () { | |
if ( options.canvas.style.display === "inline" ) { | |
options.pjsInstance.noLoop(); | |
} | |
}); | |
context.listen( "play", function() { | |
if ( options.canvas.style.display === "inline" ) { | |
options.pjsInstance.loop(); | |
} | |
}); | |
}; | |
if ( window.Processing ) { | |
options.pjsInstance = new Processing( options.canvas, options.processingCode ); | |
options.pjsInstance.noLoop(); | |
options.seeking = false; | |
context.listen( "seeking", function() { | |
options._running && options.canvas.style.display === "inline" && options.noPause && options.pjsInstance.loop(); | |
}); | |
options._running && !context.media.paused && options.pjsInstance.loop(); | |
options.noPause = options.noPause || false; | |
!options.noPause && addListeners(); | |
options.codeReady = true; | |
} else { | |
setTimeout ( function() { | |
initProcessing.call( this ); | |
}, 10 ); | |
} | |
}; | |
canvas = document.createElement( "canvas" ); | |
canvas.id = Popcorn.guid( options.target + "-sketch-" ); | |
canvas[ "data-processing-sources" ] = options.sketch; | |
canvas.style.display = "none"; | |
options.canvas = canvas; | |
options.parentTarget && options.parentTarget.appendChild( options.canvas ); | |
if ( options.sketch ) { | |
Popcorn.xhr({ | |
url: options.sketch, | |
dataType: "text", | |
success: function( responseCode ) { | |
options.codeReady = true; | |
options.processingCode = responseCode; | |
initProcessing(); | |
} | |
}); | |
} else { | |
if ( Popcorn.plugin.debug ) { | |
throw new Error( "options.sketch is undefined" ); | |
} | |
} | |
}; | |
return { | |
manifest: { | |
about: { | |
name: "Popcorn Processing Plugin", | |
version: "0.1", | |
author: "Christopher De Cairos, Benjamin Chalovich", | |
website: "cadecairos.blogspot.com, ben1amin.wordpress.org" | |
}, | |
options: { | |
start : { elem: "input", type: "text", label: "In" }, | |
end : { elem: "input", type: "text", label: "Out" }, | |
target : { elem: "input", type: "text", label: "Target" }, | |
sketch : { elem: "input", type: "text", label: "Sketch" }, | |
noPause : { elem: "select", options: [ "TRUE", "FALSE" ], label: "No Loop" } | |
} | |
}, | |
_setup: function( options ) { | |
options.codeReady = false; | |
init( this ); | |
}, | |
start: function( event, options ) { | |
options.codeReady && !this.media.paused && options.pjsInstance.loop(); | |
options.canvas.style.display = "inline"; | |
}, | |
end: function( event, options ) { | |
options.pjsInstance && options.pjsInstance.noLoop(); | |
options.canvas.style.display = "none"; | |
}, | |
_teardown: function( options ) { | |
options.pjsInstance && options.pjsInstance.exit(); | |
options.parentTarget && options.parentTarget.removeChild( options.canvas ); | |
} | |
}; | |
}); | |
}( Popcorn )); | |
// PLUGIN: Code | |
(function (Popcorn) { | |
/** | |
* Code Popcorn Plug-in | |
* | |
* Adds the ability to run arbitrary code (JavaScript functions) according to video timing. | |
* | |
* @param {Object} options | |
* | |
* Required parameters: start, end, template, data, and target. | |
* Optional parameter: static. | |
* | |
* start: the time in seconds when the mustache template should be rendered | |
* in the target div. | |
* | |
* end: the time in seconds when the rendered mustache template should be | |
* removed from the target div. | |
* | |
* onStart: the function to be run when the start time is reached. | |
* | |
* onFrame: [optional] a function to be run on each paint call | |
* (e.g., called ~60 times per second) between the start and end times. | |
* | |
* onEnd: [optional] a function to be run when the end time is reached. | |
* | |
* Example: | |
var p = Popcorn('#video') | |
// onStart function only | |
.code({ | |
start: 1, | |
end: 4, | |
onStart: function( options ) { | |
// called on start | |
} | |
}) | |
// onStart + onEnd only | |
.code({ | |
start: 6, | |
end: 8, | |
onStart: function( options ) { | |
// called on start | |
}, | |
onEnd: function ( options ) { | |
// called on end | |
} | |
}) | |
// onStart, onEnd, onFrame | |
.code({ | |
start: 10, | |
end: 14, | |
onStart: function( options ) { | |
// called on start | |
}, | |
onFrame: function ( options ) { | |
// called on every paint frame between start and end. | |
// uses mozRequestAnimationFrame, webkitRequestAnimationFrame, | |
// or setTimeout with 16ms window. | |
}, | |
onEnd: function ( options ) { | |
// called on end | |
} | |
}); | |
* | |
*/ | |
Popcorn.plugin( 'code' , function(options) { | |
var running = false; | |
// Setup a proper frame interval function (60fps), favouring paint events. | |
var step = ( function() { | |
var buildFrameRunner = function( runner ) { | |
return function( f, options ) { | |
var _f = function() { | |
f(); | |
if ( running ) { | |
runner( _f ); | |
} | |
}; | |
_f(); | |
}; | |
}; | |
// Figure out which level of browser support we have for this | |
if ( window.webkitRequestAnimationFrame ) { | |
return buildFrameRunner( window.webkitRequestAnimationFrame ); | |
} else if ( window.mozRequestAnimationFrame ) { | |
return buildFrameRunner( window.mozRequestAnimationFrame ); | |
} else { | |
return buildFrameRunner( function( f ) { | |
window.setTimeout( f, 16 ); | |
} ); | |
} | |
} )(); | |
if ( !options.onStart || typeof options.onStart !== 'function' ) { | |
if ( Popcorn.plugin.debug ) { | |
throw new Error( 'Popcorn Code Plugin Error: onStart must be a function.' ); | |
} | |
options.onStart = Popcorn.nop; | |
} | |
if ( options.onEnd && typeof options.onEnd !== 'function' ) { | |
if ( Popcorn.plugin.debug ) { | |
throw new Error( 'Popcorn Code Plugin Error: onEnd must be a function.' ); | |
} | |
options.onEnd = undefined; | |
} | |
if ( options.onFrame && typeof options.onFrame !== 'function' ) { | |
if ( Popcorn.plugin.debug ) { | |
throw new Error( 'Popcorn Code Plugin Error: onFrame must be a function.' ); | |
} | |
options.onFrame = undefined; | |
} | |
return { | |
start: function( event, options ) { | |
options.onStart( options ); | |
if ( options.onFrame ) { | |
running = true; | |
step( options.onFrame, options ); | |
} | |
}, | |
end: function( event, options ) { | |
if ( options.onFrame ) { | |
running = false; | |
} | |
if ( options.onEnd ) { | |
options.onEnd( options ); | |
} | |
} | |
}; | |
}, | |
{ | |
about: { | |
name: 'Popcorn Code Plugin', | |
version: '0.1', | |
author: 'David Humphrey (@humphd)', | |
website: 'http://vocamus.net/dave' | |
}, | |
options: { | |
start: {elem:'input', type:'text', label:'In'}, | |
end: {elem:'input', type:'text', label:'Out'}, | |
onStart: {elem:'input', type:'function', label:'onStart'}, | |
onFrame: {elem:'input', type:'function', label:'onFrame'}, | |
onEnd: {elem:'input', type:'function', label:'onEnd'} | |
} | |
}); | |
})( Popcorn ); | |
// PLUGIN: LASTFM | |
(function (Popcorn) { | |
var _artists = {}, | |
lastFMcallback = function(data){ | |
if (data.artist) { | |
var htmlString = ""; | |
htmlString = '<h3>'+data.artist.name+'</h3>'; | |
htmlString += '<a href="'+data.artist.url+'" target="_blank" style="float:left;margin:0 10px 0 0;"><img src="'+ data.artist.image[2]['#text'] +'" alt=""></a>'; | |
htmlString += '<p>'+ data.artist.bio.summary +'</p>'; | |
htmlString += '<hr /><p><h4>Tags</h4><ul>'; | |
Popcorn.forEach( data.artist.tags.tag, function( val, i) { | |
htmlString += '<li><a href="'+ val.url +'">'+ val.name +'</a></li>'; | |
}); | |
htmlString += '</ul></p>'; | |
htmlString += '<hr /><p><h4>Similar</h4><ul>'; | |
Popcorn.forEach( data.artist.similar.artist, function( val, i) { | |
htmlString += '<li><a href="'+ val.url +'">'+ val.name +'</a></li>'; | |
}); | |
htmlString += '</ul></p>'; | |
_artists[data.artist.name.toLowerCase()].htmlString = htmlString; | |
} | |
}; | |
/** | |
* LastFM popcorn plug-in | |
* Appends information about a LastFM artist to an element on the page. | |
* Options parameter will need a start, end, target, artist and apikey. | |
* Start is the time that you want this plug-in to execute | |
* End is the time that you want this plug-in to stop executing | |
* Artist is the name of who's LastFM information you wish to show | |
* Target is the id of the document element that the images are | |
* appended to, this target element must exist on the DOM | |
* ApiKey is the API key registered with LastFM for use with their API | |
* | |
* @param {Object} options | |
* | |
* Example: | |
var p = Popcorn('#video') | |
.lastfm({ | |
start: 5, // seconds, mandatory | |
end: 15, // seconds, mandatory | |
artist: 'yacht', // mandatory | |
target: 'lastfmdiv', // mandatory | |
apikey: '1234567890abcdef1234567890abcdef' // mandatory | |
} ) | |
* | |
*/ | |
Popcorn.plugin( "lastfm" , (function(){ | |
return { | |
manifest: { | |
about:{ | |
name: "Popcorn LastFM Plugin", | |
version: "0.1", | |
author: "Steven Weerdenburg", | |
website: "http://sweerdenburg.wordpress.com/" | |
}, | |
options:{ | |
start : {elem:'input', type:'text', label:'In'}, | |
end : {elem:'input', type:'text', label:'Out'}, | |
target : 'lastfm-container', | |
artist : {elem:'input', type:'text', label:'Artist'} | |
} | |
}, | |
_setup: function( options ) { | |
options._container = document.createElement( 'div' ); | |
options._container.style.display = "none"; | |
options._container.innerHTML = ""; | |
options.artist = options.artist && options.artist.toLowerCase() || ""; | |
var target = document.getElementById( options.target ); | |
if ( !target && Popcorn.plugin.debug ) { | |
throw new Error( "target container doesn't exist" ); | |
} | |
target && target.appendChild( options._container ); | |
if(!_artists[options.artist]) { | |
_artists[options.artist] = { | |
count: 0, | |
htmlString: "Unknown Artist" | |
}; | |
Popcorn.getJSONP("http://ws.audioscrobbler.com/2.0/?method=artist.getinfo&artist="+ options.artist +"&api_key="+options.apikey+"&format=json&callback=lastFMcallback", lastFMcallback, false ); | |
} | |
_artists[options.artist].count++; | |
}, | |
/** | |
* @member LastFM | |
* The start function will be executed when the currentTime | |
* of the video reaches the start time provided by the | |
* options variable | |
*/ | |
start: function( event, options ) { | |
options._container.innerHTML = _artists[options.artist].htmlString; | |
options._container.style.display = "inline"; | |
}, | |
/** | |
* @member LastFM | |
* The end function will be executed when the currentTime | |
* of the video reaches the end time provided by the | |
* options variable | |
*/ | |
end: function( event, options ) { | |
options._container.style.display = "none"; | |
options._container.innerHTML = ""; | |
}, | |
_teardown: function( options ) { | |
// cleaning possible reference to _artist array; | |
--_artists[ options.artist ].count || delete _artists[ options.artist ]; | |
document.getElementById( options.target ) && document.getElementById( options.target ).removeChild( options._container ); | |
} | |
}; | |
})()); | |
})( Popcorn ); | |
// PLUGIN: Footnote | |
(function (Popcorn) { | |
/** | |
* Footnote popcorn plug-in | |
* Adds text to an element on the page. | |
* Options parameter will need a start, end, target and text. | |
* Start is the time that you want this plug-in to execute | |
* End is the time that you want this plug-in to stop executing | |
* Text is the text that you want to appear in the target | |
* Target is the id of the document element that the text needs to be | |
* attached to, this target element must exist on the DOM | |
* | |
* @param {Object} options | |
* | |
* Example: | |
var p = Popcorn('#video') | |
.footnote({ | |
start: 5, // seconds | |
end: 15, // seconds | |
text: 'This video made exclusively for drumbeat.org', | |
target: 'footnotediv' | |
} ) | |
* | |
*/ | |
Popcorn.plugin( "footnote" , { | |
manifest: { | |
about:{ | |
name: "Popcorn Footnote Plugin", | |
version: "0.1", | |
author: "@annasob", | |
website: "annasob.wordpress.com" | |
}, | |
options:{ | |
start : {elem:'input', type:'text', label:'In'}, | |
end : {elem:'input', type:'text', label:'Out'}, | |
target : 'footnote-container', | |
text : {elem:'input', type:'text', label:'Text'} | |
} | |
}, | |
_setup: function(options) { | |
var target = document.getElementById( options.target ); | |
options._container = document.createElement( 'div' ); | |
options._container.style.display = "none"; | |
options._container.innerHTML = options.text; | |
if ( !target && Popcorn.plugin.debug ) { | |
throw new Error( "target container doesn't exist" ); | |
} | |
target && target.appendChild( options._container ); | |
}, | |
/** | |
* @member footnote | |
* The start function will be executed when the currentTime | |
* of the video reaches the start time provided by the | |
* options variable | |
*/ | |
start: function(event, options){ | |
options._container.style.display = "inline"; | |
}, | |
/** | |
* @member footnote | |
* The end function will be executed when the currentTime | |
* of the video reaches the end time provided by the | |
* options variable | |
*/ | |
end: function(event, options){ | |
options._container.style.display = "none"; | |
}, | |
_teardown: function( options ) { | |
document.getElementById( options.target ) && document.getElementById( options.target ).removeChild( options._container ); | |
} | |
}); | |
})( Popcorn ); | |
// PLUGIN: OPENMAP | |
( function ( Popcorn ) { | |
/** | |
* openmap popcorn plug-in | |
* Adds an OpenLayers map and open map tiles (OpenStreetMap [default], NASA WorldWind, or USGS Topographic) | |
* Based on the googlemap popcorn plug-in. No StreetView support | |
* Options parameter will need a start, end, target, type, zoom, lat and lng | |
* -Start is the time that you want this plug-in to execute | |
* -End is the time that you want this plug-in to stop executing | |
* -Target is the id of the DOM element that you want the map to appear in. This element must be in the DOM | |
* -Type [optional] either: ROADMAP (OpenStreetMap), SATELLITE (NASA WorldWind / LandSat), or TERRAIN (USGS). ROADMAP/OpenStreetMap is the default. | |
* -Zoom [optional] defaults to 2 | |
* -Lat and Lng are the coordinates of the map if location is not named | |
* -Location is a name of a place to center the map, geocoded to coordinates using TinyGeocoder.com | |
* -Markers [optional] is an array of map marker objects, with the following properties: | |
* --Icon is the URL of a map marker image | |
* --Size [optional] is the radius in pixels of the scaled marker image (default is 14) | |
* --Text [optional] is the HTML content of the map marker -- if your popcorn instance is named 'popped', use <script>popped.currentTime(10);</script> to control the video | |
* --Lat and Lng are coordinates of the map marker if location is not specified | |
* --Location is a name of a place for the map marker, geocoded to coordinates using TinyGeocoder.com | |
* Note: using location requires extra loading time, also not specifying both lat/lng and location will | |
* cause a JavaScript error. | |
* @param {Object} options | |
* | |
* Example: | |
var p = Popcorn( '#video' ) | |
.openmap({ | |
start: 5, | |
end: 15, | |
type: 'ROADMAP', | |
target: 'map', | |
lat: 43.665429, | |
lng: -79.403323 | |
}) | |
* | |
*/ | |
var newdiv, | |
i = 1; | |
// insert openlayers api script once | |
if ( !window.OpenLayers ) { | |
Popcorn.getScript( "http://openlayers.org/api/OpenLayers.js", function() {}); | |
} | |
function toggle( container, display ) { | |
if ( container.map ) { | |
container.map.div.style.display = display; | |
return; | |
} | |
setTimeout(function() { | |
toggle( container, display ); | |
}, 10 ); | |
} | |
Popcorn.plugin( "openmap", function( options ){ | |
var newdiv, | |
centerlonlat, | |
projection, | |
displayProjection, | |
pointLayer, | |
selectControl, | |
popup, | |
isGeoReady, | |
target = document.getElementById( options.target ); | |
// create a new div within the target div | |
// this is later passed on to the maps api | |
newdiv = document.createElement( "div" ); | |
newdiv.id = "openmapdiv" + i; | |
newdiv.style.width = "100%"; | |
newdiv.style.height = "100%"; | |
i++; | |
if ( !target && Popcorn.plugin.debug ) { | |
throw new Error( "target container doesn't exist" ); | |
} | |
target && target.appendChild( newdiv ); | |
// callback function fires when the script is run | |
isGeoReady = function() { | |
if ( !window.OpenLayers ) { | |
setTimeout(function() { | |
isGeoReady(); | |
}, 50); | |
} else { | |
if ( options.location ) { | |
// set a dummy center at start | |
location = new OpenLayers.LonLat( 0, 0 ); | |
// query TinyGeocoder and re-center in callback | |
Popcorn.getJSONP( | |
"http://tinygeocoder.com/create-api.php?q=" + options.location + "&callback=jsonp", | |
function( latlng ) { | |
centerlonlat = new OpenLayers.LonLat( latlng[1], latlng[0] ); | |
options.map.setCenter( centerlonlat ); | |
} | |
); | |
} else { | |
centerlonlat = new OpenLayers.LonLat( options.lng, options.lat ); | |
} | |
options.type = options.type || "ROADMAP"; | |
if ( options.type === "SATELLITE" ) { | |
// add NASA WorldWind / LANDSAT map | |
options.map = new OpenLayers.Map( { div: newdiv, "maxResolution": 0.28125, tileSize: new OpenLayers.Size( 512, 512 ) } ); | |
var worldwind = new OpenLayers.Layer.WorldWind( "LANDSAT", "http://worldwind25.arc.nasa.gov/tile/tile.aspx", 2.25, 4, { T: "105" } ); | |
options.map.addLayer( worldwind ); | |
displayProjection = new OpenLayers.Projection( "EPSG:4326" ); | |
projection = new OpenLayers.Projection( "EPSG:4326" ); | |
} | |
else if ( options.type === "TERRAIN" ) { | |
// add terrain map ( USGS ) | |
displayProjection = new OpenLayers.Projection( "EPSG:4326" ); | |
projection = new OpenLayers.Projection( "EPSG:4326" ); | |
options.map = new OpenLayers.Map( {div: newdiv, projection: projection } ); | |
var relief = new OpenLayers.Layer.WMS( "USGS Terraserver", "http://terraserver-usa.org/ogcmap.ashx?", { layers: 'DRG' } ); | |
options.map.addLayer( relief ); | |
} else { | |
// add OpenStreetMap layer | |
projection = new OpenLayers.Projection( 'EPSG:900913' ); | |
displayProjection = new OpenLayers.Projection( 'EPSG:4326' ); | |
centerlonlat = centerlonlat.transform( displayProjection, projection ); | |
options.map = new OpenLayers.Map( { div: newdiv, projection: projection, "displayProjection": displayProjection } ); | |
var osm = new OpenLayers.Layer.OSM(); | |
options.map.addLayer( osm ); | |
} | |
if ( options.map ) { | |
options.map.div.style.display = "none"; | |
} | |
} | |
}; | |
isGeoReady(); | |
return { | |
/** | |
* @member openmap | |
* The setup function will be executed when the plug-in is instantiated | |
*/ | |
_setup: function( options ) { | |
var isReady = function() { | |
// wait until OpenLayers has been loaded, and the start function is run, before adding map | |
if ( !options.map ) { | |
setTimeout(function() { | |
isReady(); | |
}, 13 ); | |
} else { | |
// default zoom is 2 | |
options.zoom = options.zoom || 2; | |
// make sure options.zoom is a number | |
if ( options.zoom && typeof options.zoom !== "number" ) { | |
options.zoom = +options.zoom; | |
} | |
// reset the location and zoom just in case the user played with the map | |
options.map.setCenter( centerlonlat, options.zoom ); | |
if ( options.markers ) { | |
var layerStyle = OpenLayers.Util.extend( {} , OpenLayers.Feature.Vector.style[ 'default' ] ), | |
featureSelected = function( clickInfo ) { | |
clickedFeature = clickInfo.feature; | |
if ( !clickedFeature.attributes.text ) { | |
return; | |
} | |
popup = new OpenLayers.Popup.FramedCloud( | |
"featurePopup", | |
clickedFeature.geometry.getBounds().getCenterLonLat(), | |
new OpenLayers.Size( 120, 250 ), | |
clickedFeature.attributes.text, | |
null, | |
true, | |
function( closeInfo ) { | |
selectControl.unselect( this.feature ); | |
} | |
); | |
clickedFeature.popup = popup; | |
popup.feature = clickedFeature; | |
options.map.addPopup( popup ); | |
}, | |
featureUnSelected = function( clickInfo ) { | |
feature = clickInfo.feature; | |
if ( feature.popup ) { | |
popup.feature = null; | |
options.map.removePopup( feature.popup ); | |
feature.popup.destroy(); | |
feature.popup = null; | |
} | |
}, | |
gcThenPlotMarker = function( myMarker ) { | |
Popcorn.getJSONP( | |
"http://tinygeocoder.com/create-api.php?q=" + myMarker.location + "&callback=jsonp", | |
function( latlng ) { | |
var myPoint = new OpenLayers.Geometry.Point( latlng[1], latlng[0] ).transform( displayProjection, projection ), | |
myPointStyle = OpenLayers.Util.extend( {}, layerStyle ); | |
if ( !myMarker.size || isNaN( myMarker.size ) ) { | |
myMarker.size = 14; | |
} | |
myPointStyle.pointRadius = myMarker.size; | |
myPointStyle.graphicOpacity = 1; | |
myPointStyle.externalGraphic = myMarker.icon; | |
var myPointFeature = new OpenLayers.Feature.Vector( myPoint, null, myPointStyle ); | |
if ( myMarker.text ) { | |
myPointFeature.attributes = { | |
text: myMarker.text | |
}; | |
} | |
pointLayer.addFeatures( [ myPointFeature ] ); | |
} | |
); | |
}; | |
pointLayer = new OpenLayers.Layer.Vector( "Point Layer", { style: layerStyle } ); | |
options.map.addLayer( pointLayer ); | |
for ( var m = 0, l = options.markers.length; m < l ; m++ ) { | |
var myMarker = options.markers[ m ]; | |
if( myMarker.text ){ | |
if( !selectControl ){ | |
selectControl = new OpenLayers.Control.SelectFeature( pointLayer ); | |
options.map.addControl( selectControl ); | |
selectControl.activate(); | |
pointLayer.events.on({ | |
"featureselected": featureSelected, | |
"featureunselected": featureUnSelected | |
}); | |
} | |
} | |
if ( myMarker.location ) { | |
var geocodeThenPlotMarker = gcThenPlotMarker; | |
geocodeThenPlotMarker(myMarker); | |
} else { | |
var myPoint = new OpenLayers.Geometry.Point( myMarker.lng, myMarker.lat ).transform( displayProjection, projection ), | |
myPointStyle = OpenLayers.Util.extend( {}, layerStyle ); | |
if ( !myMarker.size || isNaN( myMarker.size ) ) { | |
myMarker.size = 14; | |
} | |
myPointStyle.pointRadius = myMarker.size; | |
myPointStyle.graphicOpacity = 1; | |
myPointStyle.externalGraphic = myMarker.icon; | |
var myPointFeature = new OpenLayers.Feature.Vector( myPoint, null, myPointStyle ); | |
if ( myMarker.text ) { | |
myPointFeature.attributes = { | |
text: myMarker.text | |
}; | |
} | |
pointLayer.addFeatures( [ myPointFeature ] ); | |
} | |
} | |
} | |
} | |
}; | |
isReady(); | |
}, | |
/** | |
* @member openmap | |
* The start function will be executed when the currentTime | |
* of the video reaches the start time provided by the | |
* options variable | |
*/ | |
start: function( event, options ) { | |
toggle( options, "block" ); | |
}, | |
/** | |
* @member openmap | |
* The end function will be executed when the currentTime | |
* of the video reaches the end time provided by the | |
* options variable | |
*/ | |
end: function( event, options ) { | |
toggle( options, "none" ); | |
}, | |
_teardown: function( options ) { | |
target && target.removeChild( newdiv ); | |
newdiv = map = centerlonlat = projection = displayProjection = pointLayer = selectControl = popup = null; | |
} | |
}; | |
}, | |
{ | |
about:{ | |
name: "Popcorn OpenMap Plugin", | |
version: "0.3", | |
author: "@mapmeld", | |
website: "mapadelsur.blogspot.com" | |
}, | |
options:{ | |
start : { elem: 'input', type: 'text', label: 'In'}, | |
end : { elem: 'input', type: 'text', label: 'Out'}, | |
target : 'map-container', | |
type : { elem: 'select', options:[ 'ROADMAP', 'SATELLITE', 'TERRAIN' ], label: 'Type' }, | |
zoom : { elem: 'input', type: 'text', label: 'Zoom'}, | |
lat : { elem: 'input', type: 'text', label: 'Lat'}, | |
lng : { elem: 'input', type: 'text', label: 'Lng'}, | |
location : { elem: 'input', type: 'text', label: 'Location'}, | |
markers : { elem: 'input', type: 'text', label: 'List Markers'} | |
} | |
}); | |
}) ( Popcorn ); | |
// PLUGIN: Google Feed | |
(function ( Popcorn ) { | |
var i = 1, | |
scriptLoaded = false, | |
dynamicFeedLoad = function() { | |
var dontLoad = false, | |
k = 0, | |
links = document.getElementsByTagName( "link" ), | |
len = links.length, | |
head = document.head || document.getElementsByTagName( "head" )[ 0 ], | |
css = document.createElement( "link" ), | |
resource = "http://www.google.com/uds/solutions/dynamicfeed/gfdynamicfeedcontrol."; | |
if ( !window.GFdynamicFeedControl ) { | |
Popcorn.getScript( resource + "js", function() { | |
scriptLoaded = true; | |
}); | |
} else { | |
scriptLoaded = true; | |
} | |
// Checking if the css file is already included | |
for ( ; k < len; k++ ){ | |
if ( links[ k ].href === resource + "css" ) { | |
dontLoad = true; | |
} | |
} | |
if ( !dontLoad ) { | |
css.type = "text/css"; | |
css.rel = "stylesheet"; | |
css.href = resource + "css"; | |
head.insertBefore( css, head.firstChild ); | |
} | |
}; | |
if ( !window.google ) { | |
Popcorn.getScript( "http://www.google.com/jsapi", function() { | |
google.load( "feeds", "1", { | |
callback: function () { | |
dynamicFeedLoad(); | |
} | |
}); | |
}); | |
} else { | |
dynamicFeedLoad(); | |
} | |
/** | |
* googlefeed popcorn plug-in | |
* Adds a feed from the specified blog url at the target div | |
* Options parameter will need a start, end, target, url and title | |
* -Start is the time that you want this plug-in to execute | |
* -End is the time that you want this plug-in to stop executing | |
* -Target is the id of the DOM element that you want the map to appear in. This element must be in the DOM | |
* -Url is the url of the blog's feed you are trying to access | |
* -Title is the title of the blog you want displayed above the feed | |
* -Orientation is the orientation of the blog, accepts either Horizontal or Vertical, defaults to Vertical | |
* @param {Object} options | |
* | |
* Example: | |
var p = Popcorn("#video") | |
.googlefeed({ | |
start: 5, // seconds | |
end: 15, // seconds | |
target: "map", | |
url: "http://zenit.senecac.on.ca/~chris.tyler/planet/rss20.xml", | |
title: "Planet Feed" | |
} ) | |
* | |
*/ | |
Popcorn.plugin( "googlefeed" , function( options ) { | |
// create a new div and append it to the parent div so nothing | |
// that already exists in the parent div gets overwritten | |
var newdiv = document.createElement( "div" ), | |
target = document.getElementById( options.target ), | |
initialize = function() { | |
//ensure that the script has been loaded | |
if ( !scriptLoaded ) { | |
setTimeout( function () { | |
initialize(); | |
}, 5 ); | |
} else { | |
// Create the feed control using the user entered url and title | |
options.feed = new GFdynamicFeedControl( options.url, newdiv, { | |
vertical: options.orientation.toLowerCase() === "vertical" ? true : false, | |
horizontal: options.orientation.toLowerCase() === "horizontal" ? true : false, | |
title: options.title = options.title || "Blog" | |
}); | |
} | |
}; | |
// Default to vertical orientation if empty or incorrect input | |
if( !options.orientation || ( options.orientation.toLowerCase() !== "vertical" && | |
options.orientation.toLowerCase() !== "horizontal" ) ) { | |
options.orientation = "vertical"; | |
} | |
newdiv.style.display = "none"; | |
newdiv.id = "_feed" + i; | |
newdiv.style.width = "100%"; | |
newdiv.style.height = "100%"; | |
i++; | |
if ( !target && Popcorn.plugin.debug ) { | |
throw new Error( "target container doesn't exist" ); | |
} | |
target && target.appendChild( newdiv ); | |
initialize(); | |
return { | |
/** | |
* @member webpage | |
* The start function will be executed when the currentTime | |
* of the video reaches the start time provided by the | |
* options variable | |
*/ | |
start: function( event, options ){ | |
newdiv.setAttribute( "style", "display:inline" ); | |
}, | |
/** | |
* @member webpage | |
* The end function will be executed when the currentTime | |
* of the video reaches the end time provided by the | |
* options variable | |
*/ | |
end: function( event, options ){ | |
newdiv.setAttribute( "style", "display:none" ); | |
}, | |
_teardown: function( options ) { | |
document.getElementById( options.target ) && document.getElementById( options.target ).removeChild( newdiv ); | |
delete options.feed; | |
} | |
}; | |
}, | |
{ | |
about: { | |
name: "Popcorn Google Feed Plugin", | |
version: "0.1", | |
author: "David Seifried", | |
website: "dseifried.wordpress.com" | |
}, | |
options: { | |
start: { | |
elem: "input", | |
type: "text", | |
label: "In" | |
}, | |
end: { | |
elem: "input", | |
type: "text", | |
label: "Out" | |
}, | |
target: "feed-container", | |
url: { | |
elem: "input", | |
type: "text", | |
label: "url" | |
}, | |
title: { | |
elem: "input", | |
type: "text", | |
label: "title" | |
}, | |
orientation: { | |
elem: "select", | |
options: [ "Vertical","Horizontal" ], | |
label: "orientation" | |
} | |
} | |
}); | |
})( Popcorn ); | |
// PLUGIN: tagthisperson | |
(function (Popcorn) { | |
var peopleArray = []; | |
// one People object per options.target | |
var People = function() { | |
this.name = ""; | |
this.contains = { }; | |
this.toString = function() { | |
var r = []; | |
for ( var j in this.contains ) { | |
if ( this.contains.hasOwnProperty( j ) ) { | |
r.push( " " + this.contains[ j ] ); | |
} | |
} | |
return r.toString(); | |
}; | |
}; | |
/** | |
* tagthisperson popcorn plug-in | |
* Adds people's names to an element on the page. | |
* Options parameter will need a start, end, target, image and person. | |
* Start is the time that you want this plug-in to execute | |
* End is the time that you want this plug-in to stop executing | |
* Person is the name of the person who you want to tag | |
* Image is the url to the image of the person - optional | |
* href is the url to the webpage of the person - optional | |
* Target is the id of the document element that the text needs to be | |
* attached to, this target element must exist on the DOM | |
* | |
* @param {Object} options | |
* | |
* Example: | |
var p = Popcorn('#video') | |
.tagthisperson({ | |
start: 5, // seconds | |
end: 15, // seconds | |
person: '@annasob', | |
image: 'http://newshour.s3.amazonaws.com/photos%2Fspeeches%2Fguests%2FRichardNSmith_thumbnail.jpg', | |
href: 'http://annasob.wordpress.com', | |
target: 'tagdiv' | |
} ) | |
* | |
*/ | |
Popcorn.plugin( "tagthisperson" , ( function() { | |
return { | |
manifest: { | |
about:{ | |
name: "Popcorn tagthisperson Plugin", | |
version: "0.1", | |
author: "@annasob", | |
website: "annasob.wordpress.com" | |
}, | |
options:{ | |
start : {elem:'input', type:'text', label:'In'}, | |
end : {elem:'input', type:'text', label:'Out'}, | |
target : 'tagthisperson-container', | |
person : {elem:'input', type:'text', label:'Name'}, | |
image : {elem:'input', type:'text', label:'Image Src'}, | |
href : {elem:'input', type:'text', label:'URL'} | |
} | |
}, | |
_setup: function( options ) { | |
var exists = false, | |
target = document.getElementById( options.target ); | |
if ( !target && Popcorn.plugin.debug ) { | |
throw new Error( "target container doesn't exist" ); | |
} | |
// loop through the existing objects to ensure no duplicates | |
// the idea here is to have one object per unique options.target | |
for ( var i = 0; i < peopleArray.length; i++ ) { | |
if ( peopleArray[ i ].name === options.target ) { | |
options._p = peopleArray[ i ]; | |
exists = true; | |
break; | |
} | |
} | |
if ( !exists ) { | |
options._p = new People(); | |
options._p.name = options.target; | |
peopleArray.push( options._p ); | |
} | |
}, | |
/** | |
* @member tagthisperson | |
* The start function will be executed when the currentTime | |
* of the video reaches the start time provided by the | |
* options variable | |
*/ | |
start: function( event, options ){ | |
options._p.contains[ options.person ] = ( options.image ) ? "<img src='" + options.image + "'/> " : "" ; | |
options._p.contains[ options.person ] += ( options.href ) ? "<a href='" + options.href + "' target='_blank'> " + options.person + "</a>" : options.person ; | |
document.getElementById( options.target ).innerHTML = options._p.toString(); | |
}, | |
/** | |
* @member tagthisperson | |
* The end function will be executed when the currentTime | |
* of the video reaches the end time provided by the | |
* options variable | |
*/ | |
end: function( event, options ){ | |
delete options._p.contains[ options.person ]; | |
document.getElementById( options.target ).innerHTML = options._p.toString(); | |
} | |
}; | |
})()); | |
})( Popcorn ); | |
// PLUGIN: Wordriver | |
(function (Popcorn) { | |
var container = {}, | |
spanLocation = 0, | |
setupContainer = function( target ) { | |
container[ target ] = document.createElement( "div" ); | |
var t = document.getElementById( target ); | |
t && t.appendChild( container[ target ] ); | |
container[ target ].style.height = "100%"; | |
container[ target ].style.position = "relative"; | |
return container[ target ]; | |
}, | |
// creates an object of supported, cross platform css transitions | |
span = document.createElement( "span" ), | |
prefixes = [ "webkit", "Moz", "ms", "O", "" ], | |
specProp = [ "Transform", "TransitionDuration", "TransitionTimingFunction" ], | |
supports = {}, | |
prop; | |
document.getElementsByTagName( "head" )[ 0 ].appendChild( span ); | |
for ( var sIdx = 0, sLen = specProp.length; sIdx < sLen; sIdx++ ) { | |
for ( var pIdx = 0, pLen = prefixes.length; pIdx < pLen; pIdx++ ) { | |
prop = prefixes[ pIdx ] + specProp[ sIdx ]; | |
if ( prop in span.style ) { | |
supports[ specProp[ sIdx ].toLowerCase() ] = prop; | |
break; | |
} | |
} | |
} | |
// Garbage collect support test span | |
document.getElementsByTagName( "head" )[ 0 ].appendChild( span ); | |
/** | |
* Word River popcorn plug-in | |
* Displays a string of text, fading it in and out | |
* while transitioning across the height of the parent container | |
* for the duration of the instance (duration = end - start) | |
* | |
* @param {Object} options | |
* | |
* Example: | |
var p = Popcorn( '#video' ) | |
.wordriver({ | |
start: 5, // When to begin the Word River animation | |
end: 15, // When to finish the Word River animation | |
text: 'Hello World', // The text you want to be displayed by Word River | |
target: 'wordRiverDiv', // The target div to append the text to | |
color: "blue" // The color of the text. (can be Hex value i.e. #FFFFFF ) | |
} ) | |
* | |
*/ | |
Popcorn.plugin( "wordriver" , { | |
manifest: { | |
about:{ | |
name: "Popcorn WordRiver Plugin" | |
}, | |
options:{ | |
start : {elem:'input', type:'text', label:'In'}, | |
end : {elem:'input', type:'text', label:'Out'}, | |
target : 'wordriver-container', | |
text : {elem:'input', type:'text', label:'Text'}, | |
color : {elem:'input', type:'text', label:'Color'} | |
} | |
}, | |
_setup: function( options ) { | |
if ( !document.getElementById( options.target ) && Popcorn.plugin.debug ) { | |
throw new Error( "target container doesn't exist" ); | |
} | |
options._duration = options.end - options.start; | |
options._container = container[ options.target ] || setupContainer( options.target ); | |
options.word = document.createElement( "span" ); | |
options.word.style.position = "absolute"; | |
options.word.style.whiteSpace = "nowrap"; | |
options.word.style.opacity = 0; | |
options.word.style.MozTransitionProperty = "opacity, -moz-transform"; | |
options.word.style.webkitTransitionProperty = "opacity, -webkit-transform"; | |
options.word.style.OTransitionProperty = "opacity, -o-transform"; | |
options.word.style.transitionProperty = "opacity, transform"; | |
options.word.style[ supports.transitionduration ] = 1 + "s, " + options._duration + "s"; | |
options.word.style[ supports.transitiontimingfunction ] = "linear"; | |
options.word.innerHTML = options.text; | |
options.word.style.color = options.color || "black"; | |
}, | |
start: function( event, options ){ | |
options._container.appendChild( options.word ); | |
// Resets the transform when changing to a new currentTime before the end event occurred. | |
options.word.style[ supports.transform ] = ""; | |
options.word.style.fontSize = ~~( 30 + 20 * Math.random() ) + "px"; | |
spanLocation = spanLocation % ( options._container.offsetWidth - options.word.offsetWidth ); | |
options.word.style.left = spanLocation + "px"; | |
spanLocation += options.word.offsetWidth + 10; | |
options.word.style[ supports.transform ] = "translateY(" + | |
( options._container.offsetHeight - options.word.offsetHeight ) + "px)"; | |
options.word.style.opacity = 1; | |
// automatically clears the word based on time | |
setTimeout( function() { | |
options.word.style.opacity = 0; | |
// ensures at least one second exists, because the fade animation is 1 second | |
}, ( ( (options.end - options.start) - 1 ) || 1 ) * 1000 ); | |
}, | |
end: function( event, options ){ | |
// manually clears the word based on user interaction | |
options.word.style.opacity = 0; | |
}, | |
_teardown: function( options ) { | |
// removes word span from generated container | |
options.word.parentNode && options._container.removeChild( options.word ); | |
// if no more word spans exist in container, remove container | |
container[ options.target ] && | |
!container[ options.target ].childElementCount && | |
document.getElementById( options.target ).removeChild( container[ options.target ] ) && | |
delete container[ options.target ]; | |
} | |
}); | |
})( Popcorn ); | |
// PLUGIN: Subtitle | |
(function ( Popcorn ) { | |
var scriptLoaded = false, | |
i = 0, | |
callBack = function( data ) { | |
if ( window.google && google.load ) { | |
google.load( "language", "1", { | |
callback: function() { | |
scriptLoaded = true; | |
} | |
}); | |
} else { | |
setTimeout( function() { | |
callBack( data ); | |
}, 1); | |
} | |
}, | |
createDefaultContainer = function( context ) { | |
var ctxContainer = context.container = document.createElement( "div" ), | |
style = ctxContainer.style, | |
media = context.media; | |
var updatePosition = function() { | |
var position = context.position(); | |
// the video element must have height and width defined | |
style.fontSize = "18px"; | |
style.width = media.offsetWidth + "px"; | |
style.top = position.top + media.offsetHeight - ctxContainer.offsetHeight - 40 + "px"; | |
style.left = position.left + "px"; | |
setTimeout( updatePosition, 10 ); | |
}; | |
ctxContainer.id = Popcorn.guid(); | |
style.position = "absolute"; | |
style.color = "white"; | |
style.textShadow = "black 2px 2px 6px"; | |
style.fontWeight = "bold"; | |
style.textAlign = "center"; | |
updatePosition(); | |
context.media.parentNode.appendChild( ctxContainer ); | |
}; | |
Popcorn.getScript( "http://www.google.com/jsapi", callBack ); | |
/** | |
* Subtitle popcorn plug-in | |
* Displays a subtitle over the video, or in the target div | |
* Options parameter will need a start, and end. | |
* Optional parameters are target and text. | |
* Start is the time that you want this plug-in to execute | |
* End is the time that you want this plug-in to stop executing | |
* Target is the id of the document element that the content is | |
* appended to, this target element must exist on the DOM | |
* Text is the text of the subtitle you want to display. | |
* | |
* Language is the expected language the subtitle text is in | |
* Languagesrc is the target id of the element that contains | |
* the language value ("en", "fr", etc.) to translate the text into | |
* example: | |
* <select id="language"> | |
* <option value="zh" selected="selected">Chinese</option> | |
* <option value="en">English</option> | |
* <option value="fr">French</option> | |
* <option value="de">German</option> | |
* <option value="it">Italian</option> | |
* <option value="ja">Japanese</option> | |
* <option value="ko">Korean</option> | |
* <option value="fa">Persian</option> | |
* <option value="pl">Polish</option> | |
* <option value="pt">Portuguese</option> | |
* <option value="es">Spanish</option> | |
* </select> | |
* Accessibilitysrc is the target id of a checkbox element | |
* checked means show all subtitles regardless of language and languagesrc | |
* not checked means only translate if language and languagesrc are different | |
* if no accessibilitysrc exists, default is to display all subtitles regardless | |
* | |
* @param {Object} options | |
* | |
* Example: | |
var p = Popcorn('#video') | |
.subtitle({ | |
start: 5, // seconds, mandatory | |
end: 15, // seconds, mandatory | |
text: 'Hellow world', // optional | |
target: 'subtitlediv', // optional | |
language: 'en', // optional | |
languagesrc: 'language', // optional | |
accessibilitysrc: 'accessibility' // optional | |
} ) | |
* | |
*/ | |
// translates whatever is in options.container into selected language | |
var translate = function( options, text ) { | |
options.selectedLanguage = options.languageSrc.options[ options.languageSrc.selectedIndex ].value; | |
google.language.translate( text, "", options.selectedLanguage, function( result ) { | |
for( var k = 0, children = options.container.children, len = children.length; k < len; k++ ) { | |
if ( children[ k ].style.display === "inline" ) { | |
children[ k ].innerHTML = result.translation; | |
} | |
} | |
}); | |
}; | |
Popcorn.plugin( "subtitle" , { | |
manifest: { | |
about: { | |
name: "Popcorn Subtitle Plugin", | |
version: "0.1", | |
author: "Scott Downe", | |
website: "http://scottdowne.wordpress.com/" | |
}, | |
options: { | |
start: { | |
elem: "input", | |
type: "text", | |
label: "In" | |
}, | |
end: { | |
elem: "input", | |
type: "text", | |
label: "Out" | |
}, | |
target: "subtitle-container", | |
text: { | |
elem: "input", | |
type: "text", | |
label: "Text" | |
} | |
} | |
}, | |
_setup: function( options ) { | |
var newdiv = document.createElement( "div" ), | |
accessibility = document.getElementById( options.accessibilitysrc ); | |
newdiv.id = "subtitle-" + i; | |
newdiv.style.display = "none"; | |
i++; | |
// Creates a div for all subtitles to use | |
( !this.container && ( !options.target || options.target === "subtitle-container" ) ) && | |
createDefaultContainer( this ); | |
// if a target is specified, use that | |
if ( options.target && options.target !== "subtitle-container" ) { | |
options.container = document.getElementById( options.target ); | |
} else { | |
// use shared default container | |
options.container = this.container; | |
} | |
document.getElementById( options.container.id ).appendChild( newdiv ); | |
options.innerContainer = newdiv; | |
options.showSubtitle = function() { | |
options.innerContainer.innerHTML = options.text; | |
}; | |
options.toggleSubtitles = function() {}; | |
var readyCheck = setInterval(function() { | |
if ( !scriptLoaded ) { | |
return; | |
} | |
clearInterval( readyCheck ); | |
if ( options.languagesrc ) { | |
options.showSubtitle = translate; | |
options.languageSrc = document.getElementById( options.languagesrc ); | |
options.selectedLanguage = options.languageSrc.options[ options.languageSrc.selectedIndex ].value; | |
if ( !this.languageSources ) { | |
this.languageSources = {}; | |
} | |
if ( !this.languageSources[ options.languagesrc ] ) { | |
this.languageSources[ options.languagesrc ] = {}; | |
} | |
if ( !this.languageSources[ options.languagesrc ][ options.target ] ) { | |
this.languageSources[ options.languagesrc ][ options.target ] = true; | |
options.languageSrc.addEventListener( "change", function() { | |
options.toggleSubtitles(); | |
for( var k = 0, children = options.container.children, len = children.length; k < len; k++ ) { | |
if ( children[ k ].style.display === "inline" ) { | |
options.showSubtitle( options, children[ k ].innerHTML ); | |
} | |
} | |
}, false ); | |
} | |
} | |
if ( accessibility ) { | |
options.accessibility = accessibility; | |
options.toggleSubtitles = function() { | |
options.selectedLanguage = options.languageSrc.options[ options.languageSrc.selectedIndex ].value; | |
if ( options.accessibility.checked || options.selectedLanguage !== ( options.language || "") ) { | |
options.display = "inline"; | |
options.container.style.display = options.display; | |
} else if ( options.selectedLanguage === ( options.language || "") ) { | |
options.display = "none"; | |
options.container.style.display = options.display; | |
} | |
}; | |
options.accessibility.addEventListener( "change", options.toggleSubtitles, false ); | |
// initiate it once to set starting state | |
options.toggleSubtitles(); | |
} | |
}, 5); | |
}, | |
/** | |
* @member subtitle | |
* The start function will be executed when the currentTime | |
* of the video reaches the start time provided by the | |
* options variable | |
*/ | |
start: function( event, options ){ | |
options.innerContainer.style.display = "inline"; | |
options.showSubtitle( options, options.text ); | |
}, | |
/** | |
* @member subtitle | |
* The end function will be executed when the currentTime | |
* of the video reaches the end time provided by the | |
* options variable | |
*/ | |
end: function( event, options ) { | |
options.innerContainer.style.display = "none"; | |
options.innerContainer.innerHTML = ""; | |
}, | |
_teardown: function ( options ) { | |
options.container.removeChild( options.innerContainer ); | |
} | |
}); | |
})( Popcorn ); | |
// PLUGIN: Google News | |
(function (Popcorn) { | |
var scriptLoaded = false, | |
scriptLoading = false, | |
callBack = function( data ) { | |
if ( typeof google !== 'undefined' && google.load ) { | |
google.load("elements", "1", {packages : ["newsshow"], callback: function() {scriptLoaded = true;}}); | |
} else { | |
setTimeout( function() { | |
callBack( data ); | |
}, 1); | |
} | |
}; | |
/** | |
* Google News popcorn plug-in | |
* Displays Google News information on a topic in a targeted div. | |
* Options parameter will need a start, end and target. | |
* Optional parameters are topic. topic defaults to "top stories" | |
* Start is the time that you want this plug-in to execute | |
* End is the time that you want this plug-in to stop executing | |
* Target is the id of the document element that the content is | |
* appended to, this target element must exist on the DOM | |
* Topic is the topic of news articles you want to display. | |
* | |
* @param {Object} options | |
* | |
* Example: | |
var p = Popcorn('#video') | |
.googlenews({ | |
start: 5, // seconds, mandatory | |
end: 15, // seconds, mandatory | |
topic: 'oil spill', // optional | |
target: 'newsdiv' // mandatory | |
} ) | |
* | |
*/ | |
Popcorn.plugin( "googlenews" , { | |
manifest: { | |
about:{ | |
name: "Popcorn Google News Plugin", | |
version: "0.1", | |
author: "Scott Downe", | |
website: "http://scottdowne.wordpress.com/" | |
}, | |
options:{ | |
start : {elem:'input', type:'text', label:'In'}, | |
end : {elem:'input', type:'text', label:'Out'}, | |
target : 'news-container', | |
topic : {elem:'select', type:'text', label:'Type'} | |
} | |
}, | |
_setup : function( options ) { | |
var target = document.getElementById( options.target ); | |
options.container = document.createElement( 'div' ); | |
if ( !target && Popcorn.plugin.debug ) { | |
throw new Error( "target container doesn't exist" ); | |
} | |
var container = document.createElement( "div" ); | |
target && target.appendChild( options.container ); | |
options.container.appendChild( container ); | |
if ( !scriptLoading ) { | |
scriptLoading = true; | |
Popcorn.getScript( "http://www.google.com/jsapi", callBack ); | |
} | |
var readyCheck = setInterval(function() { | |
if ( !scriptLoaded ) { | |
return; | |
} | |
clearInterval( readyCheck ); | |
options.newsShow = target && new google.elements.NewsShow( container, { | |
format : "300x250", | |
queryList : [ | |
{ q: options.topic || "Top Stories" } | |
] | |
} ); | |
}, 5); | |
options.container.style.display = "none"; | |
}, | |
/** | |
* @member googlenews | |
* The start function will be executed when the currentTime | |
* of the video reaches the start time provided by the | |
* options variable | |
*/ | |
start: function( event, options ){ | |
options.container.setAttribute( 'style', 'display:inline' ); | |
}, | |
/** | |
* @member googlenews | |
* The end function will be executed when the currentTime | |
* of the video reaches the end time provided by the | |
* options variable | |
*/ | |
end: function( event, options ){ | |
options.container.setAttribute( 'style', 'display:none' ); | |
}, | |
_teardown: function( options ) { | |
// google news does not like this, throws an error "a is null" | |
// doesn't hurt popcorn, and only happens once | |
document.getElementById( options.target ) && document.getElementById( options.target ).removeChild( options.container ); | |
options.newsShow = null; | |
} | |
}); | |
})( Popcorn ); | |
// PLUGIN: Attribution | |
(function( Popcorn ) { | |
/** | |
* Attribution popcorn plug-in | |
* Adds text to an element on the page. | |
* Options parameter will need a mandatory start, end, target. | |
* Optional parameters include nameofwork, NameOfWorkUrl, CopyrightHolder, CopyrightHolderUrl, license & licenseUrl. | |
* Start is the time that you want this plug-in to execute | |
* End is the time that you want this plug-in to stop executing | |
* Target is the id of the document element that the text needs to be attached to, this target element must exist on the DOM | |
* nameofwork is the title of the attribution | |
* NameOfWorkUrl is a url that provides more details about the attribution | |
* CopyrightHolder is the name of the person/institution that holds the rights to the attribution | |
* CopyrightHolderUrl is the url that provides more details about the copyrightholder | |
* license is the type of license that the work is copyrighted under | |
* LicenseUrl is the url that provides more details about the ticense type | |
* @param {Object} options | |
* | |
* Example: | |
var p = Popcorn('#video') | |
.attribution({ | |
start: 5, // seconds | |
end: 15, // seconds | |
target: 'attributiondiv' | |
} ) | |
* | |
*/ | |
Popcorn.plugin( "attribution" , (function(){ | |
var | |
common = "", | |
licenses = { | |
"cc-by": common + "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", | |
"cc-by-sa": common + "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", | |
"cc-by-nd": common + "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", | |
"cc-by-nc": common + "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", | |
"cc-by-nc-sa": common + "pvSURBVHja7FptbFPXGX7utYlGJzz/2yYHYYQ2xZFWHAq0dLSx161TS9NcLylfocNmWtuVdUlKCNvIl4FAY0Id91Ob1sRrV7VaqTBfaxc6fEPQ4sRJbEaL82OVjZKoVJvm4KCpxB/vflzfE9/EThxo1Y72lY7v8T3nPPfc57znPe95z+WISMNx3FV8JZ+6EBHHASAAON19CjzPg+d5qFQq8LwKKp4Hr0pfeR4cx4PnOHAcB3CcjAICgVKEFKWQSkkpmUxK11QSyWQKqVSSlaUoxeoTkdwZlr8V5JHyjQAADgDJ5KpUKinxqum8SiWV8ao0yRw4js/kN01OmtiURGYymU6Z+aSS5FQqxYjNJPpWIlkNQEmuSg214iqlk8dPwev1YmBgAJOxSQXQEs0SrF27FuYfmFH28ENIplTg+YQ0IEkeHLj0WGZMnxRJMwHpOcRJ5A77A/C87UEoFFLUNxgMECoErFpTktfLfVFwOAD017PvQq1WM1LVarWUVGr0iOfgeMaB8fHxvDqk0+lQ/5t6lJbei0QyiWQygUQinZIJJBJJpuGZmvzR+Ed4vuMFjIRGAAAmkwlGoxEAEAwGIYoiAKDIUISd1TvxrW9/M+vzr3z0MV50vfiFwHmkfKNE8Hs9Z6BWqaeJVS/CIrUazY0t8BzzsAZarRaCIECv16O0tBQA0NPTg0gkAo/Hg4mJCVZXsAioq9+FxbctRiKRQDyRQCIRRyKRUJoMSuFq9Cp++cRTiMViEAQBTqcTer1e0dlIJILa2lp4PB5oNBq0OlpnvdS12DVU76z5wuDIdpjO9p6l3r5z1Ofvo8Ggny68HyTBIlB68pJWq6WWlhaKRqM0l3R1dZFWq2XtigxFdL6vlwaDg+Qb7KPevnPk7T1LZ8Ruevdv79Dp7lN04p3jZDAYCABZrVYFnowz8xky9lvH/6xIRYairDgup5O2btp8Uzijo6Pk6+sjX18fjY6O5oUDgHgAUKVtsFqlglql1Fyj0YhAIIDm5mZotdo5zYPVakU4HGZTaSQ0gnbHEYUt55lHInkjfp8foVAIgiCgfvfueU2Q1WqFIAgYCY1g2B9Q2MqR0AhWlZTg7rvWsfvPdXTgGYcDJ0+fxp663RgbG8sLJ7M/f3r1VZjW34OqzVtQtXkLTOvvwZnu7jlxFOtNr6+XfIM+Gr4wRK7nXUxzjEbjvFqbTaLRKBmNRobjesFFw/8Ypv4hH5339ZL3vKTF77z3FzIUS9obDofzxg+HwwSADAYD0xZ5FhR957u0YpmeSr+/np74+WMEgFpaWujQwUMEgI6+9VZeOHJ/fH19Et6d6+hn221Uv6uOVizT04plenI5nTlxsmiwpMWOZxzM3nZ1dc2rtdlEq9XC6/Wyto5DjrQvndZgLu1T8zxCl0IwmUyzbJzNZmNabrPZFGV6vR4mk0mxsodCEk5ZWRke2bgRY2NjONPdjRXL9Pjv5DVse3QbLn3wASoqK/PC0ev1iMViCAUuAgDKhZ/gD+5OtLUfxt6mRgCAu7MrJ44svOym8bzkisneQk1NDZvqNyJarRZOpxMAMD4+jpMnTrENi0Qyx9y0bM9xu91Z87Jka2M0GuE40o5Djja8/uYbqKisxIeXI3AcacfSpUvh7uxC6NKlvHBkaX1WUrjf//EVdu9H998PAIjFYvj4ypWcOIxgWZu8Xi8jp7q6mlUSRREWiwVmsxlmsxl2uz1nWUdHh8JeylrsPevN0F4OHD9N8Gchd951F9raD2N0dBT1u+pQ8r3b8fbRoyh7cAOqNm9hNnQu0Wg0cLlcuE2zBC+//HLWOp98cn1ODGmjwXHgOQ4DAwOSiyUIjBhRFGE2mxWNRFGEKIqorq6GxWKZVXbhwgV0dXUxLLfbjYGBAWkHmCZWIpdjfmW2xUzWXKvVOqs8W5uZ92KxGM50d6Ot/TCsl2woe3ADAKDf50O/z4fCwkJwi9Rz4ixSq1FfV4fbFi9m9/p9PpZfpl+Wsz8ZGiy9sLxDW7lyJatQW1vL7Ew4HIbX64Ver8f27duZJhuNRoTDYRw7dkzhIwNg+cnYpPQccBlXoLi4GKIoIhKJKDomD9DMvOyDiqIIg8Gg2FnNxPnFY4+jdd9+rLp9Jao2b8GHlyOoqKxEW/thVG3blhfO2NgYWpqasXXTZrTu24/WffvR1NAAANi9Z0/O/igIBgfFdM20J/LIWK1WZszD4TCsVisrkzcfgiCwssyFhG0bOfYz7YxvqlQMZD4i1xUqhOmNTTqfidPWfhi2HTtw5d//wj/DYbicTuxtakRFZSXsB/ajZM3qeXFsO3bAtmOHNNCdnejq7MT1T65jQ9lD2FK1NWd/FCbi85R169fBUGyAx+OBzWabpa3ZyPV4PCgyFCniAKvWlKDIUKTAKSwsxN6mRnxt8WIMDw3hVzU1N4Szt6kRP37gAVy+LGl1cXExDMXFc+IoNZiUUaxMeyJrs9vtxsTEBILBIJYvXw673c7K5G1yZlnmdJ6Oj7IfRScaWxqh0WjgdrthsVhYm8woWyQSgcViQUdHBzQaDXZW75z1Mnt+W58VZ9fuOrz+5hs3hbN6zWpUVFaiorIShuLivHBYsMc/PICCggKsv/seTMYmYbVamSZ5PJ5ZC5lsMsrLy3OWye1ra2vR0dGBJZolOP/3XkxNTWEqPoV4Io54PCEFg5IJRP8zgYP2g8yXNBqNMBqN0Gq1EEWRDfp8QZprsWtoO+hgQZrPE4cFe/qH+lFQUAB7kx2eYx5otVpEo1GFZ+ByuVgwx2Qyobm5mQ2Ay+VidTPLAGD58uWIRCK474f34YizHdenphCfQbAcN04lU/D3+3Hs6K0RrmQE+wb7sGhRAc6fO4/qpyT/1+l0oibDZt2IuN1utgs7cPAAHtzwAKbiU5iKx5GIxxFPzCA4SwD+/z3gzgNgRzomcyl0Oh0AwG63z3KdFiITExNsddXpdOlAfPoUI5VCKm2LKX3kdKsKDwApSiGZlM7R9rfuYwRZLBZFjHch5JrNZta2/tf16QB7cprkjCMjtsjSrXVkxBZ3ANTQ3ED+4QEKXgzQoz99VBFRCwQCC4p0ZUbSBItAwYsB8g8P0L7Wfez+lyhN/6l5upoGA34K3kDAPRqNUktLiyLgvmbtGrrwfpAGg35qaG74MpJL3EyntLG5AeUWAWq1GkccR/Daq6/d8JGRfX8LEokE+vsH8OTjT+bzHUHGro9j9zJ3mTP/58LJ1UZ+Rr6Bplx9WhDGzNTY3CBp8sUAdbpfIZ1Ol/eI6XQ6cj3vouDFAA0G/fTS717Ku+3MY6KZ+cx78+HM1z4frGx1FooxS4NlqXm6GlXbqthRj+jtwYnjJ+Y8tn9YeBgmUyk70Dx+/AQO2A8s5EuYWdqyEM2dWTfXdYFf52TV3lz9zLqTy1W46o4SNDY3oXCpLuM0IjPcCIXfKn94Mj42hmfbnTjXc27BL3MzpmE+kzAX/kIHLV+MOQmW5d7Se7GhbAPuWH0HvqHRpD+dmjYwRISrsRiGBodw+uTpBRP7WWnwzdrg+daET43gr+QmNhpE9PWvaPiMNhhE3P8GAG3CFDKJWtqSAAAAAElFTkSuQmCC", | |
"cc-by-nc-nd": common + "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" | |
}, | |
target; | |
return { | |
manifest: { | |
about:{ | |
name: "Popcorn Attribution Plugin", | |
version: "0.2", | |
author: "@rwaldron", | |
website: "github.com/rwldrn" | |
}, | |
options:{ | |
start: { elem:"input", type:"text", label:"In" }, | |
end: { elem:"input", type:"text", label:"Out" }, | |
nameofwork: { elem:"input", type:"text", label:"Name of Work" }, | |
nameofworkurl: { elem:"input", type:"text", label:"Url of Work" }, | |
copyrightholder: { elem:"input", type:"text", label:"Copyright Holder" }, | |
copyrightholderurl: { elem:"input", type:"text", label:"Copyright Holder Url" }, | |
license: { elem:"input", type:"text", label:"License type" }, | |
licenseurl: { elem:"input", type:"text", label:"License URL" }, | |
target: "attribution-container" | |
} | |
}, | |
_setup: function( options ) { | |
var attrib = "", | |
license = options.license && licenses[ options.license.toLowerCase() ], | |
tar = "target=_blank"; | |
// make a div to put the information into | |
options._container = document.createElement( "div" ); | |
options._container.style.display = "none"; | |
// Cache declared target | |
target = document.getElementById( options.target ); | |
if ( options.nameofworkurl ) { | |
attrib += "<a href='" + options.nameofworkurl + "' " + tar + ">"; | |
} | |
if ( options.nameofwork ) { | |
attrib += options.nameofwork; | |
} | |
if ( options.nameofworkurl ) { | |
attrib += "</a>"; | |
} | |
if ( options.copyrightholderurl ) { | |
attrib += "<a href='" + options.copyrightholderurl + "' " + tar + ">"; | |
} | |
if ( options.copyrightholder ) { | |
attrib += ", " + options.copyrightholder; | |
} | |
if ( options.copyrightholderurl ) { | |
attrib += "</a>"; | |
} | |
//if the user did not specify any parameters just pull the text from the tag | |
if ( attrib === "" ) { | |
attrib = options.text; | |
} | |
if ( options.license ) { | |
if ( license ) { | |
if ( options.licenseurl ) { | |
attrib = "<a href='" + options.licenseurl + "' " + tar + "><img src='"+ license +"' border='0'/></a> " + attrib; | |
} else { | |
attrib = "<img src='"+ license +"' />" + attrib; | |
} | |
} else { | |
attrib += ", license: "; | |
if ( options.licenseurl ) { | |
attrib += "<a href='" + options.licenseurl + "' " + tar + ">" + options.license + "</a> "; | |
} else { | |
attrib += options.license; | |
} | |
} | |
} else if ( options.licenseurl ) { | |
attrib += ", <a href='" + options.licenseurl + "' " + tar + ">license</a> "; | |
} | |
options._container.innerHTML = attrib; | |
if ( !target && Popcorn.plugin.debug ) { | |
throw new Error( "target container doesn't exist" ); | |
} | |
target && target.appendChild( options._container ); | |
}, | |
/** | |
* @member attribution | |
* The start function will be executed when the currentTime | |
* of the video reaches the start time provided by the | |
* options variable | |
*/ | |
start: function( event, options ) { | |
options._container.style.display = "inline"; | |
}, | |
/** | |
* @member attribution | |
* The end function will be executed when the currentTime | |
* of the video reaches the end time provided by the | |
* options variable | |
*/ | |
end: function( event, options ) { | |
options._container.style.display = "none"; | |
}, | |
_teardown: function( options ) { | |
// Cache declared target | |
target = document.getElementById( options.target ); | |
target && target.removeChild( options._container ); | |
} | |
}; | |
})()); | |
})( Popcorn ); | |
//PLUGIN: linkedin | |
(function (Popcorn){ | |
/** | |
* LinkedIn Popcorn plug-in | |
* Places a LinkedIn plugin inside a div ( http://developers.facebook.com/docs/plugins/ ) | |
* Options parameter will need a start, end, target, type, and an api key | |
* Optional parameters are url, counter, format, companyid, and productid | |
* Start is the time that you want this plug-in to execute | |
* End is the time that you want this plug-in to stop executing | |
* Target is the id of the document element that the plugin needs to be attached to, this target element must exist on the DOM | |
* Type is the name of the plugin, options are share, memberprofile, companyinsider, companyprofile, or recommendproduct | |
* Apikey is your own api key from obtained from https://www.linkedin.com/secure/developer | |
* Url is the desired url to share via LinkedIn. Defaults to the current page if no url is specified | |
* Counter is the position where the counter will be positioned. This is used if the type is "share" or "recommendproduct" | |
* The options are right and top (don't include this option if you do not want a counter) | |
* Format is the data format of the member and company profile plugins. The options are inlined, hover, and click. Defaults to inline | |
* Companyid must be specified if the type is "companyprofile," "companyinsider," or "recommendproduct" | |
* Productid must be specified if the type is "recommendproduct" | |
* | |
* @param {Object} options | |
* | |
* Example: | |
* <script src="popcorn.linkedin.js"></script> | |
* ... | |
* var p = Popcorn("#video") | |
* .linkedin({ | |
* type: "share", | |
* url: "http://www.google.ca", | |
* counter: "right", | |
* target: "sharediv" | |
* apikey: "ZOLRI2rzQS_oaXELpPF0aksxwFFEvoxAFZRLfHjaAhcGPfOX0Ds4snkJpWwKs8gk", | |
* start: 1, | |
* end: 3 | |
* }) | |
* | |
* This plugin will be displayed between 1 and 3 seconds, inclusive, in the video. This will show how many people have "shared" Google via LinkedIn, | |
* with the number of people (counter) displayed to the right of the share plugin. | |
*/ | |
Popcorn.plugin( "linkedin", { | |
manifest: { | |
about: { | |
name: "Popcorn LinkedIn Plugin", | |
version: "0.1", | |
author: "Dan Ventura", | |
website: "dsventura.blogspot.com" | |
}, | |
options: { | |
type: { | |
elem: "input", | |
type: "text", | |
label: "Type" | |
}, | |
target: "linkedin-container" | |
} | |
}, | |
_setup: function( options ) { | |
var apikey = options.apikey, | |
target = document.getElementById( options.target ), | |
script = document.createElement( "script" ); | |
Popcorn.getScript("http://platform.linkedin.com/in.js"); | |
options._container = document.createElement( "div" ); | |
options._container.appendChild( script ); | |
if ( apikey ) { | |
script.innerHTML = "api_key: " + apikey; | |
} | |
options.type = options.type && options.type.toLowerCase() || ""; | |
// Replace the LinkedIn plugin's error message to something more helpful | |
var errorMsg = function() { | |
options._container = document.createElement( "p" ); | |
options._container.innerHTML = "Plugin requires a valid <a href='https://www.linkedin.com/secure/developer'>apikey</a>"; | |
if ( !target && Popcorn.plugin.debug ) { | |
throw ( "target container doesn't exist" ); | |
} | |
target && target.appendChild( options._container ); | |
}; | |
var setOptions = (function ( options ) { | |
return { | |
share: function () { | |
script.setAttribute( "type", "IN/Share" ); | |
if ( options.counter ) { | |
script.setAttribute( "data-counter", options.counter ); | |
} | |
if ( options.url ) { | |
script.setAttribute( "data-url", options.url ); | |
} | |
}, | |
memberprofile: function () { | |
script.setAttribute( "type", "IN/MemberProfile" ); | |
script.setAttribute( "data-id", ( options.memberid ) ); | |
script.setAttribute( "data-format", ( options.format || "inline" ) ); | |
if ( options.text && options.format.toLowerCase() !== "inline" ) { | |
script.setAttribute( "data-text", options.text ); | |
} | |
}, | |
companyinsider: function () { | |
script.setAttribute( "type", "IN/CompanyInsider" ); | |
script.setAttribute( "data-id", options.companyid ); | |
if( options.modules ) { | |
options._container.setAttribute( "data-modules", options.modules ); | |
} | |
}, | |
companyprofile: function () { | |
script.setAttribute( "type", "IN/CompanyProfile" ); | |
script.setAttribute( "data-id", ( options.companyid ) ); | |
script.setAttribute( "data-format", ( options.format || "inline" ) ); | |
if ( options.text && options.format.toLowerCase() !== "inline" ) { | |
script.setAttribute( "data-text", options.text ); | |
} | |
if ( options.related !== undefined ) { | |
script.setAttribute( "data-related", options.related ); | |
} | |
}, | |
recommendproduct: function () { | |
script.setAttribute( "type", "IN/RecommendProduct" ); | |
script.setAttribute( "data-company", ( options.companyid || "LinkedIn" ) ); | |
script.setAttribute( "data-product", ( options.productid || "201714" ) ); | |
if ( options.counter ) { | |
script.setAttribute( "data-counter", options.counter ); | |
} | |
} | |
}; | |
})( options ); | |
if ( !apikey ) { | |
errorMsg(); | |
} else { | |
setOptions[ options.type ] && setOptions[ options.type ](); | |
} | |
if ( !target && Popcorn.plugin.debug ) { | |
throw new Error( "target container doesn't exist" ); | |
} | |
target && target.appendChild( options._container ); | |
options._container.style.display = "none"; | |
}, | |
/** | |
* @member linkedin | |
* The start function will be executed when the currentTime | |
* of the video reaches the start time provided by the | |
* options variable | |
*/ | |
start: function( event, options ) { | |
options._container.style.display = "block"; | |
}, | |
/** | |
* @member linkedin | |
* The end function will be executed when the currentTime | |
* of the video reaches the end time provided by the | |
* options variable | |
*/ | |
end: function( event, options ) { | |
options._container.style.display = "none"; | |
}, | |
_teardown: function( options ) { | |
var tar = document.getElementById( options.target ); | |
tar && tar.removeChild( options._container ); | |
} | |
}); | |
})( Popcorn ); | |
// PARSER: 0.3 SSA/ASS | |
(function (Popcorn) { | |
/** | |
* SSA/ASS popcorn parser plug-in | |
* Parses subtitle files in the identical SSA and ASS formats. | |
* Style information is ignored, and may be found in these | |
* formats: (\N \n {\pos(400,570)} {\kf89}) | |
* Out of the [Script Info], [V4 Styles], [Events], [Pictures], | |
* and [Fonts] sections, only [Events] is processed. | |
* Data parameter is given by Popcorn, will need a text. | |
* Text is the file contents to be parsed | |
* | |
* @param {Object} data | |
* | |
* Example: | |
[Script Info] | |
Title: Testing subtitles for the SSA Format | |
[V4 Styles] | |
Format: Name, Fontname, Fontsize, PrimaryColour, SecondaryColour, TertiaryColour, BackColour, Bold, Italic, BorderStyle, Outline, Shadow, Alignment, MarginL, MarginR, MarginV, AlphaLevel, Encoding | |
Style: Default,Arial,20,65535,65535,65535,-2147483640,-1,0,1,3,0,2,30,30,30,0,0 | |
[Events] | |
Format: Layer, Start, End, Style, Name, MarginL, MarginR, MarginV, Effect, Text | |
Dialogue: 0,0:00:02.40,0:00:07.20,Default,,0000,0000,0000,,Senator, {\kf89}we're \Nmaking our final \napproach into Coruscant. | |
Dialogue: 0,0:00:09.71,0:00:13.39,Default,,0000,0000,0000,,{\pos(400,570)}Very good, Lieutenant. | |
Dialogue: 0,0:00:15.04,0:00:18.04,Default,,0000,0000,0000,,It's \Na \ntrap! | |
* | |
*/ | |
// Register for SSA extensions | |
Popcorn.parser( "parseSSA", function( data ) { | |
// declare needed variables | |
var retObj = { | |
title: "", | |
remote: "", | |
data: [] | |
}, | |
subs = [], | |
startIdx, | |
endIdx, | |
textIdx, | |
lines, | |
fields, | |
numFields, | |
sub, | |
text, | |
i = 0, | |
j = 0, | |
len = 0, | |
fieldLen = 0; | |
// h:mm:ss.cc (centisec) string to SS.mmm | |
// Returns -1 if invalid | |
var toSeconds = function( t_in ) { | |
var t = t_in.split( ":" ), | |
l = t.length - 1; | |
// Not all there | |
if ( t_in.length !== 10 ) { | |
return -1; | |
} | |
return parseInt( t[0], 10 )*3600 + parseInt( t[l-1], 10 )*60 + parseFloat( t[l], 10 ); | |
}; | |
var createTrack = function( name, attributes ) { | |
var track = {}; | |
track[name] = attributes; | |
return track; | |
}; | |
// Here is where the magic happens | |
// Split on line breaks | |
lines = data.text.split( /(?:\r\n|\r|\n)/gm ); | |
len = lines.length; | |
// Ignore non-textual info | |
while ( i < len && lines[i] !== "[Events]" ) { | |
i++; | |
} | |
fields = lines[++i].substr( 8 ).split( ", " ); // Trim 'Format: ' off front, split on delim | |
numFields = fields.length; | |
//Find where in Dialogue string the start, end and text info is | |
for ( ; j < numFields; j++ ) { | |
if ( fields[j] === "Start" ) { | |
startIdx = j; | |
} else if ( fields[j] === "End" ) { | |
endIdx = j; | |
} else if ( fields[j] === "Text" ) { | |
textIdx = j; | |
} | |
} | |
while ( ++i < len && lines[i] && lines[i][0] !== "[" ) { | |
sub = {}; | |
// Trim beginning 'Dialogue: ' and split on delim | |
fields = lines[i].substr( 10 ).split( "," ); | |
sub.start = toSeconds( fields[startIdx] ); | |
sub.end = toSeconds( fields[endIdx] ); | |
// Invalid time, skip | |
if ( sub.start === -1 || sub.end === -1 ) { | |
continue; | |
} | |
if ( ( fieldLen = fields.length ) === numFields ) { | |
sub.text = fields[textIdx]; | |
} else { | |
// There were commas in the text which were split, append back together into one line | |
text = []; | |
for( j = textIdx; j < fieldLen; j++ ) { | |
text.push( fields[j] ); | |
} | |
sub.text = text.join( "," ); | |
} | |
// Eliminate advanced styles and convert forced line breaks | |
sub.text = sub.text.replace( /\{(\\[\w]+\(?([\w\d]+,?)+\)?)+\}/gi, "" ).replace( /\\N/gi, "<br />" ); | |
subs.push( createTrack( "subtitle", sub ) ); | |
} | |
retObj.data = subs; | |
return retObj; | |
}); | |
})( Popcorn ); | |
// PARSER: 0.1 TTXT | |
(function (Popcorn) { | |
/** | |
* TTXT popcorn parser plug-in | |
* Parses subtitle files in the TTXT format. | |
* Style information is ignored. | |
* Data parameter is given by Popcorn, will need an xml. | |
* Xml is the file contents to be parsed as a DOM tree | |
* | |
* @param {Object} data | |
* | |
* Example: | |
<TextSample sampleTime="00:00:00.000" text=""></TextSample> | |
*/ | |
Popcorn.parser( "parseTTXT", function( data ) { | |
// declare needed variables | |
var returnData = { | |
title: "", | |
remote: "", | |
data: [] | |
}; | |
// Simple function to convert HH:MM:SS.MMM to SS.MMM | |
// Assume valid, returns 0 on error | |
var toSeconds = function(t_in) { | |
var t = t_in.split(":"); | |
var time = 0; | |
try { | |
return parseFloat(t[0], 10)*60*60 + parseFloat(t[1], 10)*60 + parseFloat(t[2], 10); | |
} catch (e) { time = 0; } | |
return time; | |
}; | |
// creates an object of all atrributes keyed by name | |
var createTrack = function( name, attributes ) { | |
var track = {}; | |
track[name] = attributes; | |
return track; | |
}; | |
// this is where things actually start | |
var node = data.xml.lastChild.lastChild; // Last Child of TextStreamHeader | |
var lastStart = Number.MAX_VALUE; | |
var cmds = []; | |
// Work backwards through DOM, processing TextSample nodes | |
while (node) { | |
if ( node.nodeType === 1 && node.nodeName === "TextSample") { | |
var sub = {}; | |
sub.start = toSeconds(node.getAttribute('sampleTime')); | |
sub.text = node.getAttribute('text'); | |
if (sub.text) { // Only process if text to display | |
// Infer end time from prior element, ms accuracy | |
sub.end = lastStart - 0.001; | |
cmds.push( createTrack("subtitle", sub) ); | |
} | |
lastStart = sub.start; | |
} | |
node = node.previousSibling; | |
} | |
returnData.data = cmds.reverse(); | |
return returnData; | |
}); | |
})( Popcorn ); | |
// PARSER: 0.3 JSON | |
(function (Popcorn) { | |
Popcorn.parser( "parseJSON", "JSON", function( data ) { | |
// declare needed variables | |
var retObj = { | |
title: "", | |
remote: "", | |
data: [] | |
}, | |
manifestData = {}, | |
dataObj = data; | |
/* | |
TODO: add support for filling in source children of the video element | |
remote: [ | |
{ | |
src: "whatever.mp4", | |
type: 'video/mp4; codecs="avc1, mp4a"' | |
}, | |
{ | |
src: "whatever.ogv", | |
type: 'video/ogg; codecs="theora, vorbis"' | |
} | |
] | |
*/ | |
Popcorn.forEach( dataObj.data, function ( obj, key ) { | |
retObj.data.push( obj ); | |
}); | |
return retObj; | |
}); | |
})( Popcorn ); | |
// PARSER: 0.1 SBV | |
(function (Popcorn) { | |
/** | |
* SBV popcorn parser plug-in | |
* Parses subtitle files in the SBV format. | |
* Times are expected in H:MM:SS.MIL format, with hours optional | |
* Subtitles which don't match expected format are ignored | |
* Data parameter is given by Popcorn, will need a text. | |
* Text is the file contents to be parsed | |
* | |
* @param {Object} data | |
* | |
* Example: | |
0:00:02.400,0:00:07.200 | |
Senator, we're making our final approach into Coruscant. | |
*/ | |
Popcorn.parser( "parseSBV", function( data ) { | |
// declare needed variables | |
var retObj = { | |
title: "", | |
remote: "", | |
data: [] | |
}, | |
subs = [], | |
lines, | |
i = 0, | |
len = 0, | |
idx = 0; | |
// [H:]MM:SS.MIL string to SS.MIL | |
// Will thrown exception on bad time format | |
var toSeconds = function( t_in ) { | |
var t = t_in.split( ":" ), | |
l = t.length-1, | |
time; | |
try { | |
time = parseInt( t[l-1], 10 )*60 + parseFloat( t[l], 10 ); | |
// Hours optionally given | |
if ( l === 2 ) { | |
time += parseInt( t[0], 10 )*3600; | |
} | |
} catch ( e ) { | |
throw "Bad cue"; | |
} | |
return time; | |
}; | |
var createTrack = function( name, attributes ) { | |
var track = {}; | |
track[name] = attributes; | |
return track; | |
}; | |
// Here is where the magic happens | |
// Split on line breaks | |
lines = data.text.split( /(?:\r\n|\r|\n)/gm ); | |
len = lines.length; | |
while ( i < len ) { | |
var sub = {}, | |
text = [], | |
time = lines[i++].split( "," ); | |
try { | |
sub.start = toSeconds( time[0] ); | |
sub.end = toSeconds( time[1] ); | |
// Gather all lines of text | |
while ( i < len && lines[i] ) { | |
text.push( lines[i++] ); | |
} | |
// Join line breaks in text | |
sub.text = text.join( "<br />" ); | |
subs.push( createTrack( "subtitle", sub ) ); | |
} catch ( e ) { | |
// Bad cue, advance to end of cue | |
while ( i < len && lines[i] ) { | |
i++; | |
} | |
} | |
// Consume empty whitespace | |
while ( i < len && !lines[i] ) { | |
i++; | |
} | |
} | |
retObj.data = subs; | |
return retObj; | |
}); | |
})( Popcorn ); | |
// PARSER: 0.1 XML | |
(function (Popcorn) { | |
/** | |
* | |
* | |
*/ | |
Popcorn.parser( "parseXML", "XML", function( data ) { | |
// declare needed variables | |
var returnData = { | |
title: "", | |
remote: "", | |
data: [] | |
}, | |
manifestData = {}; | |
// Simple function to convert 0:05 to 0.5 in seconds | |
// acceptable formats are HH:MM:SS:MM, MM:SS:MM, SS:MM, SS | |
var toSeconds = function(time) { | |
var t = time.split(":"); | |
if (t.length === 1) { | |
return parseFloat(t[0], 10); | |
} else if (t.length === 2) { | |
return parseFloat(t[0], 10) + parseFloat(t[1] / 12, 10); | |
} else if (t.length === 3) { | |
return parseInt(t[0] * 60, 10) + parseFloat(t[1], 10) + parseFloat(t[2] / 12, 10); | |
} else if (t.length === 4) { | |
return parseInt(t[0] * 3600, 10) + parseInt(t[1] * 60, 10) + parseFloat(t[2], 10) + parseFloat(t[3] / 12, 10); | |
} | |
}; | |
// turns a node tree element into a straight up javascript object | |
// also converts in and out to start and end | |
// also links manifest data with ids | |
var objectifyAttributes = function ( nodeAttributes ) { | |
var returnObject = {}; | |
for ( var i = 0, nal = nodeAttributes.length; i < nal; i++ ) { | |
var key = nodeAttributes.item(i).nodeName, | |
data = nodeAttributes.item(i).nodeValue; | |
// converts in into start | |
if (key === "in") { | |
returnObject.start = toSeconds( data ); | |
// converts out into end | |
} else if ( key === "out" ){ | |
returnObject.end = toSeconds( data ); | |
// this is where ids in the manifest are linked | |
} else if ( key === "resourceid" ) { | |
Popcorn.extend( returnObject, manifestData[data] ); | |
// everything else | |
} else { | |
returnObject[key] = data; | |
} | |
} | |
return returnObject; | |
}; | |
// creates an object of all atrributes keyd by name | |
var createTrack = function( name, attributes ) { | |
var track = {}; | |
track[name] = attributes; | |
return track; | |
}; | |
// recursive function to process a node, or process the next child node | |
var parseNode = function ( node, allAttributes, manifest ) { | |
var attributes = {}; | |
Popcorn.extend( attributes, allAttributes, objectifyAttributes( node.attributes ), { text: node.textContent } ); | |
var childNodes = node.childNodes; | |
// processes the node | |
if ( childNodes.length < 1 || ( childNodes.length === 1 && childNodes[0].nodeType === 3 ) ) { | |
if ( !manifest ) { | |
returnData.data.push( createTrack( node.nodeName, attributes ) ); | |
} else { | |
manifestData[attributes.id] = attributes; | |
} | |
// process the next child node | |
} else { | |
for ( var i = 0; i < childNodes.length; i++ ) { | |
if ( childNodes[i].nodeType === 1 ) { | |
parseNode( childNodes[i], attributes, manifest ); | |
} | |
} | |
} | |
}; | |
// this is where things actually start | |
var x = data.documentElement.childNodes; | |
for ( var i = 0, xl = x.length; i < xl; i++ ) { | |
if ( x[i].nodeType === 1 ) { | |
// start the process of each main node type, manifest or timeline | |
if ( x[i].nodeName === "manifest" ) { | |
parseNode( x[i], {}, true ); | |
} else { // timeline | |
parseNode( x[i], {}, false ); | |
} | |
} | |
} | |
return returnData; | |
}); | |
})( Popcorn ); | |
// PARSER: 0.3 TTML | |
(function (Popcorn) { | |
/** | |
* TTML popcorn parser plug-in | |
* Parses subtitle files in the TTML format. | |
* Times may be absolute to the timeline or relative | |
* Absolute times are ISO 8601 format (hh:mm:ss[.mmm]) | |
* Relative times are a fraction followed by a unit metric (d.ddu) | |
* Relative times are relative to the time given on the parent node | |
* Styling information is ignored | |
* Data parameter is given by Popcorn, will need an xml. | |
* Xml is the file contents to be processed | |
* | |
* @param {Object} data | |
* | |
* Example: | |
<tt xmlns:tts="http://www.w3.org/2006/04/ttaf1#styling" xmlns="http://www.w3.org/2006/04/ttaf1"> | |
<body region="subtitleArea"> | |
<div> | |
<p xml:id="subtitle1" begin="0.76s" end="3.45s"> | |
It seems a paradox, does it not, | |
</p> | |
</div> | |
</body> | |
</tt> | |
*/ | |
Popcorn.parser( "parseTTML", function( data ) { | |
// declare needed variables | |
var returnData = { | |
title: "", | |
remote: "", | |
data: [] | |
}, | |
node, | |
numTracks = 0, | |
region; | |
// Convert time expression to SS.mmm | |
// Expression may be absolute to timeline (hh:mm:ss.ms) | |
// or relative ( fraction followedd by metric ) ex: 3.4s, 5.7m | |
// Returns -1 if invalid | |
var toSeconds = function ( t_in, offset ) { | |
if ( !t_in ) { | |
return -1; | |
} | |
var t = t_in.split( ":" ), | |
l = t.length - 1, | |
metric, | |
multiplier, | |
i; | |
// Try clock time | |
if ( l >= 2 ) { | |
return parseInt( t[0], 10 )*3600 + parseInt( t[l-1], 10 )*60 + parseFloat( t[l], 10 ); | |
} | |
// Was not clock time, assume relative time | |
// Take metric from end of string (may not be single character) | |
// First find metric | |
for( i = t_in.length - 1; i >= 0; i-- ) { | |
if ( t_in[i] <= "9" && t_in[i] >= "0" ) { | |
break; | |
} | |
} | |
// Point i at metric and normalize offsete time | |
i++; | |
metric = t_in.substr( i ); | |
offset = offset || 0; | |
// Determine multiplier for metric relative to seconds | |
if ( metric === "h" ) { | |
multiplier = 3600; | |
} else if ( metric === "m" ) { | |
multiplier = 60; | |
} else if ( metric === "s" ) { | |
multiplier = 1; | |
} else if ( metric === "ms" ) { | |
multiplier = 0.001; | |
} else { | |
return -1; | |
} | |
// Valid multiplier | |
return parseFloat( t_in.substr( 0, i ) ) * multiplier + offset; | |
}; | |
// creates an object of all atrributes keyd by name | |
var createTrack = function( name, attributes ) { | |
var track = {}; | |
track[name] = attributes; | |
return track; | |
}; | |
// Parse a node for text content | |
var parseNode = function( node, timeOffset ) { | |
var sub = {}; | |
// Trim left and right whitespace from text and change non-explicit line breaks to spaces | |
sub.text = node.textContent.replace(/^[\s]+|[\s]+$/gm, "").replace(/(?:\r\n|\r|\n)/gm, "<br />"); | |
sub.id = node.getAttribute( "xml:id" ); | |
sub.start = toSeconds ( node.getAttribute( "begin" ), timeOffset ); | |
sub.end = toSeconds( node.getAttribute( "end" ), timeOffset ); | |
sub.target = region; | |
if ( sub.end < 0 ) { | |
// No end given, infer duration if possible | |
// Otherwise, give end as MAX_VALUE | |
sub.end = toSeconds( node.getAttribute( "duration" ), 0 ); | |
if ( sub.end >= 0 ) { | |
sub.end += sub.start; | |
} else { | |
sub.end = Number.MAX_VALUE; | |
} | |
} | |
return sub; | |
}; | |
// Parse the children of the given node | |
var parseChildren = function( node, timeOffset ) { | |
var currNode = node.firstChild, | |
sub, | |
newOffset; | |
while ( currNode ) { | |
if ( currNode.nodeType === 1 ) { | |
if ( currNode.nodeName === "p" ) { | |
// p is a teextual node, process contents as subtitle | |
sub = parseNode( currNode, timeOffset ); | |
returnData.data.push( createTrack( "subtitle", sub ) ); | |
numTracks++; | |
} else if ( currNode.nodeName === "div" ) { | |
// div is container for subtitles, recurse | |
newOffset = toSeconds( currNode.getAttribute("begin") ); | |
if (newOffset < 0 ) { | |
newOffset = timeOffset; | |
} | |
parseChildren( currNode, newOffset ); | |
} | |
} | |
currNode = currNode.nextSibling; | |
} | |
}; | |
// Null checks | |
if ( !data.xml || !data.xml.documentElement || !( node = data.xml.documentElement.firstChild ) ) { | |
return returnData; | |
} | |
// Find body tag | |
while ( node.nodeName !== "body" ) { | |
node = node.nextSibling; | |
} | |
region = ""; | |
parseChildren( node, 0 ); | |
return returnData; | |
}); | |
})( Popcorn ); | |
// PARSER: 0.3 SRT | |
(function (Popcorn) { | |
/** | |
* SRT popcorn parser plug-in | |
* Parses subtitle files in the SRT format. | |
* Times are expected in HH:MM:SS,MIL format, though HH:MM:SS.MIL also supported | |
* Ignore styling, which may occur after the end time or in-text | |
* While not part of the "official" spec, majority of players support HTML and SSA styling tags | |
* SSA-style tags are stripped, HTML style tags are left for the browser to handle: | |
* HTML: <font>, <b>, <i>, <u>, <s> | |
* SSA: \N or \n, {\cmdArg1}, {\cmd(arg1, arg2, ...)} | |
* Data parameter is given by Popcorn, will need a text. | |
* Text is the file contents to be parsed | |
* | |
* @param {Object} data | |
* | |
* Example: | |
1 | |
00:00:25,712 --> 00:00:30.399 | |
This text is <font color="red">RED</font> and has not been {\pos(142,120)} positioned. | |
This takes \Nup three \nentire lines. | |
This contains nested <b>bold, <i>italic, <u>underline</u> and <s>strike-through</s></u></i></b> HTML tags | |
Unclosed but <b>supported tags are left in | |
<ggg>Unsupported</ggg> HTML tags are left in, even if <hhh>not closed. | |
SSA tags with {\i1} would open and close italicize {\i0}, but are stripped | |
Multiple {\pos(142,120)\b1}SSA tags are stripped | |
*/ | |
Popcorn.parser( "parseSRT", function( data ) { | |
// declare needed variables | |
var retObj = { | |
title: "", | |
remote: "", | |
data: [] | |
}, | |
subs = [], | |
i = 0, | |
len = 0, | |
idx = 0, | |
lines, | |
time, | |
text, | |
sub; | |
// Simple function to convert HH:MM:SS,MMM or HH:MM:SS.MMM to SS.MMM | |
// Assume valid, returns 0 on error | |
var toSeconds = function( t_in ) { | |
var t = t_in.split( ':' ); | |
try { | |
var s = t[2].split( ',' ); | |
// Just in case a . is decimal seperator | |
if ( s.length === 1 ) { | |
s = t[2].split( '.' ); | |
} | |
return parseFloat( t[0], 10 )*3600 + parseFloat( t[1], 10 )*60 + parseFloat( s[0], 10 ) + parseFloat( s[1], 10 )/1000; | |
} catch ( e ) { | |
return 0; | |
} | |
}; | |
var createTrack = function( name, attributes ) { | |
var track = {}; | |
track[name] = attributes; | |
return track; | |
}; | |
// Here is where the magic happens | |
// Split on line breaks | |
lines = data.text.split( /(?:\r\n|\r|\n)/gm ); | |
len = lines.length; | |
for( i=0; i < len; i++ ) { | |
sub = {}; | |
text = []; | |
sub.id = parseInt( lines[i++], 10 ); | |
// Split on '-->' delimiter, trimming spaces as well | |
time = lines[i++].split( /[\t ]*-->[\t ]*/ ); | |
sub.start = toSeconds( time[0] ); | |
// So as to trim positioning information from end | |
idx = time[1].indexOf( " " ); | |
if ( idx !== -1) { | |
time[1] = time[1].substr( 0, idx ); | |
} | |
sub.end = toSeconds( time[1] ); | |
// Build single line of text from multi-line subtitle in file | |
while ( i < len && lines[i] ) { | |
text.push( lines[i++] ); | |
} | |
// Join into 1 line, SSA-style linebreaks | |
// Strip out other SSA-style tags | |
sub.text = text.join( "\\N" ).replace( /\{(\\[\w]+\(?([\w\d]+,?)+\)?)+\}/gi, "" ); | |
// Escape HTML entities | |
sub.text = sub.text.replace( /</g, "<" ).replace( />/g, ">" ); | |
// Unescape great than and less than when it makes a valid html tag of a supported style (font, b, u, s, i) | |
// Modified version of regex from Phil Haack's blog: http://haacked.com/archive/2004/10/25/usingregularexpressionstomatchhtml.aspx | |
// Later modified by kev: http://kevin.deldycke.com/2007/03/ultimate-regular-expression-for-html-tag-parsing-with-php/ | |
sub.text = sub.text.replace( /<(\/?(font|b|u|i|s))((\s+(\w|\w[\w\-]*\w)(\s*=\s*(?:\".*?\"|'.*?'|[^'\">\s]+))?)+\s*|\s*)(\/?)>/gi, "<$1$3$7>" ); | |
sub.text = sub.text.replace( /\\N/gi, "<br />" ); | |
subs.push( createTrack( "subtitle", sub ) ); | |
} | |
retObj.data = subs; | |
return retObj; | |
}); | |
})( Popcorn ); | |
// PARSER: 0.3 WebSRT/VTT | |
(function ( Popcorn ) { | |
/** | |
* WebSRT/VTT popcorn parser plug-in | |
* Parses subtitle files in the WebSRT/VTT format. | |
* Styles which appear after timing information are ignored. | |
* Inline styling tags follow HTML conventions and are left in for the browser to handle | |
* TrackEvents (cues) which are malformed are ignored. | |
* Data parameter is given by Popcorn, will need a text. | |
* Text is the file contents to be parsed | |
* | |
* @param {Object} data | |
* | |
* Example: | |
Track-3 | |
00:00:15.542 --> 00:00:18.542 A:start D:vertical L:98% | |
It's a <i>trap!</i> | |
*/ | |
Popcorn.parser( "parseVTT", function( data ) { | |
// declare needed variables | |
var retObj = { | |
title: "", | |
remote: "", | |
data: [] | |
}, | |
subs = [], | |
i = 0, | |
len = 0, | |
idx = 0, | |
lines, | |
time, | |
text, | |
sub; | |
// [HH:]MM:SS.mmm string to SS.mmm float | |
// Throws exception if invalid | |
var toSeconds = function( t_in ) { | |
var t = t_in.split( ":" ), | |
l = t_in.length, | |
time; | |
// Invalid time string provided | |
if ( l !== 12 && l !== 9 ) { | |
throw "Bad cue"; | |
} | |
l = t.length - 1; | |
try { | |
time = parseInt( t[l-1], 10 )*60 + parseFloat( t[l], 10 ); | |
// Hours were given | |
if ( l === 2 ) { | |
time += parseInt( t[0], 10 )*3600; | |
} | |
} catch ( e ) { | |
throw "Bad cue"; | |
} | |
return time; | |
}; | |
var createTrack = function( name, attributes ) { | |
var track = {}; | |
track[name] = attributes; | |
return track; | |
}; | |
// Here is where the magic happens | |
// Split on line breaks | |
lines = data.text.split( /(?:\r\n|\r|\n)/gm ); | |
len = lines.length; | |
while ( i < len ) { | |
sub = {}; | |
text = []; | |
try { | |
sub.id = lines[i++]; | |
// Ignore if id contains "-->" | |
if ( !sub.id || sub.id.indexOf( "-->" ) !== -1 ) { | |
throw "Bad cue"; | |
} | |
time = lines[i++].split( /[\t ]*-->[\t ]*/ ); | |
sub.start = toSeconds(time[0]); | |
// Filter out any trailing styling info | |
idx = time[1].indexOf( " " ); | |
if ( idx !== -1 ) { | |
time[1] = time[1].substr( 0, idx ); | |
} | |
sub.end = toSeconds( time[1] ); | |
// Build single line of text from multi-line subtitle in file | |
while ( i < len && lines[i] ) { | |
text.push( lines[i++] ); | |
} | |
// Join lines together to one and build subtitle | |
sub.text = text.join( "<br />" ); | |
subs.push( createTrack( "subtitle", sub ) ); | |
} catch ( e ) { | |
// Bad cue, advance to end of cue | |
while ( i < len && lines[i] ) { | |
i++; | |
} | |
} | |
// Consume empty whitespace after a cue | |
while ( i < len && !lines[i] ) { | |
i++; | |
} | |
} | |
retObj.data = subs; | |
return retObj; | |
}); | |
})( Popcorn ); | |
(function( global, doc ) { | |
Popcorn.baseplayer = function() { | |
return new Popcorn.baseplayer.init(); | |
}; | |
Popcorn.baseplayer.init = function() { | |
this.readyState = 0; | |
this.currentTime = 0; | |
this.baselineTime = new Date(); | |
this.duration = 0; | |
this.paused = 1; | |
this.ended = 0; | |
this.volume = 1; | |
this.muted = 0; | |
this.playbackRate = 1; | |
// These are considered to be "on" by being defined. Initialize to undefined | |
this.autoplay; | |
this.loop; | |
// List of events | |
this._events = {}; | |
// The underlying player resource. May be <canvas>, <iframe>, <object>, array, etc | |
this._resource; | |
// The container div of the resource | |
this._container; | |
this.offsetWidth = this.width = 0; | |
this.offsetHeight = this.height = 0; | |
this.offsetLeft = 0; | |
this.offsetTop = 0; | |
this.offsetParent; | |
}; | |
Popcorn.baseplayer.init.prototype = { | |
load: function() {}, | |
play: function() { | |
this.paused = 0; | |
this.timeupdate(); | |
}, | |
pause: function() { | |
this.paused = 1; | |
}, | |
timeupdate: function() { | |
// So we can refer to the instance when setTimeout is run | |
var self = this; | |
if( !this.paused ) { | |
this.currentTime += ( new Date() - this.baselineTime ) / 1000; | |
this.dispatchEvent( "timeupdate" ); | |
} | |
this.baselineTime = new Date(); | |
setTimeout(function() { | |
self.timeupdate.call( self ); | |
}, 50 ); | |
}, | |
// By default, assumes this.resource is a DOM Element | |
// Changing the type of this.resource requires this method to be overridden | |
getBoundingClientRect: function() { | |
return Popcorn.position( this._resource || this._container ); | |
}, | |
// Add an event listener to the object | |
addEventListener: function( evtName, fn ) { | |
if ( !this._events[ evtName ] ) { | |
this._events[ evtName ] = []; | |
} | |
this._events[ evtName ].push( fn ); | |
return fn; | |
}, | |
// Can take event object or simple string | |
dispatchEvent: function( oEvent ) { | |
var evt, | |
self = this, | |
eventInterface, | |
eventName = oEvent.type; | |
// A string was passed, create event object | |
if ( !eventName ) { | |
eventName = oEvent; | |
eventInterface = Popcorn.events.getInterface( eventName ); | |
if ( eventInterface ) { | |
evt = document.createEvent( eventInterface ); | |
evt.initEvent( eventName, true, true, window, 1 ); | |
} | |
} | |
Popcorn.forEach( this._events[ eventName ], function( val ) { | |
val.call( self, evt, self ); | |
}); | |
}, | |
// Extracts values from container onto this object | |
extractContainerValues: function( id ) { | |
this._container = document.getElementById( id ); | |
if ( !this._container ) { | |
return; | |
} | |
var bounds = this._container.getBoundingClientRect(); | |
this.offsetWidth = this.width = this.getStyle( "width" ) || 0; | |
this.offsetHeight = this.height = this.getStyle( "height" ) || 0; | |
this.offsetLeft = bounds.left; | |
this.offsetTop = bounds.top; | |
this.offsetParent = this._container.offsetParent; | |
return this._container; | |
}, | |
// By default, assumes this.resource is a DOM Element | |
// Changing the type of this.resource requires this method to be overridden | |
// Returns the computed value for CSS style 'prop' as computed by the browser | |
getStyle: function( prop ) { | |
var elem = this._resource || this._container; | |
if ( elem.currentStyle ) { | |
// IE syntax | |
return elem.currentStyle[ prop ]; | |
} else if ( global.getComputedStyle ) { | |
// Firefox, Chrome et. al | |
return doc.defaultView.getComputedStyle( elem, null ).getPropertyValue( prop ); | |
} else { | |
// Fallback, just in case | |
return elem.style[ prop ]; | |
} | |
} | |
}; | |
})( window, document ); | |
// Popcorn Vimeo Player Wrapper | |
( function( Popcorn, global ) { | |
/** | |
* Vimeo wrapper for Popcorn. | |
* This player adds enables Popcorn.js to handle Vimeo videos. It does so by masking an embedded Vimeo video Flash object | |
* as a video and implementing the HTML5 Media Element interface. | |
* | |
* You can specify the video in four ways: | |
* 1. Use the embed code path supplied by Vimeo as a div's src, and pass the div id into a new Popcorn.vimeo object | |
* | |
* <div id="player_1" width="500" height="281" src="http://player.vimeo.com/video/11127501" ></div> | |
* <script type="text/javascript"> | |
* document.addEventListener("DOMContentLoaded", function() { | |
* var popcorn = Popcorn( Popcorn.vimeo( "player_1" ) ); | |
* }, false); | |
* </script> | |
& | |
* 2. Pass the div id and the embed code path supplied by Vimeo into a new Popcorn.vimeo object | |
* | |
* <div id="player_1" width="500" height="281" ></div> | |
* <script type="text/javascript"> | |
* document.addEventListener("DOMContentLoaded", function() { | |
* var popcorn = Popcorn( Popcorn.vimeo( "player_1", "http://player.vimeo.com/video/11127501" ) ); | |
* }, false); | |
* </script> | |
* | |
* 3. Use a web url as a div's src, and pass the div id into a new Popcorn.vimeo object | |
* | |
* <div id="player_1" width="500" height="281" src="http://vimeo.com/11127501" ></div> | |
* <script type="text/javascript"> | |
* document.addEventListener("DOMContentLoaded", function() { | |
* var popcorn = Popcorn( Popcorn.vimeo( "player_1" ) ); | |
* }, false); | |
* </script> | |
* | |
* 4. Pass the div id and the web url into a new Popcorn.vimeo object | |
* | |
* <div id="player_1" width="500" height="281" ></div> | |
* <script type="text/javascript"> | |
* document.addEventListener("DOMContentLoaded", function() { | |
* var popcorn = Popcorn( Popcorn.vimeo( "player_1", "http://vimeo.com/11127501" ) ); | |
* }, false); | |
* </script> | |
* | |
* Due to Vimeo's API, certain events must be subscribed to at different times, and some not at all. | |
* These events are completely custom-implemented and may be subscribed to at any time: | |
* canplaythrough | |
* durationchange | |
* load | |
* loadedmetadata | |
* loadstart | |
* play | |
* readystatechange | |
* volumechange | |
* | |
* These events are related to player functionality and must be subscribed to during or after the load event: | |
* abort | |
* emptied | |
* ended | |
* pause | |
* playing | |
* progress | |
* seeked | |
* timeupdate | |
* | |
* These events are not supported: | |
* canplay | |
* error | |
* loadeddata | |
* ratechange | |
* seeking | |
* stalled | |
* suspend | |
* waiting | |
* | |
* Due to Vimeo's API, some attributes are be supported while others are not. | |
* Supported media attributes: | |
* autoplay ( via Popcorn ) | |
* currentTime | |
* duration ( get only ) | |
* ended ( get only ) | |
* initialTime ( get only, always 0 ) | |
* loop ( get only, set by calling setLoop() ) | |
* muted ( get only ) | |
* paused ( get only ) | |
* readyState ( get only ) | |
* volume | |
* | |
* load() function | |
* mute() function ( toggles on/off ) | |
* | |
* Unsupported media attributes: | |
* buffered | |
* defaultPlaybackRate | |
* networkState | |
* playbackRate | |
* played | |
* preload | |
* seekable | |
* seeking | |
* src | |
* startOffsetTime | |
*/ | |
// Trackers | |
var timeupdateInterval = 33, | |
timeCheckInterval = 0.75, | |
abs = Math.abs, | |
registry = {}; | |
// base object for DOM-related behaviour like events | |
var EventManager = function ( owner ) { | |
var evts = {}; | |
function makeHandler( evtName ) { | |
if ( !evts[evtName] ) { | |
evts[evtName] = []; | |
// Create a wrapper function to all registered listeners | |
this["on"+evtName] = function( args ) { | |
Popcorn.forEach( evts[evtName], function( fn ) { | |
if ( fn ) { | |
fn.call( owner, args ); | |
} | |
}); | |
}; | |
} | |
} | |
return { | |
addEventListener: function( evtName, fn, doFire ) { | |
evtName = evtName.toLowerCase(); | |
makeHandler.call( this, evtName ); | |
evts[evtName].push( fn ); | |
if ( doFire ) { | |
dispatchEvent( evtName ); | |
} | |
return fn; | |
}, | |
// Add many listeners for a single event | |
// Takes an event name and array of functions | |
addEventListeners: function( evtName, events ) { | |
evtName = evtName.toLowerCase(); | |
makeHandler.call( this, evtName ); | |
evts[evtName] = evts[evtName].concat( events ); | |
}, | |
removeEventListener: function( evtName, fn ) { | |
var evtArray = this.getEventListeners( evtName ), | |
i, | |
l; | |
// Find and remove from events array | |
for ( i = 0, l = evtArray.length; i < l; i++) { | |
if ( evtArray[i] === fn ) { | |
var removed = evtArray[i]; | |
evtArray[i] = 0; | |
return removed; | |
} | |
} | |
}, | |
getEventListeners: function( evtName ) { | |
if( evtName ) { | |
return evts[ evtName.toLowerCase() ] || []; | |
} else { | |
return evts; | |
} | |
}, | |
dispatchEvent: function( evt, args ) { | |
// If event object was passed in, toString will yield event type as string (timeupdate) | |
// If a string, toString() will return the string itself (timeupdate) | |
evt = "on"+evt.toString().toLowerCase(); | |
this[evt] && this[evt]( args ); | |
} | |
}; | |
}; | |
Popcorn.vimeo = function( mediaId, list, options ) { | |
return new Popcorn.vimeo.init( mediaId, list, options ); | |
}; | |
Popcorn.vimeo.onLoad = function( playerId ) { | |
var player = registry[ playerId ]; | |
player.swfObj = document.getElementById( playerId ); | |
// For calculating position relative to video (like subtitles) | |
player.offsetWidth = player.swfObj.offsetWidth; | |
player.offsetHeight = player.swfObj.offsetHeight; | |
player.offsetParent = player.swfObj.offsetParent; | |
player.offsetLeft = player.swfObj.offsetLeft; | |
player.offsetTop = player.swfObj.offsetTop; | |
player.dispatchEvent( "load" ); | |
}; | |
Popcorn.getScript( "http://ajax.googleapis.com/ajax/libs/swfobject/2.2/swfobject.js" ); | |
// A constructor, but we need to wrap it to allow for "static" functions | |
Popcorn.vimeo.init = (function() { | |
var rPlayerUri = /^http:\/\/player\.vimeo\.com\/video\/[\d]+/i, | |
rWebUrl = /vimeo\.com\/[\d]+/, | |
hasAPILoaded = false; | |
// Extract the numeric video id from container uri: 'http://player.vimeo.com/video/11127501' or 'http://player.vimeo.com/video/4282282' | |
// Expect id to be a valid 32/64-bit unsigned integer | |
// Returns string, empty string if could not match | |
function extractIdFromUri( uri ) { | |
if ( !uri ) { | |
return; | |
} | |
var matches = uri.match( rPlayerUri ); | |
return matches ? matches[0].substr(30) : ""; | |
} | |
// Extract the numeric video id from url: 'http://vimeo.com/11127501' or simply 'vimeo.com/4282282' | |
// Ignores protocol and subdomain, but one would expecct it to be http://www.vimeo.com/####### | |
// Expect id to be a valid 32/64-bit unsigned integer | |
// Returns string, empty string if could not match | |
function extractIdFromUrl( url ) { | |
if ( !url ) { | |
return; | |
} | |
var matches = url.match( rWebUrl ); | |
return matches ? matches[0].substr(10) : ""; | |
} | |
function makeSwf( self, vidId, containerId ) { | |
if ( !window.swfobject ) { | |
setTimeout( function() { | |
makeSwf( self, vidId, containerId ); | |
}, 1); | |
return; | |
} | |
var params, | |
flashvars, | |
attributes = {}; | |
flashvars = { | |
clip_id: vidId, | |
show_portrait: 1, | |
show_byline: 1, | |
show_title: 1, | |
// required in order to use the Javascript API | |
js_api: 1, | |
// moogaloop will call this JS function when it's done loading (optional) | |
js_onLoad: 'Popcorn.vimeo.onLoad', | |
// this will be passed into all event methods so you can keep track of multiple moogaloops (optional) | |
js_swf_id: containerId | |
}; | |
params = { | |
allowscriptaccess: 'always', | |
allowfullscreen: 'true', | |
// This is so we can overlay html ontop o fFlash | |
wmode: 'transparent' | |
}; | |
swfobject.embedSWF( "http://vimeo.com/moogaloop.swf", containerId, self.offsetWidth, self.offsetHeight, "9.0.0", "expressInstall.swf", flashvars, params, attributes ); | |
} | |
// If container id is not supplied, assumed to be same as player id | |
var ctor = function ( containerId, videoUrl, options ) { | |
if ( !containerId ) { | |
throw "Must supply an id!"; | |
} else if ( /file/.test( location.protocol ) ) { | |
throw "Must run from a web server!"; | |
} | |
var vidId, | |
that = this, | |
tmp; | |
this._container = document.createElement( "div" ); | |
this._container.id = containerId + "object"; | |
this._target = document.getElementById( containerId ); | |
this._target.appendChild( this._container ); | |
options = options || {}; | |
options.css && Popcorn.extend( this._target.style, options.css ); | |
this.addEventFn; | |
this.evtHolder; | |
this.paused = true; | |
this.duration = Number.MAX_VALUE; | |
this.ended = 0; | |
this.currentTime = 0; | |
this.volume = 1; | |
this.loop = 0; | |
this.initialTime = 0; | |
this.played = 0; | |
this.readyState = 0; | |
this.previousCurrentTime = this.currentTime; | |
this.previousVolume = this.volume; | |
this.evtHolder = new EventManager( this ); | |
// For calculating position relative to video (like subtitles) | |
this.width = this._target.style.width || "504px"; | |
this.height = this._target.style.height || "340px"; | |
if ( !/[\d]%/.test( this.width ) ) { | |
this.offsetWidth = parseInt( this.width, 10 ); | |
this._target.style.width = this.width + "px"; | |
} else { | |
// convert from pct to abs pixels | |
tmp = this._target.style.width; | |
this._target.style.width = this.width; | |
this.offsetWidth = this._target.offsetWidth; | |
this._target.style.width = tmp; | |
} | |
if ( !/[\d]%/.test( this.height ) ) { | |
this.offsetHeight = parseInt( this.height, 10 ); | |
this._target.style.height = this.height + "px"; | |
} else { | |
// convert from pct to abs pixels | |
tmp = this._target.style.height; | |
this._target.style.height = this.height; | |
this.offsetHeight = this._target.offsetHeight; | |
this._target.style.height = tmp; | |
} | |
this.offsetLeft = 0; | |
this.offsetTop = 0; | |
// Try and get a video id from a vimeo site url | |
// Try either from ctor param or from iframe itself | |
vidId = extractIdFromUrl( videoUrl ) || extractIdFromUri( videoUrl ); | |
if ( !vidId ) { | |
throw "No video id"; | |
} | |
registry[ this._container.id ] = this; | |
makeSwf( this, vidId, this._container.id ); | |
// Set up listeners to internally track state as needed | |
this.addEventListener( "load", function() { | |
var hasLoaded = false; | |
that.duration = that.swfObj.api_getDuration(); | |
that.evtHolder.dispatchEvent( "durationchange" ); | |
that.evtHolder.dispatchEvent( "loadedmetadata" ); | |
// Chain events and calls together so that this.currentTime reflects the current time of the video | |
// Done by Getting the Current Time while the video plays | |
that.addEventListener( "timeupdate", function() { | |
that.currentTime = that.swfObj.api_getCurrentTime(); | |
}); | |
// Add pause listener to keep track of playing state | |
that.addEventListener( "pause", function() { | |
that.paused = true; | |
}); | |
// Add play listener to keep track of playing state | |
that.addEventListener( "playing", function() { | |
that.paused = false; | |
that.ended = 0; | |
}); | |
// Add ended listener to keep track of playing state | |
that.addEventListener( "ended", function() { | |
if ( that.loop !== "loop" ) { | |
that.paused = true; | |
that.ended = 1; | |
} | |
}); | |
// Add progress listener to keep track of ready state | |
that.addEventListener( "progress", function( data ) { | |
if ( !hasLoaded ) { | |
hasLoaded = 1; | |
that.readyState = 3; | |
that.evtHolder.dispatchEvent( "readystatechange" ); | |
} | |
// Check if fully loaded | |
if ( data.percent === 100 ) { | |
that.readyState = 4; | |
that.evtHolder.dispatchEvent( "readystatechange" ); | |
that.evtHolder.dispatchEvent( "canplaythrough" ); | |
} | |
}); | |
}); | |
}; | |
return ctor; | |
})(); | |
Popcorn.vimeo.init.prototype = Popcorn.vimeo.prototype; | |
// Sequence object prototype | |
Popcorn.extend( Popcorn.vimeo.prototype, { | |
// Do everything as functions instead of get/set | |
setLoop: function( val ) { | |
if ( !val ) { | |
return; | |
} | |
this.loop = val; | |
var isLoop = val === "loop" ? 1 : 0; | |
// HTML convention says to loop if value is 'loop' | |
this.swfObj.api_setLoop( isLoop ); | |
}, | |
// Set the volume as a value between 0 and 1 | |
setVolume: function( val ) { | |
if ( !val && val !== 0 ) { | |
return; | |
} | |
// Normalize in case outside range of expected values | |
if ( val < 0 ) { | |
val = -val; | |
} | |
if ( val > 1 ) { | |
val %= 1; | |
} | |
// HTML video expects to be 0.0 -> 1.0, Vimeo expects 0-100 | |
this.volume = this.previousVolume = val; | |
this.swfObj.api_setVolume( val*100 ); | |
this.evtHolder.dispatchEvent( "volumechange" ); | |
}, | |
// Seeks the video | |
setCurrentTime: function ( time ) { | |
if ( !time && time !== 0 ) { | |
return; | |
} | |
this.currentTime = this.previousCurrentTime = time; | |
this.ended = time >= this.duration; | |
this.swfObj.api_seekTo( time ); | |
// Fire events for seeking and time change | |
this.evtHolder.dispatchEvent( "seeked" ); | |
this.evtHolder.dispatchEvent( "timeupdate" ); | |
}, | |
// Play the video | |
play: function() { | |
// In case someone is cheeky enough to try this before loaded | |
if ( !this.swfObj ) { | |
this.addEventListener( "load", this.play ); | |
return; | |
} | |
if ( !this.played ) { | |
this.played = 1; | |
this.startTimeUpdater(); | |
this.evtHolder.dispatchEvent( "loadstart" ); | |
} | |
this.evtHolder.dispatchEvent( "play" ); | |
this.swfObj.api_play(); | |
}, | |
// Pause the video | |
pause: function() { | |
// In case someone is cheeky enough to try this before loaded | |
if ( !this.swfObj ) { | |
this.addEventListener( "load", this.pause ); | |
return; | |
} | |
this.swfObj.api_pause(); | |
}, | |
// Toggle video muting | |
// Unmuting will leave it at the old value | |
mute: function() { | |
// In case someone is cheeky enough to try this before loaded | |
if ( !this.swfObj ) { | |
this.addEventListener( "load", this.mute ); | |
return; | |
} | |
if ( !this.muted() ) { | |
this.oldVol = this.volume; | |
if ( this.paused ) { | |
this.setVolume( 0 ); | |
} else { | |
this.volume = 0; | |
} | |
} else { | |
if ( this.paused ) { | |
this.setVolume( this.oldVol ); | |
} else { | |
this.volume = this.oldVol; | |
} | |
} | |
}, | |
muted: function() { | |
return this.volume === 0; | |
}, | |
// Force loading by playing the player. Pause afterwards | |
load: function() { | |
// In case someone is cheeky enough to try this before loaded | |
if ( !this.swfObj ) { | |
this.addEventListener( "load", this.load ); | |
return; | |
} | |
this.play(); | |
this.pause(); | |
}, | |
unload: function() { | |
// In case someone is cheeky enough to try this before loaded | |
if ( !this.swfObj ) { | |
this.addEventListener( "load", this.unload ); | |
return; | |
} | |
this.pause(); | |
this.swfObj.api_unload(); | |
this.evtHolder.dispatchEvent( "abort" ); | |
this.evtHolder.dispatchEvent( "emptied" ); | |
}, | |
// Hook an event listener for the player event into internal event system | |
// Stick to HTML conventions of add event listener and keep lowercase, without prependinng "on" | |
addEventListener: function( evt, fn ) { | |
var playerEvt, | |
that = this; | |
// In case event object is passed in | |
evt = evt.type || evt.toLowerCase(); | |
// If it's an HTML media event supported by player, map | |
if ( evt === "seeked" ) { | |
playerEvt = "onSeek"; | |
} else if ( evt === "timeupdate" ) { | |
playerEvt = "onProgress"; | |
} else if ( evt === "progress" ) { | |
playerEvt = "onLoading"; | |
} else if ( evt === "ended" ) { | |
playerEvt = "onFinish"; | |
} else if ( evt === "playing" ) { | |
playerEvt = "onPlay"; | |
} else if ( evt === "pause" ) { | |
// Direct mapping, CamelCase the event name as vimeo API expects | |
playerEvt = "on"+evt[0].toUpperCase() + evt.substr(1); | |
} | |
// Vimeo only stores 1 callback per event | |
// Have vimeo call internal collection of callbacks | |
this.evtHolder.addEventListener( evt, fn, false ); | |
// Link manual event structure with Vimeo's if not already | |
if( playerEvt && this.evtHolder.getEventListeners( evt ).length === 1 ) { | |
// Setup global functions on Popcorn.vimeo to sync player events to an internal collection | |
// Some events expect 2 args, some only one (the player id) | |
if ( playerEvt === "onSeek" || playerEvt === "onProgress" || playerEvt === "onLoading" ) { | |
Popcorn.vimeo[playerEvt] = function( arg1, arg2 ) { | |
var player = registry[arg2]; | |
player.evtHolder.dispatchEvent( evt, arg1 ); | |
}; | |
} else { | |
Popcorn.vimeo[playerEvt] = function( arg1 ) { | |
var player = registry[arg1]; | |
player.evtHolder.dispatchEvent( evt ); | |
}; | |
} | |
this.swfObj.api_addEventListener( playerEvt, "Popcorn.vimeo."+playerEvt ); | |
} | |
}, | |
removeEventListener: function( evtName, fn ) { | |
return this.evtHolder.removeEventListener( evtName, fn ); | |
}, | |
dispatchEvent: function( evtName ) { | |
return this.evtHolder.dispatchEvent( evtName ); | |
}, | |
getBoundingClientRect: function() { | |
return this._target.getBoundingClientRect(); | |
}, | |
startTimeUpdater: function() { | |
var self = this, | |
seeked = 0; | |
if ( abs( this.currentTime - this.previousCurrentTime ) > timeCheckInterval ) { | |
// Has programatically set the currentTime | |
this.setCurrentTime( this.currentTime ); | |
seeked = 1; | |
} else { | |
this.previousCurrentTime = this.currentTime; | |
} | |
if ( this.volume !== this.previousVolume ) { | |
this.setVolume( this.volume ); | |
} | |
if ( !self.paused || seeked ) { | |
this.dispatchEvent( 'timeupdate' ); | |
} | |
if( !self.ended ) { | |
setTimeout( function() { | |
self.startTimeUpdater.call(self); | |
}, timeupdateInterval); | |
} | |
} | |
}); | |
})( Popcorn, window ); | |
// Popcorn Youtube Player Wrapper | |
var onYouTubePlayerReady; | |
( function( Popcorn ) { | |
/** | |
* Youtube wrapper for popcorn. | |
* This plug-in adds capability for Popcorn.js to deal with Youtube | |
* videos. This plug-in also doesn't use Popcorn's plugin() API and | |
* instead hacks directly into Popcorn's core. | |
* | |
* To use this plug-in, onYouTubePlayerReady() event handler needs to be | |
* called by the Youtube video player, before videos can be registered. | |
* Once videos are registered, calls to them can be made the same way as | |
* regular Popcorn objects. Also note that enablejsapi=1 needs to be added | |
* to the embed code, in order for Youtube's JavaScript API to work. | |
* | |
* Note that there are a few methods, properties and events that are not | |
* supported. See the bottom of this plug-in for a complete list. | |
*/ | |
// Intended | |
var undef; | |
// Config parameters | |
// 33 ms per update is suitable for 30 fps | |
// 0.05 sec tolerance between old and new times to determine if currentTime has been set programatically | |
// 250 ms progress interval as specified by WHATWG | |
var timeupdateInterval = 33, | |
timeCheckInterval = 0.5, | |
progressInterval = 250; | |
// Ready State Constants | |
var READY_STATE_HAVE_NOTHING = 0, | |
READY_STATE_HAVE_METADATA = 1, | |
READY_STATE_HAVE_CURRENT_DATA = 2, | |
READY_STATE_HAVE_FUTURE_DATA = 3, | |
READY_STATE_HAVE_ENOUGH_DATA = 4; | |
// Youtube State Constants | |
var YOUTUBE_STATE_UNSTARTED = -1, | |
YOUTUBE_STATE_ENDED = 0, | |
YOUTUBE_STATE_PLAYING = 1, | |
YOUTUBE_STATE_PAUSED = 2, | |
YOUTUBE_STATE_BUFFERING = 3, | |
YOUTUBE_STATE_CUED = 5; | |
var urlRegex = /^.*[\/=](.{11})/; | |
// Collection of all Youtube players | |
var registry = {}, | |
loadedPlayers = {}; | |
var abs = Math.abs; | |
Popcorn.getScript( "http://ajax.googleapis.com/ajax/libs/swfobject/2.2/swfobject.js" ); | |
// Extract the id from a web url | |
function extractIdFromUrl( url ) { | |
if ( !url ) { | |
return; | |
} | |
var matches = urlRegex.exec( url ); | |
// Return id, which comes after first equals sign | |
return matches ? matches[1] : ""; | |
} | |
// Extract the id from a player url | |
function extractIdFromUri( url ) { | |
if ( !url ) { | |
return; | |
} | |
var matches = urlRegex.exec( url ); | |
// Return id, which comes after first equals sign | |
return matches ? matches[1] : ""; | |
} | |
function getPlayerAddress( vidId, playerId ) { | |
if( !vidId ) { | |
return; | |
} | |
return "http://www.youtube.com/e/" + id; | |
} | |
function makeSWF( url, container ) { | |
var bounds, | |
params, | |
flashvars, | |
attributes, | |
self = this; | |
if ( !window.swfobject ) { | |
setTimeout( function() { | |
makeSWF.call( self, url, container ); | |
}, 1 ); | |
return; | |
} | |
bounds = container.getBoundingClientRect(); | |
// Determine width/height/etc based on container | |
this.width = container.style.width || 460; | |
this.height = container.style.height || 350; | |
// Just in case we got the attributes as strings. We'll need to do math with these later | |
this.width = parseFloat(this.width); | |
this.height = parseFloat(this.height); | |
// For calculating position relative to video (like subtitles) | |
this.offsetWidth = this.width; | |
this.offsetHeight = this.height; | |
this.offsetLeft = bounds.left; | |
this.offsetTop = bounds.top; | |
this.offsetParent = container.offsetParent; | |
flashvars = { | |
playerapiid: this.playerId, | |
controls: this.controls, | |
iv_load_policy: this.iv_load_policy | |
}; | |
params = { | |
allowscriptaccess: 'always', | |
allowfullscreen: 'true', | |
// This is so we can overlay html on top of Flash | |
wmode: 'transparent' | |
}; | |
attributes = { | |
id: this.playerId | |
}; | |
swfobject.embedSWF( "http://www.youtube.com/e/" + this.vidId +"?enablejsapi=1&playerapiid=" + this.playerId + "&version=3", | |
this.playerId, this.width, this.height, "8", null, flashvars, params, attributes ); | |
} | |
// Called when a player is loaded | |
// Playerid must match the element id | |
onYouTubePlayerReady = function ( playerId ) { | |
var vid = registry[playerId]; | |
loadedPlayers[playerId] = 1; | |
// Video hadn't loaded yet when ctor was called | |
vid.video = document.getElementById( playerId ); | |
vid.duration = vid.video.getDuration(); | |
// Issue load event | |
vid.dispatchEvent( 'load' ); | |
vid.dispatchEvent( "durationchange" ); | |
}; | |
Popcorn.youtube = function( elementId, url, options ) { | |
return new Popcorn.youtube.init( elementId, url, options ); | |
}; | |
Popcorn.youtube.init = function( elementId, url, options ) { | |
if ( !elementId ) { | |
throw "Element id is invalid."; | |
} else if ( /file/.test( location.protocol ) ) { | |
throw "This must be run from a web server."; | |
} | |
options = options || {}; | |
var self = this; | |
this.playerId = elementId + Popcorn.guid(); | |
this.readyState = READY_STATE_HAVE_NOTHING; | |
this._eventListeners = {}; | |
this.loadStarted = false; | |
this.loadedData = false; | |
this.fullyLoaded = false; | |
this.paused = false; | |
// If supplied as number, append 'px' on end | |
// If suppliied as '###' or '###px', convert to number and append 'px' back on end | |
options.width = options.width && (+options.width)+"px"; | |
options.height = options.height && (+options.height)+"px"; | |
// show controls on video. Integer value - 1 is for show, 0 is for hide | |
this.controls = +options.controls === 0 || +options.controls === 1 ? options.controls : 1; | |
// show video annotations, 1 is show, 3 is hide | |
this.iv_load_policy = +options.annotations === 1 || +options.annotations === 3 ? options.annotations : 1; | |
this._target = document.getElementById( elementId ); | |
this._container = document.createElement( "div" ); | |
this._container.id = this.playerId; | |
this._container.style.height = this._target.style.height = options.height || this._target.style.height || "350px"; | |
this._container.style.width = this._target.style.width = options.width || this._target.style.width || "460px"; | |
this._target.appendChild( this._container ); | |
this.offsetHeight = +this._target.offsetHeight; | |
this.offsetWidth = +this._target.offsetWidth; | |
this.currentTime = this.previousCurrentTime = 0; | |
this.volume = this.previousVolume = this.preMuteVol = 1; | |
this.duration = 0; | |
this.vidId = extractIdFromUrl( url ) || extractIdFromUri( url ); | |
if ( !this.vidId ) { | |
throw "Could not find video id"; | |
} | |
this.addEventListener( "load", function() { | |
// For calculating position relative to video (like subtitles) | |
this.offsetWidth = this.video.offsetWidth; | |
this.offsetHeight = this.video.offsetHeight; | |
this.offsetParent = this.video.offsetParent; | |
// Set up stuff that requires the API to be loaded | |
this.registerYoutubeEventHandlers(); | |
this.registerInternalEventHandlers(); | |
}); | |
(function() { | |
var hasBeenCalled = 0; | |
self.addEventListener( "playing", function() { | |
if (hasBeenCalled) { | |
return; | |
} | |
hasBeenCalled = 1; | |
self.duration = self.video.getDuration(); | |
self.dispatchEvent( "durationchange" ); | |
}); | |
})(); | |
if ( loadedPlayers[this.playerId] ) { | |
this.video = registry[this.playerId].video; | |
this.vidId = this.vidId || extractIdFromUrl( this._container.getAttribute( "src" ) ) || extractIdFromUri( this._container.getAttribute( "src" ) ); | |
if (this.vidId !== registry[this.playerId].vidId ) { | |
this.video.cueVideoById( this.vidId ); | |
} else { | |
// Same video, new ctor. Force a seek to the beginning | |
this.previousCurrentTime = 1; | |
} | |
this.dispatchEvent( 'load' ); | |
} else if ( this._container ) { | |
makeSWF.call( this, url, this._container ); | |
} else { | |
// Container not yet loaded, get it on DOMDontentLoad | |
document.addEventListener( "DOMContentLoaded", function() { | |
self._container = document.getElementById( elementId ); | |
if ( !self._container ) { | |
throw "Could not find container!"; | |
} | |
makeSWF.call( self, url, self._container ); | |
}, false); | |
} | |
registry[this.playerId] = this; | |
}; | |
// end Popcorn.youtube.init | |
Popcorn.extend( Popcorn.youtube.init.prototype, { | |
// For internal use only. | |
// Register handlers to YouTube events. | |
registerYoutubeEventHandlers: function() { | |
var youcorn = this, | |
stateChangeHandler = 'Popcorn.youtube.stateChangeEventHandler', | |
errorHandler = 'Popcorn.youtube.errorEventHandler'; | |
this.video.addEventListener( 'onStateChange', stateChangeHandler ); | |
this.video.addEventListener( 'onError', errorHandler ); | |
/** | |
* Since Flash can only call named functions, they are declared | |
* separately here. | |
*/ | |
Popcorn.youtube.stateChangeEventHandler = function( state ) { | |
// In case ctor has been called many times for many ctors | |
// Only use latest ctor call for each player id | |
var self = registry[youcorn.playerId]; | |
if ( state === YOUTUBE_STATE_UNSTARTED ) { | |
self.readyState = READY_STATE_HAVE_METADATA; | |
self.dispatchEvent( 'loadedmetadata' ); | |
} else if ( state === YOUTUBE_STATE_ENDED ) { | |
self.dispatchEvent( 'ended' ); | |
} else if ( state === YOUTUBE_STATE_PLAYING ) { | |
// Being able to play means current data is loaded. | |
if ( !this.loadedData ) { | |
this.loadedData = true; | |
self.dispatchEvent( 'loadeddata' ); | |
} | |
self.readyState = READY_STATE_HAVE_CURRENT_DATA; | |
self.dispatchEvent( 'playing' ); | |
} else if ( state === YOUTUBE_STATE_PAUSED ) { | |
self.dispatchEvent( 'pause' ); | |
} else if ( state === YOUTUBE_STATE_BUFFERING ) { | |
self.dispatchEvent( 'waiting' ); | |
} else if ( state === YOUTUBE_STATE_CUED ) { | |
// not handled | |
} | |
}; | |
Popcorn.youtube.errorEventHandler = function( state ) { | |
youcorn.dispatchEvent( 'error' ); | |
}; | |
}, | |
// For internal use only. | |
// Start current time and loading progress syncing intervals. | |
registerInternalEventHandlers: function() { | |
this.addEventListener( 'playing', function() { | |
this.startTimeUpdater(); | |
}); | |
this.addEventListener( 'loadedmetadata', function() { | |
this.startProgressUpdater(); | |
}); | |
}, | |
play: function() { | |
// In case called before video is loaded, defer acting | |
if ( !loadedPlayers[this.playerId] ) { | |
this.addEventListener( "load", function() { | |
this.play(); | |
}); | |
return; | |
} | |
this.dispatchEvent( 'play' ); | |
this.video.playVideo(); | |
}, | |
pause: function() { | |
// In case called before video is loaded, defer acting | |
if ( !loadedPlayers[this.playerId] ) { | |
this.addEventListener( "load", this.pause ); | |
return; | |
} | |
this.video.pauseVideo(); | |
// pause event is raised by Youtube. | |
}, | |
load: function() { | |
// In case called before video is loaded, defer acting | |
if ( !loadedPlayers[this.playerId] ) { | |
this.addEventListener( "load", function() { | |
this.load(); | |
}); | |
return; | |
} | |
this.video.playVideo(); | |
this.video.pauseVideo(); | |
}, | |
seekTo: function( time ) { | |
var playing = this.video.getPlayerState() == YOUTUBE_STATE_PLAYING; | |
this.video.seekTo( time, true ); | |
// Prevent Youtube's behaviour to start playing video after seeking. | |
if ( !playing ) { | |
this.video.paused = true; | |
this.video.pauseVideo(); | |
} else { | |
this.video.paused = false; | |
} | |
// Data need to be loaded again. | |
if ( !this.fullyLoaded ) { | |
this.loadedData = false; | |
} | |
// Raise event. | |
this.dispatchEvent( 'seeked' ); | |
}, | |
// Mute is toggleable | |
mute: function() { | |
// In case called before video is loaded, defer acting | |
if ( !loadedPlayers[this.playerId] ) { | |
this.addEventListener( "load", this.mute ); | |
return; | |
} | |
if ( this.volume !== 0 ) { | |
this.preMuteVol = this.volume; | |
this.setVolume( 0 ); | |
} else { | |
this.setVolume( this.preMuteVol ); | |
} | |
}, | |
// Expects beteween 0 and 1 | |
setVolume: function( vol ) { | |
this.volume = this.previousVolume = vol; | |
this.video.setVolume( vol * 100 ); | |
this.dispatchEvent( 'volumechange' ); | |
}, | |
addEventListener: function( evt, func ) { | |
var evtName = evt.type || evt; | |
if ( !this._eventListeners[evtName] ) { | |
this._eventListeners[evtName] = []; | |
} | |
this._eventListeners[evtName].push( func ); | |
}, | |
/** | |
* Notify event listeners about an event. | |
*/ | |
dispatchEvent: function( name ) { | |
var evtName = name.type || name; | |
if ( !this._eventListeners[evtName] ) { | |
return; | |
} | |
var self = this; | |
Popcorn.forEach( this._eventListeners[evtName], function( evt ) { | |
evt.call( self, null ); | |
}); | |
}, | |
/* Unsupported methods. */ | |
defaultPlaybackRate: function( arg ) { | |
}, | |
playbackRate: function( arg ) { | |
}, | |
getBoundingClientRect: function() { | |
var b, | |
self = this; | |
if ( this.video ) { | |
b = this.video.getBoundingClientRect(); | |
return { | |
bottom: b.bottom, | |
left: b.left, | |
right: b.right, | |
top: b.top, | |
// These not guaranteed to be in there | |
width: b.width || ( b.right - b.left ), | |
height: b.height || ( b.bottom - b.top ) | |
}; | |
} else { | |
b = self._container.getBoundingClientRect(); | |
// Update bottom, right for expected values once the container loads | |
return { | |
left: b.left, | |
top: b.top, | |
width: self._target.offsetWidth, | |
height: self._target.offsetHeight, | |
bottom: b.top + self._target.offsetWidth, | |
right: b.left + self._target.offsetHeight | |
}; | |
} | |
}, | |
startTimeUpdater: function() { | |
var state = typeof this.video.getPlayerState != "function" ? this.readyState : this.video.getPlayerState(), | |
self = this, | |
seeked = 0; | |
if ( abs( this.currentTime - this.previousCurrentTime ) > timeCheckInterval ) { | |
// Has programatically set the currentTime | |
this.previousCurrentTime = this.currentTime - timeCheckInterval; | |
this.seekTo( this.currentTime ); | |
seeked = 1; | |
} else { | |
this.previousCurrentTime = this.currentTime; | |
this.currentTime = typeof this.video.getCurrentTime != "function" ? this.currentTime : this.video.getCurrentTime(); | |
} | |
if ( this.volume !== this.previousVolume ) { | |
this.setVolume( this.volume ); | |
} | |
if ( state !== YOUTUBE_STATE_ENDED && state !== YOUTUBE_STATE_PAUSED || seeked ) { | |
this.dispatchEvent( 'timeupdate' ); | |
} | |
if( state !== YOUTUBE_STATE_ENDED ) { | |
setTimeout( function() { | |
self.startTimeUpdater.call(self); | |
}, timeupdateInterval); | |
} | |
}, | |
startProgressUpdater: function() { | |
var bytesLoaded = this.video.getVideoBytesLoaded(), | |
bytesToLoad = this.video.getVideoBytesTotal(), | |
self = this; | |
// do nothing if size is not yet determined | |
if ( bytesToLoad === 0 ) { | |
return; | |
} | |
// raise an event if load has just started | |
if ( !this.loadStarted ) { | |
this.loadStarted = true; | |
this.dispatchEvent( 'loadstart' ); | |
} | |
// fully loaded | |
if ( bytesLoaded >= bytesToLoad ) { | |
this.fullyLoaded = true; | |
this.readyState = READY_STATE_HAVE_ENOUGH_DATA; | |
this.dispatchEvent( 'canplaythrough' ); | |
return; | |
} | |
this.dispatchEvent( 'progress' ); | |
setTimeout( function() { | |
self.startProgressUpdater.call( self ); | |
}, progressInterval); | |
} | |
}); // end Popcorn.extend | |
/* Unsupported properties and events. */ | |
/** | |
* Unsupported events are: | |
* * suspend | |
* * abort | |
* * emptied | |
* * stalled | |
* * canplay | |
* * seeking | |
* * ratechange | |
*/ | |
})( Popcorn ); | |
// Popcorn Soundcloud Player Wrapper | |
( function( Popcorn, global ) { | |
/** | |
* Soundcloud wrapper for Popcorn. | |
* This player adds enables Popcorn.js to handle Soundcloud audio. It does so by masking an embedded Soundcloud Flash object | |
* as a video and implementing the HTML5 Media Element interface. | |
* | |
* You can configure the video source and dimensions in two ways: | |
* 1. Use the embed code path supplied by Soundcloud the id of the desired location into a new Popcorn.soundcloud object. | |
* Width and height can be configured throughh CSS. | |
* | |
* <div id="player_1" style="width: 500px; height: 81px"></div> | |
* <script type="text/javascript"> | |
* document.addEventListener("DOMContentLoaded", function() { | |
* var popcorn = Popcorn( Popcorn.soundcloud( "player_1", "http://soundcloud.com/forss/flickermood" )); | |
* }, false); | |
* </script> | |
* | |
* 2. Width and height may also be configured directly with the player; this will override any CSS. This is useful for | |
* when different sizes are desired. for multiple players within the same parent container. | |
* | |
* <div id="player_1"></div> | |
* <script type="text/javascript"> | |
* document.addEventListener("DOMContentLoaded", function() { | |
* var popcorn = Popcorn( Popcorn.soundcloud( "player_1", "http://soundcloud.com/forss/flickermood", { | |
* width: "500", // Optional, will default to CSS values | |
* height: "81" // Optional, will default to CSS values | |
* })); | |
* }, false); | |
* </script> | |
* | |
* The player can be further configured to integrate with the SoundCloud API: | |
* | |
* var popcorn = Popcorn( Popcorn.soundcloud( "player_1", "http://soundcloud.com/forss/flickermood", { | |
* width: "100%", // Optional, the width for the player. May also be as '##px' | |
* // Defaults to the maximum possible width | |
* height: "81px", // Optional, the height for the player. May also be as '###%' | |
* // Defaults to 81px | |
* api: { // Optional, information for Soundcloud API interaction | |
* key: "abcdefsdfsdf", // Required for API interaction. The Soundcloud API key | |
* commentdiv: "divId_for_output", // Required for comment retrieval, the Div Id for outputting comments. | |
* commentformat: function( comment ) {} // Optional, a function to format a comment. Returns HTML string | |
* } | |
* })); | |
* | |
* Comments are retrieved from Soundcloud when the player is registered with Popcorn by calling the registerWithPopcorn() | |
* function. For this to work, the api_key and commentdiv attributes must be set. Comments are output by default similar to | |
* how Soundcloud formats them in-player, but a custom formatting function may be supplied. It receives a comment object and | |
* the current date. A comment object has: | |
* | |
* var comment = { | |
* start: 0, // Required. Start time in ms. | |
* date: new Date(), // Required. Date comment wasa posted. | |
* text: "", // Required. Comment text | |
* user: { // Required. Describes the user who posted the comment | |
* name: "", // Required. User name | |
* profile: "", // Required. User profile link | |
* avatar: "" // Required. User avatar link | |
* } | |
* } | |
* | |
* These events are completely custom-implemented and may be subscribed to at any time: | |
* canplaythrough | |
* durationchange | |
* load | |
* loadedmetadata | |
* loadstart | |
* play | |
* readystatechange | |
* volumechange | |
* | |
* These events are related to player functionality and must be subscribed to during or after the load event: | |
* canplay | |
* ended | |
* error | |
* pause | |
* playing | |
* progress | |
* seeked | |
* timeupdate | |
* | |
* These events are not supported: | |
* abort | |
* emptied | |
* loadeddata | |
* ratechange | |
* seeking | |
* stalled | |
* suspend | |
* waiting | |
* | |
* Supported media attributes: | |
* autoplay ( via Popcorn ) | |
* currentTime | |
* defaultPlaybackRate ( get only ) | |
* duration ( get only ) | |
* ended ( get only ) | |
* initialTime ( get only, always 0 ) | |
* loop ( get only, set by calling setLoop() ) | |
* muted ( get only ) | |
* paused ( get only ) | |
* playbackRate ( get only ) | |
* played ( get only, 0/1 only ) | |
* readyState ( get only ) | |
* src ( get only ) | |
* volume | |
* | |
* load() function | |
* mute() function ( toggles on/off ) | |
* play() function | |
* pause() function | |
* | |
* Unsupported media attributes: | |
* buffered | |
* networkState | |
* preload | |
* seekable | |
* seeking | |
* startOffsetTime | |
* | |
* canPlayType() function | |
*/ | |
// Trackers | |
var timeupdateInterval = 33, | |
timeCheckInterval = 0.25, | |
abs = Math.abs, | |
floor = Math.floor, | |
round = Math.round, | |
registry = {}; | |
function hasAllDependencies() { | |
return global.swfobject && global.soundcloud; | |
} | |
// Borrowed from: http://www.quirksmode.org/dom/getstyles.html | |
// Gets the style for the given element | |
function getStyle( elem, styleProp ) { | |
if ( elem.currentStyle ) { | |
// IE way | |
return elem.currentStyle[styleProp]; | |
} else if ( global.getComputedStyle ) { | |
// Firefox, Chrome, et. al | |
return document.defaultView.getComputedStyle( elem, null ).getPropertyValue( styleProp ); | |
} | |
} | |
function formatComment( comment ) { | |
// Calclate the difference between d and now, express as "n units ago" | |
function ago( d ) { | |
var diff = ( ( new Date() ).getTime() - d.getTime() )/1000; | |
function pluralize( value, unit ) { | |
return value + " " + unit + ( value > 1 ? "s" : "") + " ago"; | |
} | |
if ( diff < 60 ) { | |
return pluralize( round( diff ), "second" ); | |
} | |
diff /= 60; | |
if ( diff < 60 ) { | |
return pluralize( round( diff ), "minute" ); | |
} | |
diff /= 60; | |
if ( diff < 24 ) { | |
return pluralize( round( diff ), "hour" ); | |
} | |
diff /= 24; | |
// Rough approximation of months | |
if ( diff < 30 ) { | |
return pluralize( round( diff ), "day" ); | |
} | |
if ( diff < 365 ) { | |
return pluralize( round( diff/30 ), "month" ); | |
} | |
return pluralize( round( diff/365 ), "year" ); | |
} | |
// Converts sec to min.sec | |
function timeToFraction ( totalSec ) { | |
var min = floor( totalSec / 60 ), | |
sec = round( totalSec % 60 ); | |
return min + "." + ( sec < 10 ? "0" : "" ) + sec; | |
} | |
return '<div><a href="' + comment.user.profile + '">' + | |
'<img width="16px height="16px" src="' + comment.user.avatar + '"></img>' + | |
comment.user.name + '</a> at ' + timeToFraction( comment.start ) + ' ' + | |
ago( comment.date ) + | |
'<br />' + comment.text + '</span>'; | |
} | |
function isReady( self ) { | |
if ( !hasAllDependencies() ) { | |
setTimeout( function() { | |
isReady( self ); | |
}, 15 ); | |
return; | |
} | |
var flashvars = { | |
enable_api: true, | |
object_id: self._playerId, | |
url: self.src, | |
// Hide comments in player if showing them elsewhere | |
show_comments: !self._options.api.key && !self._options.api.commentdiv | |
}, | |
params = { | |
allowscriptaccess: "always", | |
// This is so we can overlay html ontop of Flash | |
wmode: 'transparent' | |
}, | |
attributes = { | |
id: self._playerId, | |
name: self._playerId | |
}, | |
actualTarget = document.createElement( 'div' ); | |
actualTarget.setAttribute( "id", self._playerId ); | |
self._container.appendChild( actualTarget ); | |
swfobject.embedSWF( "http://player.soundcloud.com/player.swf", self._playerId, self.offsetWidth, self.height, "9.0.0", "expressInstall.swf", flashvars, params, attributes ); | |
} | |
Popcorn.getScript( "http://ajax.googleapis.com/ajax/libs/swfobject/2.2/swfobject.js" ); | |
// Source file originally from 'https://github.com/soundcloud/Widget-JS-API/raw/master/soundcloud.player.api.js' | |
Popcorn.getScript( "http://popcornjs.org/code/players/soundcloud/lib/soundcloud.player.api.js", function() { | |
// Play event is fired twice when player is first started. Ignore second one | |
var ignorePlayEvt = 1; | |
// Register the wrapper's load event with the player | |
soundcloud.addEventListener( 'onPlayerReady', function( object, data ) { | |
var wrapper = registry[object.api_getFlashId()]; | |
wrapper.swfObj = object; | |
wrapper.duration = object.api_getTrackDuration(); | |
wrapper.currentTime = object.api_getTrackPosition(); | |
// This eliminates volumechangee event from firing on load | |
wrapper.volume = wrapper.previousVolume = object.api_getVolume()/100; | |
// The numeric id of the track for use with Soundcloud API | |
wrapper._mediaId = data.mediaId; | |
wrapper.dispatchEvent( 'load' ); | |
wrapper.dispatchEvent( 'canplay' ); | |
wrapper.dispatchEvent( 'durationchange' ); | |
wrapper.timeupdate(); | |
}); | |
// Register events for when the flash player plays a track for the first time | |
soundcloud.addEventListener( 'onMediaStart', function( object, data ) { | |
var wrapper = registry[object.api_getFlashId()]; | |
wrapper.played = 1; | |
wrapper.dispatchEvent( 'playing' ); | |
}); | |
// Register events for when the flash player plays a track | |
soundcloud.addEventListener( 'onMediaPlay', function( object, data ) { | |
if ( ignorePlayEvt ) { | |
ignorePlayEvt = 0; | |
return; | |
} | |
var wrapper = registry[object.api_getFlashId()]; | |
wrapper.dispatchEvent( 'play' ); | |
}); | |
// Register events for when the flash player pauses a track | |
soundcloud.addEventListener( 'onMediaPause', function( object, data ) { | |
var wrapper = registry[object.api_getFlashId()]; | |
wrapper.dispatchEvent( 'pause' ); | |
}); | |
// Register events for when the flash player is buffering | |
soundcloud.addEventListener( 'onMediaBuffering', function( object, data ) { | |
var wrapper = registry[object.api_getFlashId()]; | |
wrapper.dispatchEvent( 'progress' ); | |
if ( wrapper.readyState === 0 ) { | |
wrapper.readyState = 3; | |
wrapper.dispatchEvent( "readystatechange" ); | |
} | |
}); | |
// Register events for when the flash player is done buffering | |
soundcloud.addEventListener( 'onMediaDoneBuffering', function( object, data ) { | |
var wrapper = registry[object.api_getFlashId()]; | |
wrapper.dispatchEvent( 'canplaythrough' ); | |
}); | |
// Register events for when the flash player has finished playing | |
soundcloud.addEventListener( 'onMediaEnd', function( object, data ) { | |
var wrapper = registry[object.api_getFlashId()]; | |
wrapper.paused = 1; | |
//wrapper.pause(); | |
wrapper.dispatchEvent( 'ended' ); | |
}); | |
// Register events for when the flash player has seeked | |
soundcloud.addEventListener( 'onMediaSeek', function( object, data ) { | |
var wrapper = registry[object.api_getFlashId()]; | |
wrapper.setCurrentTime( object.api_getTrackPosition() ); | |
if ( wrapper.paused ) { | |
wrapper.dispatchEvent( "timeupdate" ); | |
} | |
}); | |
// Register events for when the flash player has errored | |
soundcloud.addEventListener( 'onPlayerError', function( object, data ) { | |
var wrapper = registry[object.api_getFlashId()]; | |
wrapper.dispatchEvent( 'error' ); | |
}); | |
}); | |
Popcorn.soundcloud = function( containerId, src, options ) { | |
return new Popcorn.soundcloud.init( containerId, src, options ); | |
}; | |
// A constructor, but we need to wrap it to allow for "static" functions | |
Popcorn.soundcloud.init = (function() { | |
function pullFromContainer( that ) { | |
var options = that._options, | |
container = that._container, | |
bounds = container.getBoundingClientRect(), | |
tmp, | |
undef; | |
that.width = options.width || getStyle( container, "width" ) || "100%"; | |
that.height = options.height || getStyle( container, "height" ) || "81px"; | |
that.src = options.src; | |
that.autoplay = options.autoplay; | |
if ( parseFloat( that.height, 10 ) !== 81 ) { | |
that.height = "81px"; | |
} | |
that.offsetLeft = bounds.left; | |
that.offsetTop = bounds.top; | |
that.offsetHeight = parseFloat( that.height, 10 ); | |
that.offsetWidth = parseFloat( that.width, 10 ); | |
// Width and height may've been specified as a %, find the value now in case a plugin needs it (like subtitle) | |
if ( /[\d]+%/.test( that.width ) ) { | |
tmp = getStyle( container, "width" ); | |
that._container.style.width = that.width; | |
that.offsetWidth = that._container.offsetWidth; | |
that._container.style.width = tmp; | |
} | |
if ( /[\d]+%/.test( that.height ) ) { | |
tmp = getStyle( container, "height" ); | |
that._container.style.height = that.height; | |
that.offsetHeight = that._container.offsetHeight; | |
that._container.style.height = tmp; | |
} | |
} | |
// If container id is not supplied, assumed to be same as player id | |
var ctor = function ( containerId, src, options ) { | |
if ( !containerId ) { | |
throw "Must supply an id!"; | |
} else if ( !src ) { | |
throw "Must supply a source!"; | |
} else if ( /file/.test( location.protocol ) ) { | |
throw "Must run from a web server!"; | |
} | |
var container = this._container = document.getElementById( containerId ); | |
if ( !container ) { | |
throw "Could not find that container in the DOM!"; | |
} | |
options = options || {}; | |
options.api = options.api || {}; | |
options.target = containerId; | |
options.src = src; | |
options.api.commentformat = options.api.commentformat || formatComment; | |
this._mediaId = 0; | |
this._listeners = {}; | |
this._playerId = Popcorn.guid( options.target ); | |
this._containerId = options.target; | |
this._options = options; | |
this._comments = []; | |
this._popcorn; | |
pullFromContainer( this ); | |
this.duration = 0; | |
this.volume = 1; | |
this.currentTime = 0; | |
this.ended = 0; | |
this.paused = 1; | |
this.readyState = 0; | |
this.playbackRate = 1; | |
this.top = 0; | |
this.left = 0; | |
this.autoplay; | |
this.played = 0; | |
this.addEventListener( "load", function() { | |
var boundRect = this.getBoundingClientRect(); | |
this.top = boundRect.top; | |
this.left = boundRect.left; | |
this.offsetWidth = this.swfObj.offsetWidth; | |
this.offsetHeight = this.swfObj.offsetHeight; | |
this.offsetLeft = this.swfObj.offsetLeft; | |
this.offsetTop = this.swfObj.offsetTop; | |
}); | |
registry[ this._playerId ] = this; | |
isReady( this ); | |
}; | |
return ctor; | |
})(); | |
Popcorn.soundcloud.init.prototype = Popcorn.soundcloud.prototype; | |
// Sequence object prototype | |
Popcorn.extend( Popcorn.soundcloud.prototype, { | |
// Set the volume as a value between 0 and 1 | |
setVolume: function( val ) { | |
if ( !val && val !== 0 ) { | |
return; | |
} | |
// Normalize in case outside range of expected values of 0 .. 1 | |
if ( val < 0 ) { | |
val = -val; | |
} | |
if ( val > 1 ) { | |
val %= 1; | |
} | |
// HTML video expects to be 0.0 -> 1.0, Flash object expects 0-100 | |
this.volume = this.previousVolume = val; | |
this.swfObj.api_setVolume( val*100 ); | |
this.dispatchEvent( "volumechange" ); | |
}, | |
// Seeks the video | |
setCurrentTime: function ( time ) { | |
if ( !time && time !== 0 ) { | |
return; | |
} | |
this.currentTime = this.previousCurrentTime = time; | |
this.ended = time >= this.duration; | |
// Fire events for seeking and time change | |
this.dispatchEvent( "seeked" ); | |
}, | |
// Play the video | |
play: function() { | |
// In case someone is cheeky enough to try this before loaded | |
if ( !this.swfObj ) { | |
this.addEventListener( "load", this.play ); | |
return; | |
} else if ( !this.paused ) { | |
// No need to process if already playing | |
return; | |
} | |
this.paused = 0; | |
this.swfObj.api_play(); | |
}, | |
// Pause the video | |
pause: function() { | |
// In case someone is cheeky enough to try this before loaded | |
if ( !this.swfObj ) { | |
this.addEventListener( "load", this.pause ); | |
return; | |
} else if ( this.paused ) { | |
// No need to process if already playing | |
return; | |
} | |
this.paused = 1; | |
this.swfObj.api_pause(); | |
}, | |
// Toggle video muting | |
// Unmuting will leave it at the old value | |
mute: function() { | |
// In case someone is cheeky enough to try this before loaded | |
if ( !this.swfObj ) { | |
this.addEventListener( "load", this.mute ); | |
return; | |
} | |
if ( !this.muted() ) { | |
this.oldVol = this.volume; | |
if ( this.paused ) { | |
this.setVolume( 0 ); | |
} else { | |
this.volume = 0; | |
} | |
} else { | |
if ( this.paused ) { | |
this.setVolume( this.oldVol ); | |
} else { | |
this.volume = this.oldVol; | |
} | |
} | |
}, | |
muted: function() { | |
return this.volume === 0; | |
}, | |
// Force loading by playing the player. Pause afterwards | |
load: function() { | |
// In case someone is cheeky enough to try this before loaded | |
if ( !this.swfObj ) { | |
this.addEventListener( "load", this.load ); | |
return; | |
} | |
this.play(); | |
this.pause(); | |
}, | |
// Hook an event listener for the player event into internal event system | |
// Stick to HTML conventions of add event listener and keep lowercase, without prepending "on" | |
addEventListener: function( evt, fn ) { | |
if ( !this._listeners[evt] ) { | |
this._listeners[evt] = []; | |
} | |
this._listeners[evt].push( fn ); | |
return fn; | |
}, | |
dispatchEvent: function( evt ) { | |
var self = this, | |
evtName = evt.type || evt; | |
// Manually triggered a UI event, have it invoke rather than just the event handlers | |
if ( evtName === "play" && this.paused || evtName === "pause" && !this.paused ) { | |
this[evtName](); | |
return; | |
} | |
Popcorn.forEach( this._listeners[evtName], function( fn ) { | |
fn.call( self ); | |
}); | |
}, | |
timeupdate: function() { | |
var self = this, | |
checkedVolume = this.swfObj.api_getVolume()/100, | |
seeked = 0; | |
// If has been changed through setting currentTime attribute | |
if ( abs( this.currentTime - this.previousCurrentTime ) > timeCheckInterval ) { | |
// Has programatically set the currentTime | |
this.swfObj.api_seekTo( this.currentTime ); | |
seeked = 1; | |
} else { | |
this.previousCurrentTime = this.currentTime = this.swfObj.api_getTrackPosition(); | |
} | |
// If has been changed throughh volume attribute | |
if ( checkedVolume !== this.previousVolume ) { | |
this.setVolume( checkedVolume ); | |
} else if ( this.volume !== this.previousVolume ) { | |
this.setVolume( this.volume ); | |
} | |
if ( !this.paused ) { | |
this.dispatchEvent( 'timeupdate' ); | |
} | |
if( !self.ended ) { | |
setTimeout( function() { | |
self.timeupdate.call( self ); | |
}, timeupdateInterval); | |
} | |
}, | |
getBoundingClientRect: function() { | |
var b, | |
self = this; | |
if ( this.swfObj ) { | |
b = this.swfObj.getBoundingClientRect(); | |
return { | |
bottom: b.bottom, | |
left: b.left, | |
right: b.right, | |
top: b.top, | |
// These not guaranteed to be in there | |
width: b.width || ( b.right - b.left ), | |
height: b.height || ( b.bottom - b.top ) | |
}; | |
} else { | |
//container = document.getElementById( this.playerId ); | |
tmp = this._container.getBoundingClientRect(); | |
// Update bottom, right for expected values once the container loads | |
return { | |
left: tmp.left, | |
top: tmp.top, | |
width: self.offsetWidth, | |
height: self.offsetHeight, | |
bottom: tmp.top + this.width, | |
right: tmp.top + this.height | |
}; | |
} | |
}, | |
registerPopcornWithPlayer: function( popcorn ) { | |
if ( !this.swfObj ) { | |
this.addEventListener( "load", function() { | |
this.registerPopcornWithPlayer( popcorn ); | |
}); | |
return; | |
} | |
this._popcorn = popcorn; | |
var api = this._options.api; | |
if ( api.key && api.commentdiv ) { | |
var self = this; | |
Popcorn.xhr({ | |
url: "http://api.soundcloud.com/tracks/" + self._mediaId + "/comments.js?consumer_key=" + api.key, | |
success: function( data ) { | |
Popcorn.forEach( data.json, function ( obj ) { | |
self.addComment({ | |
start: obj.timestamp/1000, | |
date: new Date( obj.created_at ), | |
text: obj.body, | |
user: { | |
name: obj.user.username, | |
profile: obj.user.permalink_url, | |
avatar: obj.user.avatar_url | |
} | |
}); | |
}); | |
} | |
}); | |
} | |
} | |
}); | |
Popcorn.extend( Popcorn.soundcloud.prototype, { | |
addComment: function( obj, displayFn ) { | |
var self = this, | |
comment = { | |
start: obj.start || 0, | |
date: obj.date || new Date(), | |
text: obj.text || "", | |
user: { | |
name: obj.user.name || "", | |
profile: obj.user.profile || "", | |
avatar: obj.user.avatar || "" | |
}, | |
display: function() { | |
return ( displayFn || self._options.api.commentformat )( comment ); | |
} | |
}; | |
this._comments.push( comment ); | |
if ( !this._popcorn ) { | |
return; | |
} | |
this._popcorn.subtitle({ | |
start: comment.start, | |
target: this._options.api.commentdiv, | |
display: 'inline', | |
language: 'en', | |
text: comment.display() | |
}); | |
} | |
}); | |
})( Popcorn, window );// EFFECT: applyclass | |
(function (Popcorn) { | |
/** | |
* apply css class to jquery selector | |
* selector is relative to plugin target's id | |
* so .overlay is actually jQuery( "#target .overlay") | |
* | |
* @param {Object} options | |
* | |
* Example: | |
var p = Popcorn('#video') | |
.footnote({ | |
start: 5, // seconds | |
end: 15, // seconds | |
text: 'This video made exclusively for drumbeat.org', | |
target: 'footnotediv', | |
effect: 'applyclass', | |
applyclass: 'selector: class' | |
}) | |
* | |
*/ | |
var toggleClass = function( event, options ) { | |
var idx = 0, len = 0, elements; | |
Popcorn.forEach( options.classes, function( key, val ) { | |
elements = []; | |
if ( key === "parent" ) { | |
elements[ 0 ] = document.querySelectorAll("#" + options.target )[ 0 ].parentNode; | |
} else { | |
elements = document.querySelectorAll("#" + options.target + " " + key ); | |
} | |
for ( idx = 0, len = elements.length; idx < len; idx++ ) { | |
elements[ idx ].classList.toggle( val ); | |
} | |
}); | |
}; | |
Popcorn.compose( "applyclass", { | |
manifest: { | |
about: { | |
name: "Popcorn applyclass Effect", | |
version: "0.1", | |
author: "@scottdowne", | |
website: "scottdowne.wordpress.com" | |
}, | |
options: {} | |
}, | |
_setup: function( options ) { | |
options.classes = {}; | |
options.applyclass = options.applyclass || ""; | |
var classes = options.applyclass.replace( /\s/g, "" ).split( "," ), | |
item = [], | |
idx = 0, len = classes.length; | |
for ( ; idx < len; idx++ ) { | |
item = classes[ idx ].split( ":" ); | |
if ( item[ 0 ] ) { | |
options.classes[ item[ 0 ] ] = item[ 1 ] || ""; | |
} | |
} | |
}, | |
start: toggleClass, | |
end: toggleClass | |
}); | |
})( Popcorn ); |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment