aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--assets/js/docs.js9
-rw-r--r--assets/js/simple-media.js1069
-rw-r--r--assets/less/docs.less23
-rw-r--r--dist/css/docs.css2
-rw-r--r--dist/js/docs.js2
-rw-r--r--dist/js/simple-media.js2
-rw-r--r--docs/audio.html41
-rw-r--r--docs/index.html67
-rw-r--r--docs/video.html47
9 files changed, 639 insertions, 623 deletions
diff --git a/assets/js/docs.js b/assets/js/docs.js
index 010fe06f..3527a8b2 100644
--- a/assets/js/docs.js
+++ b/assets/js/docs.js
@@ -4,14 +4,9 @@
/*global simpleMedia, templates */
-// Register a callback
-simpleMedia.on("setup", function() {
- //console.log(this);
-});
-
-//execute shout
+// Setup the player
simpleMedia.setup({
- //debug: true,
+ debug: true,
title: "Video demo",
html: templates.controls.render({})
}); \ No newline at end of file
diff --git a/assets/js/simple-media.js b/assets/js/simple-media.js
index a8b36a0a..3df0fbb4 100644
--- a/assets/js/simple-media.js
+++ b/assets/js/simple-media.js
@@ -8,15 +8,12 @@
/*global ActiveXObject*/
-(function(api){
+(function (api) {
"use strict";
// Globals
var fullscreen, config;
- // Object cache
- var player = {};
-
// Handler cache
var handlers = {};
@@ -69,7 +66,7 @@
// Credits: http://paypal.github.io/accessible-html5-video-player/
// Unfortunately, due to scattered support, browser sniffing is required
- function browserSniff() {
+ function _browserSniff() {
var nAgt = navigator.userAgent,
browserName = navigator.appName,
fullVersion = ""+parseFloat(navigator.appVersion),
@@ -130,60 +127,14 @@
// Return data
return [browserName, majorVersion];
}
- // Utilities for caption time codes
- function video_timecode_min(tc) {
- var tcpair = [];
- tcpair = tc.split(" --> ");
- return videosub_tcsecs(tcpair[0]);
- }
- function video_timecode_max(tc) {
- var tcpair = [];
- tcpair = tc.split(" --> ");
- return videosub_tcsecs(tcpair[1]);
- }
- function videosub_tcsecs(tc) {
- if (tc === null || tc === undefined) {
- return 0;
- }
- else {
- var tc1 = [],
- tc2 = [],
- seconds;
- tc1 = tc.split(",");
- tc2 = tc1[0].split(":");
- seconds = Math.floor(tc2[0]*60*60) + Math.floor(tc2[1]*60) + Math.floor(tc2[2]);
- return seconds;
- }
- }
- // For "manual" captions, adjust caption position when play time changed (via rewind, clicking progress bar, etc.)
- // http://paypal.github.io/accessible-html5-video-player/
- function adjustManualCaptions(player) {
- player.subcount = 0;
- while (video_timecode_max(player.captions[player.subcount][0]) < player.media.currentTime.toFixed(1)) {
- player.subcount++;
- if (player.subcount > player.captions.length-1) {
- player.subcount = player.captions.length-1;
- break;
- }
- }
- }
- // Display captions container and button (for initialization)
- function showCaptions(player) {
- player.container.className += " " + config.classes.captions.enabled;
-
- if (config.captions.defaultActive) {
- player.container.className += " " + config.classes.captions.active;
- player.buttons.captions.setAttribute("checked", "checked");
- }
- }
-
+
// Replace all
- function replaceAll(string, find, replace) {
+ function _replaceAll(string, find, replace) {
return string.replace(new RegExp(find.replace(/([.*+?^=!:${}()|\[\]\/\\])/g, "\\$1"), "g"), replace);
}
// Wrap an element
- function wrap(elements, wrapper) {
+ function _wrap(elements, wrapper) {
// Convert `elms` to an array, if necessary.
if (!elements.length) {
elements = [elements];
@@ -214,6 +165,24 @@
}
}
+ // Toggle class on an element
+ function _toggleClass(element, name, state) {
+ if(element){
+ if(element.classList) {
+ element.classList[state ? "add" : "remove"](name);
+ }
+ else {
+ var className = (" " + element.className + " ").replace(/\s+/g, " ").replace(" " + name + " ", "");
+ element.className = className + (state ? " " + name : "")
+ }
+ }
+ }
+
+ // Bind event
+ function _on(element, event, callback) {
+ element.addEventListener(event, callback, false);
+ }
+
// Get click position relative to parent
// http://www.kirupa.com/html5/getting_mouse_click_position.htm
function _getClickPosition(event) {
@@ -321,407 +290,389 @@
return fullscreen;
}
- // Our internal function that executes handlers with a given name
- function executeHandlers(eventName){
- // Get all handlers with the selected name
- var handler = handlers[eventName] || [],
- len = handler.length,
- i;
-
- // Execute each
- for(i = 0; i< len; i++){
- // You can use apply to specify what "this" and parameters the callback gets
- handler[i].apply(player.media,[]);
- }
- }
+ // Player instance
+ function Player(container) {
+ var player = this;
+ player.container = container;
+ // Captions functions
+ // Credits: http://paypal.github.io/accessible-html5-video-player/
- // Insert controls
- function injectControls() {
- // Insert custom video controls
- if (config.debug) {
- console.log("Injecting custom controls");
+ // For "manual" captions, adjust caption position when play time changed (via rewind, clicking progress bar, etc.)
+ function _adjustManualCaptions() {
+ player.subcount = 0;
+ while (_timecodeMax(player.captions[player.subcount][0]) < player.media.currentTime.toFixed(1)) {
+ player.subcount++;
+ if (player.subcount > player.captions.length-1) {
+ player.subcount = player.captions.length-1;
+ break;
+ }
+ }
}
+ // Display captions container and button (for initialization)
+ function _showCaptions() {
+ _toggleClass(player.container, config.classes.captions.enabled, true);
- // Use specified html
- // Need to do a default?
- var html = config.html;
-
- // Replace aria label instances
- html = replaceAll(html, "{aria-label}", config.playAriaLabel);
-
- // Replace all id references
- html = replaceAll(html, "{id}", player.random);
-
- // Inject into the container
- player.container.insertAdjacentHTML("beforeend", html);
- }
+ if (config.captions.defaultActive) {
+ _toggleClass(player.container, config.classes.captions.active, true);
+ player.buttons.captions.setAttribute("checked", "checked");
+ }
+ }
+ // Utilities for caption time codes
+ function _timecodeMin(tc) {
+ var tcpair = [];
+ tcpair = tc.split(" --> ");
+ return _subTcSecs(tcpair[0]);
+ }
+ function _timecodeMax(tc) {
+ var tcpair = [];
+ tcpair = tc.split(" --> ");
+ return _subTcSecs(tcpair[1]);
+ }
+ function _subTcSecs(tc) {
+ if (tc === null || tc === undefined) {
+ return 0;
+ }
+ else {
+ var tc1 = [],
+ tc2 = [],
+ seconds;
+ tc1 = tc.split(",");
+ tc2 = tc1[0].split(":");
+ seconds = Math.floor(tc2[0]*60*60) + Math.floor(tc2[1]*60) + Math.floor(tc2[2]);
+ return seconds;
+ }
+ }
- // Find all elements
- function getElements(selector) {
- return player.container.querySelectorAll(selector);
- }
+ // Find all elements
+ function _getElements(selector) {
+ return player.container.querySelectorAll(selector);
+ }
- // Find a single element
- function getElement(selector) {
- return getElements(selector)[0];
- }
+ // Find a single element
+ function _getElement(selector) {
+ return _getElements(selector)[0];
+ }
- // Find the UI controls and store references
- function findElements() {
- player.controls = getElement(config.selectors.controls);
-
- // Buttons
- player.buttons = {};
- player.buttons.play = getElement(config.selectors.buttons.play);
- player.buttons.pause = getElement(config.selectors.buttons.pause);
- player.buttons.restart = getElement(config.selectors.buttons.restart);
- player.buttons.rewind = getElement(config.selectors.buttons.rewind);
- player.buttons.forward = getElement(config.selectors.buttons.forward);
- player.buttons.mute = getElement(config.selectors.buttons.mute);
- player.buttons.captions = getElement(config.selectors.buttons.captions);
- player.buttons.fullscreen = getElement(config.selectors.buttons.fullscreen);
-
- // Progress
- player.progress = {};
- player.progress.bar = getElement(config.selectors.progress);
- player.progress.text = player.progress.bar.getElementsByTagName("span")[0];
-
- // Volume
- player.volume = getElement(config.selectors.buttons.volume);
-
- // Timing
- player.duration = getElement(config.selectors.duration);
- player.seekTime = getElements(config.selectors.seekTime);
- }
+ // Insert controls
+ function _injectControls() {
+ // Insert custom video controls
+ if (config.debug) {
+ console.log("Injecting custom controls");
+ }
- // Play media
- function play() {
- player.media.play();
- player.container.className = player.container.className.replace(config.classes.stopped, config.classes.playing);
- }
+ // Use specified html
+ // Need to do a default?
+ var html = config.html;
- // Pause media
- function pause() {
- player.media.pause();
- player.container.className = player.container.className.replace(config.classes.playing, config.classes.stopped);
- }
+ // Replace aria label instances
+ html = _replaceAll(html, "{aria-label}", config.playAriaLabel);
- // Restart playback
- function restart() {
- // Move to beginning
- player.media.currentTime = 0;
+ // Replace all id references
+ html = _replaceAll(html, "{id}", player.random);
- // Special handling for "manual" captions
- if (!player.isTextTracks) {
- player.subcount = 0;
+ // Inject into the container
+ player.container.insertAdjacentHTML("beforeend", html);
}
- // Play and ensure the play button is in correct state
- play();
- }
+ // Find the UI controls and store references
+ function _findElements() {
+ player.controls = _getElement(config.selectors.controls);
- // Set volume
- function setVolume() {
- player.volume.value = config.volume;
- player.media.volume = parseFloat(config.volume / 10);
- checkMute();
- }
+ // Buttons
+ player.buttons = {};
+ player.buttons.play = _getElement(config.selectors.buttons.play);
+ player.buttons.pause = _getElement(config.selectors.buttons.pause);
+ player.buttons.restart = _getElement(config.selectors.buttons.restart);
+ player.buttons.rewind = _getElement(config.selectors.buttons.rewind);
+ player.buttons.forward = _getElement(config.selectors.buttons.forward);
+ player.buttons.mute = _getElement(config.selectors.buttons.mute);
+ player.buttons.captions = _getElement(config.selectors.buttons.captions);
+ player.buttons.fullscreen = _getElement(config.selectors.buttons.fullscreen);
- // Check mute state
- function checkMute() {
- if(player.media.volume === 0 || player.media.muted) {
- player.container.className += " " + config.classes.muted;
- }
- else {
- player.container.className = player.container.className.replace(config.classes.muted, "");
- }
- }
+ // Progress
+ player.progress = {};
+ player.progress.bar = _getElement(config.selectors.progress);
+ player.progress.text = player.progress.bar.getElementsByTagName("span")[0];
- // Setup media
- function setupMedia() {
- player.media = player.container.querySelectorAll("audio, video")[0];
+ // Volume
+ player.volume = _getElement(config.selectors.buttons.volume);
- // If there's no media, bail
- if(!player.media) {
- console.error("No audio or video element found!");
- return false;
+ // Timing
+ player.duration = _getElement(config.selectors.duration);
+ player.seekTime = _getElements(config.selectors.seekTime);
}
- // If there's no autoplay attribute, assume the video is stopped
- if(player.media.getAttribute("autoplay") === null) {
- player.container.className += " " + config.classes.stopped;
- }
-
- // Remove native video controls
- player.media.removeAttribute("controls");
+ // Setup media
+ function _setupMedia() {
+ player.media = player.container.querySelectorAll("audio, video")[0];
- // Set type
- player.type = (player.media.tagName.toLowerCase() == "video" ? "video" : "audio");
+ // If there's no media, bail
+ if(!player.media) {
+ console.error("No audio or video element found!");
+ return false;
+ }
- // Inject the player wrapper
- if(player.type === "video") {
- // Create the wrapper div
- var wrapper = document.createElement("div");
- wrapper.setAttribute("class", config.classes.videoContainer);
+ // If there's no autoplay attribute, assume the video is stopped
+ _toggleClass(player.container, config.classes.stopped, (player.media.getAttribute("autoplay") === null));
- // Wrap the video in a container
- wrap(player.media, wrapper);
+ // Remove native video controls
+ player.media.removeAttribute("controls");
- // Cache the container
- player.videoContainer = wrapper;
- }
- }
+ // Set type
+ player.type = (player.media.tagName.toLowerCase() == "video" ? "video" : "audio");
- // Setup captions
- function setupCaptions() {
- if(player.type === "video") {
- // Inject the container
- player.videoContainer.insertAdjacentHTML("afterbegin", "<div class='" + config.selectors.captions.replace(".", "") + "'></div>");
+ // Inject the player wrapper
+ if(player.type === "video") {
+ // Create the wrapper div
+ var wrapper = document.createElement("div");
+ wrapper.setAttribute("class", config.classes.videoContainer);
- // Cache selector
- player.captionsContainer = getElement(config.selectors.captions);
+ // Wrap the video in a container
+ _wrap(player.media, wrapper);
- // Determine if HTML5 textTracks is supported
- player.isTextTracks = false;
- if (player.media.textTracks) {
- player.isTextTracks = true;
+ // Cache the container
+ player.videoContainer = wrapper;
}
+ }
- // Get URL of caption file if exists
- var captionSrc = "",
- kind,
- children = player.media.childNodes;
+ // Setup captions
+ function _setupCaptions() {
+ if(player.type === "video") {
+ // Inject the container
+ player.videoContainer.insertAdjacentHTML("afterbegin", "<div class='" + config.selectors.captions.replace(".", "") + "'></div>");
- for (var i = 0; i < children.length; i++) {
- if (children[i].nodeName.toLowerCase() === "track") {
- kind = children[i].getAttribute("kind");
- if (kind === "captions") {
- captionSrc = children[i].getAttribute("src");
- }
- }
- }
+ // Cache selector
+ player.captionsContainer = _getElement(config.selectors.captions);
- // Record if caption file exists or not
- player.captionExists = true;
- if (captionSrc === "") {
- player.captionExists = false;
- if (config.debug) {
- console.log("No caption track found.");
- }
- }
- else {
- if (config.debug) {
- console.log("Caption track found; URI: " + captionSrc);
+ // Determine if HTML5 textTracks is supported
+ player.isTextTracks = false;
+ if (player.media.textTracks) {
+ player.isTextTracks = true;
}
- }
- // If no caption file exists, hide container for caption text
- if (!player.captionExists) {
- player.container.className = player.container.className.replace(config.classes.captions.enabled, "");
- }
+ // Get URL of caption file if exists
+ var captionSrc = "",
+ kind,
+ children = player.media.childNodes;
- // If caption file exists, process captions
- else {
- var track = {}, tracks, j;
-
- // If IE 10/11 or Firefox 31+ or Safari 7+, don"t use native captioning (still doesn"t work although they claim it"s now supported)
- if ((player.browserName === "IE" && player.browserMajorVersion === 10) ||
- (player.browserName === "IE" && player.browserMajorVersion === 11) ||
- (player.browserName === "Firefox" && player.browserMajorVersion >= 31) ||
- (player.browserName === "Safari" && player.browserMajorVersion >= 7)) {
- if (config.debug) {
- console.log("Detected IE 10/11 or Firefox 31+ or Safari 7+");
- }
- // set to false so skips to "manual" captioning
- player.isTextTracks = false;
-
- // turn off native caption rendering to avoid double captions [doesn"t work in Safari 7; see patch below]
- track = {};
- tracks = player.media.textTracks;
- for (j=0; j < tracks.length; j++) {
- track = player.media.textTracks[j];
- track.mode = "hidden";
+ for (var i = 0; i < children.length; i++) {
+ if (children[i].nodeName.toLowerCase() === "track") {
+ kind = children[i].getAttribute("kind");
+ if (kind === "captions") {
+ captionSrc = children[i].getAttribute("src");
+ }
}
}
- // Rendering caption tracks - native support required - http://caniuse.com/webvtt
- if (player.isTextTracks) {
+ // Record if caption file exists or not
+ player.captionExists = true;
+ if (captionSrc === "") {
+ player.captionExists = false;
if (config.debug) {
- console.log("textTracks supported");
- }
- showCaptions(player);
-
- track = {};
- tracks = player.media.textTracks;
- for (j=0; j < tracks.length; j++) {
- track = player.media.textTracks[j];
- track.mode = "hidden";
- if (track.kind === "captions") {
- track.addEventListener("cuechange",function() {
- if (this.activeCues[0]) {
- if (this.activeCues[0].hasOwnProperty("text")) {
- player.captionsContainer.innerHTML = this.activeCues[0].text;
- }
- }
- },false);
- }
+ console.log("No caption track found.");
}
}
- // Caption tracks not natively supported
else {
if (config.debug) {
- console.log("textTracks not supported so rendering captions manually");
+ console.log("Caption track found; URI: " + captionSrc);
}
- showCaptions(player);
-
- // Render captions from array at appropriate time
- player.currentCaption = "";
- player.subcount = 0;
- player.captions = [];
-
- player.media.addEventListener("timeupdate", function() {
- // Check if the next caption is in the current time range
- if (player.media.currentTime.toFixed(1) > video_timecode_min(player.captions[player.subcount][0]) &&
- player.media.currentTime.toFixed(1) < video_timecode_max(player.captions[player.subcount][0])) {
- player.currentCaption = player.captions[player.subcount][1];
- }
- // Is there a next timecode?
- if (player.media.currentTime.toFixed(1) > video_timecode_max(player.captions[player.subcount][0]) &&
- player.subcount < (player.captions.length-1)) {
- player.subcount++;
+ }
+
+ // If no caption file exists, hide container for caption text
+ if (!player.captionExists) {
+ _toggleClass(player.container, config.classes.captions.enabled);
+ }
+ // If caption file exists, process captions
+ else {
+ var track = {}, tracks, j;
+
+ // If IE 10/11 or Firefox 31+ or Safari 7+, don"t use native captioning (still doesn"t work although they claim it"s now supported)
+ if ((player.browserName === "IE" && player.browserMajorVersion === 10) ||
+ (player.browserName === "IE" && player.browserMajorVersion === 11) ||
+ (player.browserName === "Firefox" && player.browserMajorVersion >= 31) ||
+ (player.browserName === "Safari" && player.browserMajorVersion >= 7)) {
+ if (config.debug) {
+ console.log("Detected IE 10/11 or Firefox 31+ or Safari 7+");
}
- // Render the caption
- player.captionsContainer.innerHTML = player.currentCaption;
- }, false);
-
- if (captionSrc !== "") {
- // Create XMLHttpRequest Object
- var xhr;
- if (window.XMLHttpRequest) {
- xhr = new XMLHttpRequest();
- }
- else if (window.ActiveXObject) { // IE8
- xhr = new ActiveXObject("Microsoft.XMLHTTP");
+ // set to false so skips to "manual" captioning
+ player.isTextTracks = false;
+
+ // turn off native caption rendering to avoid double captions [doesn"t work in Safari 7; see patch below]
+ track = {};
+ tracks = player.media.textTracks;
+ for (j=0; j < tracks.length; j++) {
+ track = player.media.textTracks[j];
+ track.mode = "hidden";
}
- xhr.onreadystatechange = function() {
- if (xhr.readyState === 4) {
- if (xhr.status === 200) {
- if (config.debug) {
- console.log("xhr = 200");
- }
-
- player.captions = [];
- var records = [],
- record,
- req = xhr.responseText;
- records = req.split("\n\n");
- for (var r=0; r < records.length; r++) {
- record = records[r];
- player.captions[r] = [];
- player.captions[r] = record.split("\n");
- }
- // Remove first element ("VTT")
- player.captions.shift();
+ }
- if (config.debug) {
- console.log("Successfully loaded the caption file via ajax.");
+ // Rendering caption tracks - native support required - http://caniuse.com/webvtt
+ if (player.isTextTracks) {
+ if (config.debug) {
+ console.log("textTracks supported");
+ }
+ _showCaptions(player);
+
+ track = {};
+ tracks = player.media.textTracks;
+ for (j=0; j < tracks.length; j++) {
+ track = player.media.textTracks[j];
+ track.mode = "hidden";
+ if (track.kind === "captions") {
+ _on(track, "cuechange",function() {
+ if (this.activeCues[0]) {
+ if (this.activeCues[0].hasOwnProperty("text")) {
+ player.captionsContainer.innerHTML = this.activeCues[0].text;
+ }
}
- } else {
- if (config.debug) {
- console.log("There was a problem loading the caption file via ajax.");
+ });
+ }
+ }
+ }
+ // Caption tracks not natively supported
+ else {
+ if (config.debug) {
+ console.log("textTracks not supported so rendering captions manually");
+ }
+ _showCaptions(player);
+
+ // Render captions from array at appropriate time
+ player.currentCaption = "";
+ player.subcount = 0;
+ player.captions = [];
+
+ _on(player.media, "timeupdate", function() {
+ // Check if the next caption is in the current time range
+ if (player.media.currentTime.toFixed(1) > _timecodeMin(player.captions[player.subcount][0]) &&
+ player.media.currentTime.toFixed(1) < _timecodeMax(player.captions[player.subcount][0])) {
+ player.currentCaption = player.captions[player.subcount][1];
+ }
+ // Is there a next timecode?
+ if (player.media.currentTime.toFixed(1) > _timecodeMax(player.captions[player.subcount][0]) &&
+ player.subcount < (player.captions.length-1)) {
+ player.subcount++;
+ }
+ // Render the caption
+ player.captionsContainer.innerHTML = player.currentCaption;
+ });
+
+ if (captionSrc !== "") {
+ // Create XMLHttpRequest Object
+ var xhr;
+ if (window.XMLHttpRequest) {
+ xhr = new XMLHttpRequest();
+ }
+ else if (window.ActiveXObject) { // IE8
+ xhr = new ActiveXObject("Microsoft.XMLHTTP");
+ }
+ xhr.onreadystatechange = function() {
+ if (xhr.readyState === 4) {
+ if (xhr.status === 200) {
+ if (config.debug) {
+ console.log("xhr = 200");
+ }
+
+ player.captions = [];
+ var records = [],
+ record,
+ req = xhr.responseText;
+ records = req.split("\n\n");
+ for (var r=0; r < records.length; r++) {
+ record = records[r];
+ player.captions[r] = [];
+ player.captions[r] = record.split("\n");
+ }
+ // Remove first element ("VTT")
+ player.captions.shift();
+
+ if (config.debug) {
+ console.log("Successfully loaded the caption file via ajax.");
+ }
+ }
+ else {
+ if (config.debug) {
+ console.log("There was a problem loading the caption file via ajax.");
+ }
}
}
}
+ xhr.open("get", captionSrc, true);
+ xhr.send();
}
- xhr.open("get", captionSrc, true);
- xhr.send();
}
- }
- // If Safari 7, removing track from DOM [see "turn off native caption rendering" above]
- if (player.browserName === "Safari" && player.browserMajorVersion === 7) {
- if (config.debug) {
- console.log("Safari 7 detected; removing track from DOM");
+ // If Safari 7, removing track from DOM [see "turn off native caption rendering" above]
+ if (player.browserName === "Safari" && player.browserMajorVersion === 7) {
+ if (config.debug) {
+ console.log("Safari 7 detected; removing track from DOM");
+ }
+ tracks = player.media.getElementsByTagName("track");
+ player.media.removeChild(tracks[0]);
}
- tracks = player.media.getElementsByTagName("track");
- player.media.removeChild(tracks[0]);
}
}
}
- }
- // Setup seeking
- function setupSeeking() {
- // Update number of seconds in rewind and fast forward buttons
- player.seekTime[0].innerHTML = config.seekInterval;
- player.seekTime[1].innerHTML = config.seekInterval;
- }
+ // Setup seeking
+ function _setupSeeking() {
+ // Update number of seconds in rewind and fast forward buttons
+ player.seekTime[0].innerHTML = config.seekInterval;
+ player.seekTime[1].innerHTML = config.seekInterval;
+ }
- // Setup fullscreen
- function setupFullscreen() {
- if(player.type === "video" && config.fullscreen.enabled) {
-
- if(config.debug) {
- console.log(fullscreen.supportsFullScreen ? "Fullscreen supported" : "No fullscreen supported");
- }
- if(fullscreen.supportsFullScreen) {
+ // Setup fullscreen
+ function _setupFullscreen() {
+ if(player.type === "video" && config.fullscreen.enabled) {
+
if(config.debug) {
- console.log("Fullscreen enabled");
+ console.log(fullscreen.supportsFullScreen ? "Fullscreen supported" : "No fullscreen supported");
}
+ if(fullscreen.supportsFullScreen) {
+ if(config.debug) {
+ console.log("Fullscreen enabled");
+ }
- player.container.className += " " + config.classes.fullscreen.enabled;
- }
- else if(config.debug) {
- console.warn("Fullscreen not supported");
- }
- }
- }
-
- // Listen for events
- function listeners() {
- // Fullscreen
- player.buttons.fullscreen.addEventListener("click", function() {
- if(!fullscreen.isFullScreen()) {
- fullscreen.requestFullScreen(player.container);
- }
- else {
- fullscreen.cancelFullScreen();
- }
- }, false);
-
- // Click video
- if(player.type === "video" && config.clickToPause) {
- player.videoContainer.addEventListener("click", function() {
- if(player.media.paused) {
- play();
+ _toggleClass(player.container, config.classes.fullscreen.enabled, true);
}
- else if(player.media.ended) {
- restart();
+ else if(config.debug) {
+ console.warn("Fullscreen not supported");
}
- else {
- pause();
- }
- }, false);
+ }
}
- // Play
- player.buttons.play.addEventListener("click", function() {
- play();
- player.buttons.pause.focus();
- }, false);
+ // Play media
+ function _play() {
+ player.media.play();
- // Pause
- player.buttons.pause.addEventListener("click", function() {
- pause();
- player.buttons.play.focus();
- }, false);
+ _toggleClass(player.container, config.classes.stopped);
+ _toggleClass(player.container, config.classes.playing, true);
+ }
- // Restart
- player.buttons.restart.addEventListener("click", restart, false);
+ // Pause media
+ function _pause() {
+ player.media.pause();
+
+ _toggleClass(player.container, config.classes.playing);
+ _toggleClass(player.container, config.classes.stopped, true);
+ }
+
+ // Restart playback
+ function _restart() {
+ // Move to beginning
+ player.media.currentTime = 0;
+
+ // Special handling for "manual" captions
+ if (!player.isTextTracks) {
+ player.subcount = 0;
+ }
+
+ // Play and ensure the play button is in correct state
+ _play();
+ }
// Rewind
- player.buttons.rewind.addEventListener("click", function() {
+ function _rewind() {
var targetTime = player.media.currentTime - config.seekInterval;
if (targetTime < 0) {
@@ -732,12 +683,12 @@
}
// Special handling for "manual" captions
if (!player.isTextTracks && player.type === "video") {
- adjustManualCaptions(player);
+ _adjustManualCaptions(player);
}
- }, false);
+ }
// Fast forward
- player.buttons.forward.addEventListener("click", function() {
+ function _forward() {
var targetTime = player.media.currentTime + config.seekInterval;
if (targetTime > player.media.duration) {
@@ -748,118 +699,232 @@
}
// Special handling for "manual" captions
if (!player.isTextTracks && player.type === "video") {
- adjustManualCaptions(player);
+ _adjustManualCaptions(player);
}
- }, false);
+ }
- // Get the HTML5 range input element and append audio volume adjustment on change
- player.volume.addEventListener("change", function() {
- config.volume = this.value;
- setVolume();
- }, false);
+ // Toggle fullscreen
+ function _toggleFullscreen() {
+ if(!fullscreen.isFullScreen()) {
+ fullscreen.requestFullScreen(player.container);
+ }
+ else {
+ fullscreen.cancelFullScreen();
+ }
+ }
+
+ // Set volume
+ function _setVolume(volume) {
+ player.volume.value = volume;
+ player.media.volume = parseFloat(volume / 10);
+ _checkMute();
+ }
// Mute
- player.buttons.mute.addEventListener("click", function() {
- if (player.media.muted === true) {
- player.media.muted = false;
+ function _toggleMute(muted) {
+ // If the method is called without parameter, toggle based on current value
+ if(typeof active === "undefined") {
+ muted = !player.media.muted;
+ player.buttons.mute.checked = muted;
+ }
+
+ player.media.muted = muted;
+ _checkMute();
+ }
+
+ // Toggle captions
+ function _toggleCaptions(active) {
+ // If the method is called without parameter, toggle based on current value
+ if(typeof active === "undefined") {
+ active = (player.container.className.indexOf(config.classes.captions.active) === -1);
+ player.buttons.captions.checked = active;
}
+
+ if (active) {
+ _toggleClass(player.container, config.classes.captions.active, true);
+ }
else {
- player.media.muted = true;
+ _toggleClass(player.container, config.classes.captions.active);
}
- checkMute();
- }, false);
-
- // Duration
- player.media.addEventListener("timeupdate", function() {
- player.secs = parseInt(player.media.currentTime % 60);
- player.mins = parseInt((player.media.currentTime / 60) % 60);
-
- // Ensure it"s two digits. For example, 03 rather than 3.
- player.secs = ("0" + player.secs).slice(-2);
- player.mins = ("0" + player.mins).slice(-2);
+ }
- // Render
- player.duration.innerHTML = player.mins + ":" + player.secs;
- }, false);
+ // Check mute state
+ function _checkMute() {
+ _toggleClass(player.container, config.classes.muted, (player.media.volume === 0 || player.media.muted));
+ }
- // Progress bar
- player.media.addEventListener("timeupdate", function() {
- player.percent = (100 / player.media.duration) * player.media.currentTime;
+ // Listen for events
+ function _listeners() {
+ // Fullscreen
+ _on(player.buttons.fullscreen, "click", _toggleFullscreen);
- if (player.percent > 0) {
- player.progress.bar.value = player.percent;
- player.progress.text.innerHTML = player.percent;
+ // Click video
+ if(player.type === "video" && config.clickToPause) {
+ _on(player.videoContainer, "click", function() {
+ if(player.media.paused) {
+ _play();
+ }
+ else if(player.media.ended) {
+ _restart();
+ }
+ else {
+ _pause();
+ }
+ });
}
- }, false);
- // Skip when clicking progress bar
- player.progress.bar.addEventListener("click", function(event) {
- player.pos = _getClickPosition(event).x / this.offsetWidth;
- player.media.currentTime = player.pos * player.media.duration;
+ // Play
+ _on(player.buttons.play, "click", function() {
+ _play();
+ player.buttons.pause.focus();
+ });
+
+ // Pause
+ _on(player.buttons.pause, "click", function() {
+ _pause();
+ player.buttons.play.focus();
+ });
+
+ // Restart
+ _on(player.buttons.restart, "click", _restart);
+
+ // Rewind
+ _on(player.buttons.rewind, "click", _rewind);
+
+ // Fast forward
+ _on(player.buttons.forward, "click", _forward);
+
+ // Get the HTML5 range input element and append audio volume adjustment on change
+ _on(player.volume, "change", function() {
+ _setVolume(this.value);
+ });
+
+ // Mute
+ _on(player.buttons.mute, "change", function() {
+ _toggleMute(this.checked);
+ });
- // Special handling for "manual" captions
- if (!player.isTextTracks && player.type === "video") {
- adjustManualCaptions(player);
+ // Duration
+ _on(player.media, "timeupdate", function() {
+ player.secs = parseInt(player.media.currentTime % 60);
+ player.mins = parseInt((player.media.currentTime / 60) % 60);
+
+ // Ensure it"s two digits. For example, 03 rather than 3.
+ player.secs = ("0" + player.secs).slice(-2);
+ player.mins = ("0" + player.mins).slice(-2);
+
+ // Render
+ player.duration.innerHTML = player.mins + ":" + player.secs;
+ });
+
+ // Progress bar
+ _on(player.media, "timeupdate", function() {
+ player.percent = (100 / player.media.duration) * player.media.currentTime;
+
+ if (player.percent > 0) {
+ player.progress.bar.value = player.percent;
+ player.progress.text.innerHTML = player.percent;
+ }
+ });
+
+ // Skip when clicking progress bar
+ _on(player.progress.bar, "click", function(event) {
+ player.pos = _getClickPosition(event).x / this.offsetWidth;
+ player.media.currentTime = player.pos * player.media.duration;
+
+ // Special handling for "manual" captions
+ if (!player.isTextTracks && player.type === "video") {
+ _adjustManualCaptions(player);
+ }
+ });
+
+ // Captions
+ _on(player.buttons.captions, "click", function() {
+ _toggleCaptions(this.checked);
+ });
+
+ // Clear captions at end of video
+ _on(player.media, "ended", function() {
+ if(player.type === "video") {
+ player.captionsContainer.innerHTML = "";
+ }
+ _toggleClass(player.container, config.classes.stopped, true);
+ _toggleClass(player.container, config.classes.playing);
+ });
+ }
+
+ function _init() {
+ // Setup the fullscreen api
+ fullscreen = _fullscreen();
+
+ // Sniff
+ player.browserInfo = _browserSniff();
+ player.browserName = player.browserInfo[0];
+ player.browserMajorVersion = player.browserInfo[1];
+
+ // Debug info
+ if(config.debug) {
+ console.log(player.browserName + " " + player.browserMajorVersion);
}
- });
- // Captions
- player.buttons.captions.addEventListener("click", function() {
- if (this.checked) {
- player.container.className += " " + config.classes.captions.active;
- }
- else {
- player.container.className = player.container.className.replace(config.classes.captions.active, "");
+ // If IE8, stop customization (use fallback)
+ // If IE9, stop customization (use native controls)
+ if (player.browserName === "IE" && (player.browserMajorVersion === 8 || player.browserMajorVersion === 9) ) {
+ if(config.debug) {
+ console.error("Browser not suppported.");
+ }
+ return false;
}
- }, false);
- // Clear captions at end of video
- player.media.addEventListener("ended", function() {
- if(player.type === "video") {
- player.captionsContainer.innerHTML = "";
+ // Set up aria-label for Play button with the title option
+ if (typeof(config.title) === "undefined" || !config.title.length) {
+ config.playAriaLabel = "Play";
+ }
+ else {
+ config.playAriaLabel = "Play " + config.title;
}
- player.container.className = player.container.className.replace(config.classes.playing, config.classes.stopped);
- });
- }
-
- function setupPlayer(element) {
- player.container = element;
-
- // Setup media
- setupMedia();
- // Generate random number for id/for attribute values for controls
- player.random = Math.floor(Math.random() * (10000));
+ // Setup media
+ _setupMedia();
- // Inject custom controls
- injectControls();
+ // Generate random number for id/for attribute values for controls
+ player.random = Math.floor(Math.random() * (10000));
- // Find the elements
- findElements();
+ // Inject custom controls
+ _injectControls();
- // Set volume
- setVolume();
+ // Find the elements
+ _findElements();
- // Setup fullscreen
- setupFullscreen();
+ // Set volume
+ _setVolume(config.volume);
- // Captions
- setupCaptions();
+ // Setup fullscreen
+ _setupFullscreen();
- // Seeking
- setupSeeking();
+ // Captions
+ _setupCaptions();
- // Listeners
- listeners();
- }
+ // Seeking
+ _setupSeeking();
- // Our "on" function which collects handlers
- api.on = function(eventName, handler){
- // If no handler collection exists, create one
- if(!handlers[eventName]){
- handlers[eventName] = [];
+ // Listeners
+ _listeners();
+ }
+
+ _init();
+
+ return {
+ media: player.media,
+ play: _play,
+ pause: _pause,
+ restart: _restart,
+ rewind: _rewind,
+ forward: _forward,
+ setVolume: _setVolume,
+ toggleMute: _toggleMute,
+ toggleCaptions: _toggleCaptions
}
- handlers[eventName].push(handler);
}
// Expose setup function
@@ -868,58 +933,24 @@
config = _extend(defaults, options);
// If enabled carry on
+ // You may want to disable certain UAs etc
if(!config.enabled) {
return false;
}
- // Setup the fullscreen api
- fullscreen = _fullscreen();
-
- // Sniff
- player.browserInfo = browserSniff();
- player.browserName = player.browserInfo[0];
- player.browserMajorVersion = player.browserInfo[1];
-
- // Debug info
- if(config.debug) {
- console.log(player.browserName + " " + player.browserMajorVersion);
- }
-
- // If IE8, stop customization (use fallback)
- // If IE9, stop customization (use native controls)
- if (player.browserName === "IE" && (player.browserMajorVersion === 8 || player.browserMajorVersion === 9) ) {
- if(config.debug) {
- console.error("Browser not suppported.");
- }
- return false;
- }
-
- // Set up aria-label for Play button with the title option
- if (typeof(config.title) === "undefined" || !config.title.length) {
- config.playAriaLabel = "Play";
- }
- else {
- config.playAriaLabel = "Play " + config.title;
- }
-
- // Get the container and video container
- var element = document.querySelector(config.selectors.container);
- if(element === null) {
- if(config.debug) {
- console.error("Selector " + config.selectors.container + " not found!");
- }
- return false;
- }
- setupPlayer(element);
-
- /*var elements = document.querySelectorAll(config.selectors.container);
+ // Get the players
+ var elements = document.querySelectorAll(config.selectors.container);
+ // Create a player instance for each element
for (var i = elements.length - 1; i >= 0; i--) {
- setupPlayer(elements[i]);
- }*/
+ // Get the current element
+ var element = elements[i];
+
+ // Setup a player instance
+ var player = new Player(element);
- // Trigger the setup event
- executeHandlers("setup");
+ // Add the player object to the element
+ element.player = player;
+ }
}
-
}(this.simpleMedia = this.simpleMedia || {})); \ No newline at end of file
diff --git a/assets/less/docs.less b/assets/less/docs.less
index 23814f8b..48676d0e 100644
--- a/assets/less/docs.less
+++ b/assets/less/docs.less
@@ -35,14 +35,14 @@ html {
}
body {
font-family: "Avenir", "Helvetica Neue", Helvetica, Arial, sans-serif;
- .font-size(18);
- color: #6D797F;
- line-height: 1.5;
background: #fff;
+ .font-size(16);
+ line-height: 1.5;
text-align: center;
+ color: #6D797F;
}
-h1 {
- .font-size(48);
+h1,
+h2 {
letter-spacing: -.025em;
color: #2E3C44;
margin: 0 0 10px;
@@ -50,16 +50,27 @@ h1 {
-webkit-font-smoothing: antialiased;
-moz-osx-font-smoothing: grayscale;
}
+h1 {
+ .font-size(48);
+}
+h2 {
+ .font-size(32);
+ margin: 0 0 20px;
+}
p {
margin: 0 0 20px;
}
header {
margin: 40px 0;
+
+ p {
+ .font-size(18);
+ }
}
.example-audio .player,
.example-video .player {
- margin: 0 auto;
+ margin: 0 auto 40px;
&:fullscreen {
max-width: none;
diff --git a/dist/css/docs.css b/dist/css/docs.css
index a9d1c876..67d86fa5 100644
--- a/dist/css/docs.css
+++ b/dist/css/docs.css
@@ -1 +1 @@
-/*! normalize.css v2.1.3 | MIT License | git.io/normalize */article,aside,details,figcaption,figure,footer,header,hgroup,main,nav,section,summary{display:block}audio,canvas,video{display:inline-block}audio:not([controls]){display:none;height:0}[hidden],template{display:none}html{font-family:sans-serif;-ms-text-size-adjust:100%;-webkit-text-size-adjust:100%}body{margin:0}a{background:0 0}a:focus{outline:dotted thin}a:active,a:hover{outline:0}abbr[title]{border-bottom:1px dotted}b,strong{font-weight:700}dfn{font-style:italic}hr{box-sizing:content-box;height:0}mark{background:#ff0;color:#000}code,kbd,pre,samp{font-family:monospace,serif;font-size:1em}pre{white-space:pre-wrap}q{quotes:"\201C" "\201D" "\2018" "\2019"}small{font-size:80%}sub,sup{font-size:75%;line-height:0;position:relative;vertical-align:baseline}sup{top:-.5em}sub{bottom:-.25em}img{border:0}svg:not(:root){overflow:hidden}figure{margin:0}fieldset{border:1px solid silver;margin:0 2px;padding:.35em .625em .75em}legend{border:0;padding:0}button,input,select,textarea{font-family:inherit;font-size:100%;margin:0}button,input{line-height:normal}button,select{text-transform:none}button,html input[type=button],input[type=reset],input[type=submit]{-webkit-appearance:button;cursor:pointer}button[disabled],html input[disabled]{cursor:default}input[type=checkbox],input[type=radio]{box-sizing:border-box;padding:0}input[type=search]{-webkit-appearance:textfield;box-sizing:content-box}input[type=search]::-webkit-search-cancel-button,input[type=search]::-webkit-search-decoration{-webkit-appearance:none}button::-moz-focus-inner,input::-moz-focus-inner{border:0;padding:0}textarea{overflow:auto;vertical-align:top}table{border-collapse:collapse;border-spacing:0}@font-face{font-family:Avenir;src:url(../../assets/fonts/AvenirLTStd-Medium.woff2) format("woff2"),url(../../assets/fonts/AvenirLTStd-Medium.woff) format("woff"),url(../../assets/fonts/AvenirLTStd-Medium.ttf) format("truetype");font-style:normal;font-weight:400}@font-face{font-family:Avenir;src:url(../../assets/fonts/AvenirLTStd-Heavy.woff2) format("woff2"),url(../../assets/fonts/AvenirLTStd-Heavy.woff) format("woff"),url(../../assets/fonts/AvenirLTStd-Heavy.ttf) format("truetype");font-style:normal;font-weight:600}*,::after,::before{box-sizing:border-box}html{font-size:62.5%}body{font-family:Avenir,"Helvetica Neue",Helvetica,Arial,sans-serif;font-size:18px;font-size:1.8rem;color:#6D797F;line-height:1.5;background:#fff;text-align:center}h1{font-size:48px;font-size:4.8rem;letter-spacing:-.025em;color:#2E3C44;margin:0 0 10px;line-height:1.2;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale}p{margin:0 0 20px}header{margin:40px 0}.example-audio .player,.example-video .player{margin:0 auto}.example-audio .player:-webkit-full-screen,.example-video .player:-webkit-full-screen{max-width:none}.example-audio .player:-moz-full-screen,.example-video .player:-moz-full-screen{max-width:none}.example-audio .player:-ms-fullscreen,.example-video .player:-ms-fullscreen{max-width:none}.example-audio .player:fullscreen,.example-video .player:fullscreen{max-width:none}.example-audio .player{max-width:480px}.example-video .player{max-width:1200px} \ No newline at end of file
+/*! normalize.css v2.1.3 | MIT License | git.io/normalize */article,aside,details,figcaption,figure,footer,header,hgroup,main,nav,section,summary{display:block}audio,canvas,video{display:inline-block}audio:not([controls]){display:none;height:0}[hidden],template{display:none}html{font-family:sans-serif;-ms-text-size-adjust:100%;-webkit-text-size-adjust:100%}body{margin:0}a{background:0 0}a:focus{outline:dotted thin}a:active,a:hover{outline:0}abbr[title]{border-bottom:1px dotted}b,strong{font-weight:700}dfn{font-style:italic}hr{box-sizing:content-box;height:0}mark{background:#ff0;color:#000}code,kbd,pre,samp{font-family:monospace,serif;font-size:1em}pre{white-space:pre-wrap}q{quotes:"\201C" "\201D" "\2018" "\2019"}small{font-size:80%}sub,sup{font-size:75%;line-height:0;position:relative;vertical-align:baseline}sup{top:-.5em}sub{bottom:-.25em}img{border:0}svg:not(:root){overflow:hidden}figure{margin:0}fieldset{border:1px solid silver;margin:0 2px;padding:.35em .625em .75em}legend{border:0;padding:0}button,input,select,textarea{font-family:inherit;font-size:100%;margin:0}button,input{line-height:normal}button,select{text-transform:none}button,html input[type=button],input[type=reset],input[type=submit]{-webkit-appearance:button;cursor:pointer}button[disabled],html input[disabled]{cursor:default}input[type=checkbox],input[type=radio]{box-sizing:border-box;padding:0}input[type=search]{-webkit-appearance:textfield;box-sizing:content-box}input[type=search]::-webkit-search-cancel-button,input[type=search]::-webkit-search-decoration{-webkit-appearance:none}button::-moz-focus-inner,input::-moz-focus-inner{border:0;padding:0}textarea{overflow:auto;vertical-align:top}table{border-collapse:collapse;border-spacing:0}@font-face{font-family:Avenir;src:url(../../assets/fonts/AvenirLTStd-Medium.woff2) format("woff2"),url(../../assets/fonts/AvenirLTStd-Medium.woff) format("woff"),url(../../assets/fonts/AvenirLTStd-Medium.ttf) format("truetype");font-style:normal;font-weight:400}@font-face{font-family:Avenir;src:url(../../assets/fonts/AvenirLTStd-Heavy.woff2) format("woff2"),url(../../assets/fonts/AvenirLTStd-Heavy.woff) format("woff"),url(../../assets/fonts/AvenirLTStd-Heavy.ttf) format("truetype");font-style:normal;font-weight:600}*,::after,::before{box-sizing:border-box}html{font-size:62.5%}body{font-family:Avenir,"Helvetica Neue",Helvetica,Arial,sans-serif;background:#fff;font-size:16px;font-size:1.6rem;line-height:1.5;text-align:center;color:#6D797F}h1,h2{letter-spacing:-.025em;color:#2E3C44;margin:0 0 10px;line-height:1.2;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale}h1{font-size:48px;font-size:4.8rem}h2{font-size:32px;font-size:3.2rem;margin:0 0 20px}p{margin:0 0 20px}header{margin:40px 0}header p{font-size:18px;font-size:1.8rem}.example-audio .player,.example-video .player{margin:0 auto 40px}.example-audio .player:-webkit-full-screen,.example-video .player:-webkit-full-screen{max-width:none}.example-audio .player:-moz-full-screen,.example-video .player:-moz-full-screen{max-width:none}.example-audio .player:-ms-fullscreen,.example-video .player:-ms-fullscreen{max-width:none}.example-audio .player:fullscreen,.example-video .player:fullscreen{max-width:none}.example-audio .player{max-width:480px}.example-video .player{max-width:1200px} \ No newline at end of file
diff --git a/dist/js/docs.js b/dist/js/docs.js
index af9de668..779edf9e 100644
--- a/dist/js/docs.js
+++ b/dist/js/docs.js
@@ -1 +1 @@
-var Hogan={};!function(t){function n(t,n,e){var s;return n&&"object"==typeof n&&(void 0!==n[t]?s=n[t]:e&&n.get&&"function"==typeof n.get&&(s=n.get(t))),s}function e(t,n,e,s,a,r){function i(){}function o(){}i.prototype=t,o.prototype=t.subs;var l,u=new i;u.subs=new o,u.subsText={},u.buf="",s=s||{},u.stackSubs=s,u.subsText=r;for(l in n)s[l]||(s[l]=n[l]);for(l in s)u.subs[l]=s[l];a=a||{},u.stackPartials=a;for(l in e)a[l]||(a[l]=e[l]);for(l in a)u.partials[l]=a[l];return u}function s(t){return String(null===t||void 0===t?"":t)}function a(t){return t=s(t),c.test(t)?t.replace(r,"&amp;").replace(i,"&lt;").replace(o,"&gt;").replace(l,"&#39;").replace(u,"&quot;"):t}t.Template=function(t,n,e,s){t=t||{},this.r=t.code||this.r,this.c=e,this.options=s||{},this.text=n||"",this.partials=t.partials||{},this.subs=t.subs||{},this.buf=""},t.Template.prototype={r:function(){return""},v:a,t:s,render:function(t,n,e){return this.ri([t],n||{},e)},ri:function(t,n,e){return this.r(t,n,e)},ep:function(t,n){var s=this.partials[t],a=n[s.name];if(s.instance&&s.base==a)return s.instance;if("string"==typeof a){if(!this.c)throw new Error("No compiler available.");a=this.c.compile(a,this.options)}if(!a)return null;if(this.partials[t].base=a,s.subs){n.stackText||(n.stackText={});for(key in s.subs)n.stackText[key]||(n.stackText[key]=void 0!==this.activeSub&&n.stackText[this.activeSub]?n.stackText[this.activeSub]:this.text);a=e(a,s.subs,s.partials,this.stackSubs,this.stackPartials,n.stackText)}return this.partials[t].instance=a,a},rp:function(t,n,e,s){var a=this.ep(t,e);return a?a.ri(n,e,s):""},rs:function(t,n,e){var s=t[t.length-1];if(!p(s))return e(t,n,this),void 0;for(var a=0;a<s.length;a++)t.push(s[a]),e(t,n,this),t.pop()},s:function(t,n,e,s,a,r,i){var o;return p(t)&&0===t.length?!1:("function"==typeof t&&(t=this.ms(t,n,e,s,a,r,i)),o=!!t,!s&&o&&n&&n.push("object"==typeof t?t:n[n.length-1]),o)},d:function(t,e,s,a){var r,i=t.split("."),o=this.f(i[0],e,s,a),l=this.options.modelGet,u=null;if("."===t&&p(e[e.length-2]))o=e[e.length-1];else for(var c=1;c<i.length;c++)r=n(i[c],o,l),void 0!==r?(u=o,o=r):o="";return a&&!o?!1:(a||"function"!=typeof o||(e.push(u),o=this.mv(o,e,s),e.pop()),o)},f:function(t,e,s,a){for(var r=!1,i=null,o=!1,l=this.options.modelGet,u=e.length-1;u>=0;u--)if(i=e[u],r=n(t,i,l),void 0!==r){o=!0;break}return o?(a||"function"!=typeof r||(r=this.mv(r,e,s)),r):a?!1:""},ls:function(t,n,e,a,r){var i=this.options.delimiters;return this.options.delimiters=r,this.b(this.ct(s(t.call(n,a)),n,e)),this.options.delimiters=i,!1},ct:function(t,n,e){if(this.options.disableLambda)throw new Error("Lambda features disabled.");return this.c.compile(t,this.options).render(n,e)},b:function(t){this.buf+=t},fl:function(){var t=this.buf;return this.buf="",t},ms:function(t,n,e,s,a,r,i){var o,l=n[n.length-1],u=t.call(l);return"function"==typeof u?s?!0:(o=this.activeSub&&this.subsText&&this.subsText[this.activeSub]?this.subsText[this.activeSub]:this.text,this.ls(u,l,e,o.substring(a,r),i)):u},mv:function(t,n,e){var a=n[n.length-1],r=t.call(a);return"function"==typeof r?this.ct(s(r.call(a)),a,e):r},sub:function(t,n,e,s){var a=this.subs[t];a&&(this.activeSub=t,a(n,e,this,s),this.activeSub=!1)}};var r=/&/g,i=/</g,o=/>/g,l=/\'/g,u=/\"/g,c=/[&<>\"\']/,p=Array.isArray||function(t){return"[object Array]"===Object.prototype.toString.call(t)}}("undefined"!=typeof exports?exports:Hogan),function(t){function n(t){"}"===t.n.substr(t.n.length-1)&&(t.n=t.n.substring(0,t.n.length-1))}function e(t){return t.trim?t.trim():t.replace(/^\s*|\s*$/g,"")}function s(t,n,e){if(n.charAt(e)!=t.charAt(0))return!1;for(var s=1,a=t.length;a>s;s++)if(n.charAt(e+s)!=t.charAt(s))return!1;return!0}function a(n,e,s,o){var l=[],u=null,c=null,p=null;for(c=s[s.length-1];n.length>0;){if(p=n.shift(),c&&"<"==c.tag&&!(p.tag in k))throw new Error("Illegal content in < super tag.");if(t.tags[p.tag]<=t.tags.$||r(p,o))s.push(p),p.nodes=a(n,p.tag,s,o);else{if("/"==p.tag){if(0===s.length)throw new Error("Closing tag without opener: /"+p.n);if(u=s.pop(),p.n!=u.n&&!i(p.n,u.n,o))throw new Error("Nesting error: "+u.n+" vs. "+p.n);return u.end=p.i,l}"\n"==p.tag&&(p.last=0==n.length||"\n"==n[0].tag)}l.push(p)}if(s.length>0)throw new Error("missing closing tag: "+s.pop().n);return l}function r(t,n){for(var e=0,s=n.length;s>e;e++)if(n[e].o==t.n)return t.tag="#",!0}function i(t,n,e){for(var s=0,a=e.length;a>s;s++)if(e[s].c==t&&e[s].o==n)return!0}function o(t){var n=[];for(var e in t)n.push('"'+u(e)+'": function(c,p,t,i) {'+t[e]+"}");return"{ "+n.join(",")+" }"}function l(t){var n=[];for(var e in t.partials)n.push('"'+u(e)+'":{name:"'+u(t.partials[e].name)+'", '+l(t.partials[e])+"}");return"partials: {"+n.join(",")+"}, subs: "+o(t.subs)}function u(t){return t.replace(m,"\\\\").replace(g,'\\"').replace(d,"\\n").replace(v,"\\r").replace(y,"\\u2028").replace(x,"\\u2029")}function c(t){return~t.indexOf(".")?"d":"f"}function p(t,n){var e="<"+(n.prefix||""),s=e+t.n+w++;return n.partials[s]={name:t.n,partials:{}},n.code+='t.b(t.rp("'+u(s)+'",c,p,"'+(t.indent||"")+'"));',s}function b(t,n){n.code+="t.b(t.t(t."+c(t.n)+'("'+u(t.n)+'",c,p,0)));'}function f(t){return"t.b("+t+");"}var h=/\S/,g=/\"/g,d=/\n/g,v=/\r/g,m=/\\/g,y=/\u2028/,x=/\u2029/;t.tags={"#":1,"^":2,"<":3,$:4,"/":5,"!":6,">":7,"=":8,_v:9,"{":10,"&":11,_t:12},t.scan=function(a,r){function i(){m.length>0&&(y.push({tag:"_t",text:new String(m)}),m="")}function o(){for(var n=!0,e=w;e<y.length;e++)if(n=t.tags[y[e].tag]<t.tags._v||"_t"==y[e].tag&&null===y[e].text.match(h),!n)return!1;return n}function l(t,n){if(i(),t&&o())for(var e,s=w;s<y.length;s++)y[s].text&&((e=y[s+1])&&">"==e.tag&&(e.indent=y[s].text.toString()),y.splice(s,1));else n||y.push({tag:"\n"});x=!1,w=y.length}function u(t,n){var s="="+S,a=t.indexOf(s,n),r=e(t.substring(t.indexOf("=",n)+1,a)).split(" ");return T=r[0],S=r[r.length-1],a+s.length-1}var c=a.length,p=0,b=1,f=2,g=p,d=null,v=null,m="",y=[],x=!1,k=0,w=0,T="{{",S="}}";for(r&&(r=r.split(" "),T=r[0],S=r[1]),k=0;c>k;k++)g==p?s(T,a,k)?(--k,i(),g=b):"\n"==a.charAt(k)?l(x):m+=a.charAt(k):g==b?(k+=T.length-1,v=t.tags[a.charAt(k+1)],d=v?a.charAt(k+1):"_v","="==d?(k=u(a,k),g=p):(v&&k++,g=f),x=k):s(S,a,k)?(y.push({tag:d,n:e(m),otag:T,ctag:S,i:"/"==d?x-T.length:k+S.length}),m="",k+=S.length-1,g=p,"{"==d&&("}}"==S?k++:n(y[y.length-1]))):m+=a.charAt(k);return l(x,!0),y};var k={_t:!0,"\n":!0,$:!0,"/":!0};t.stringify=function(n){return"{code: function (c,p,i) { "+t.wrapMain(n.code)+" },"+l(n)+"}"};var w=0;t.generate=function(n,e,s){w=0;var a={code:"",subs:{},partials:{}};return t.walk(n,a),s.asString?this.stringify(a,e,s):this.makeTemplate(a,e,s)},t.wrapMain=function(t){return'var t=this;t.b(i=i||"");'+t+"return t.fl();"},t.template=t.Template,t.makeTemplate=function(t,n,e){var s=this.makePartials(t);return s.code=new Function("c","p","i",this.wrapMain(t.code)),new this.template(s,n,this,e)},t.makePartials=function(t){var n,e={subs:{},partials:t.partials,name:t.name};for(n in e.partials)e.partials[n]=this.makePartials(e.partials[n]);for(n in t.subs)e.subs[n]=new Function("c","p","t","i",t.subs[n]);return e},t.codegen={"#":function(n,e){e.code+="if(t.s(t."+c(n.n)+'("'+u(n.n)+'",c,p,1),c,p,0,'+n.i+","+n.end+',"'+n.otag+" "+n.ctag+'")){t.rs(c,p,function(c,p,t){',t.walk(n.nodes,e),e.code+="});c.pop();}"},"^":function(n,e){e.code+="if(!t.s(t."+c(n.n)+'("'+u(n.n)+'",c,p,1),c,p,1,0,0,"")){',t.walk(n.nodes,e),e.code+="};"},">":p,"<":function(n,e){var s={partials:{},code:"",subs:{},inPartial:!0};t.walk(n.nodes,s);var a=e.partials[p(n,e)];a.subs=s.subs,a.partials=s.partials},$:function(n,e){var s={subs:{},code:"",partials:e.partials,prefix:n.n};t.walk(n.nodes,s),e.subs[n.n]=s.code,e.inPartial||(e.code+='t.sub("'+u(n.n)+'",c,p,i);')},"\n":function(t,n){n.code+=f('"\\n"'+(t.last?"":" + i"))},_v:function(t,n){n.code+="t.b(t.v(t."+c(t.n)+'("'+u(t.n)+'",c,p,0)));'},_t:function(t,n){n.code+=f('"'+u(t.text)+'"')},"{":b,"&":b},t.walk=function(n,e){for(var s,a=0,r=n.length;r>a;a++)s=t.codegen[n[a].tag],s&&s(n[a],e);return e},t.parse=function(t,n,e){return e=e||{},a(t,"",[],e.sectionTags||[])},t.cache={},t.cacheKey=function(t,n){return[t,!!n.asString,!!n.disableLambda,n.delimiters,!!n.modelGet].join("||")},t.compile=function(n,e){e=e||{};var s=t.cacheKey(n,e),a=this.cache[s];if(a){var r=a.partials;for(var i in r)delete r[i].instance;return a}return a=this.generate(this.parse(this.scan(n,e.delimiters),n,e),n,e),this.cache[s]=a}}("undefined"!=typeof exports?exports:Hogan);var Mustache=function(t){function n(n,e,s,a){var r=this.f(n,e,s,0),i=e;return r&&(i=i.concat(r)),t.Template.prototype.rp.call(this,n,i,s,a)}var e=function(e,s,a){this.rp=n,t.Template.call(this,e,s,a)};e.prototype=t.Template.prototype;var s,a=function(){this.cache={},this.generate=function(t,n){return new e(new Function("c","p","i",t),n,s)}};return a.prototype=t,s=new a,{to_html:function(t,n,e,a){var r=s.compile(t),i=r.render(n,e);return a?(a(i),void 0):i}}}(Hogan),templates={};templates.controls=new Hogan.Template({code:function(t,n,e){var s=this;return s.b(e=e||""),s.b('<div class="player-controls">'),s.b("\n"+e),s.b(' <progress class="player-progress" max="100" value="0">'),s.b("\n"+e),s.b(" <span>0</span>% played"),s.b("\n"+e),s.b(" </progress>"),s.b("\n"+e),s.b(' <span class="player-controls-playback">'),s.b("\n"+e),s.b(' <button data-player="restart">'),s.b("\n"+e),s.b(' <svg><use xlink:href="#icon-refresh"></use></svg>'),s.b("\n"+e),s.b(' <span class="sr-only">Restart</span>'),s.b("\n"+e),s.b(" </button>"),s.b("\n"+e),s.b(' <button data-player="rewind">'),s.b("\n"+e),s.b(' <svg><use xlink:href="#icon-rewind"></use></svg>'),s.b("\n"+e),s.b(' <span class="sr-only">Rewind <span class="player-seek-time">10</span> seconds</span>'),s.b("\n"+e),s.b(" </button>"),s.b("\n"+e),s.b(' <button aria-label="{aria-label}" data-player="play">'),s.b("\n"+e),s.b(' <svg><use xlink:href="#icon-play"></use></svg>'),s.b("\n"+e),s.b(' <span class="sr-only">Play</span>'),s.b("\n"+e),s.b(" </button>"),s.b("\n"+e),s.b(' <button data-player="pause">'),s.b("\n"+e),s.b(' <svg><use xlink:href="#icon-pause"></use></svg>'),s.b("\n"+e),s.b(' <span class="sr-only">Pause</span>'),s.b("\n"+e),s.b(" </button>"),s.b("\n"+e),s.b(' <button data-player="fast-forward">'),s.b("\n"+e),s.b(' <svg><use xlink:href="#icon-fast-forward"></use></svg>'),s.b("\n"+e),s.b(' <span class="sr-only">Fast forward <span class="player-seek-time">10</span> seconds</span>'),s.b("\n"+e),s.b(" </button>"),s.b("\n"+e),s.b(' <span class="player-time">'),s.b("\n"+e),s.b(' <span class="sr-only">Time</span>'),s.b("\n"+e),s.b(' <span class="player-duration">00:00</span>'),s.b("\n"+e),s.b(" </span>"),s.b("\n"+e),s.b(" </span>"),s.b("\n"+e),s.b(' <span class="player-controls-sound">'),s.b("\n"+e),s.b(' <input class="inverted sr-only" id="mute{id}" type="checkbox" data-player="mute">'),s.b("\n"+e),s.b(' <label id="mute{id}" for="mute{id}">'),s.b("\n"+e),s.b(' <svg class="icon-muted"><use xlink:href="#icon-muted"></use></svg>'),s.b("\n"+e),s.b(' <svg><use xlink:href="#icon-sound"></use></svg>'),s.b("\n"+e),s.b(' <span class="sr-only">Mute</span>'),s.b("\n"+e),s.b(" </label>"),s.b("\n"),s.b("\n"+e),s.b(' <label for="volume{id}" class="sr-only">Volume:</label>'),s.b("\n"+e),s.b(' <input id="volume{id}" class="player-volume" type="range" min="0" max="10" value="5" data-player="volume">'),s.b("\n"),s.b("\n"+e),s.b(' <input class="sr-only" id="captions{id}" type="checkbox" data-player="captions">'),s.b("\n"+e),s.b(' <label for="captions{id}">'),s.b("\n"+e),s.b(' <svg><use xlink:href="#icon-bubble"></use></svg>'),s.b("\n"+e),s.b(' <span class="sr-only">Captions</span>'),s.b("\n"+e),s.b(" </label>"),s.b("\n"),s.b("\n"+e),s.b(' <button data-player="fullscreen">'),s.b("\n"+e),s.b(' <svg class="icon-exit-fullscreen"><use xlink:href="#icon-collapse"></use></svg>'),s.b("\n"+e),s.b(' <svg><use xlink:href="#icon-expand"></use></svg>'),s.b("\n"+e),s.b(' <span class="sr-only">Toggle fullscreen</span>'),s.b("\n"+e),s.b(" </button>"),s.b("\n"+e),s.b(" </span>"),s.b("\n"+e),s.b("</div>"),s.fl()},partials:{},subs:{}}),simpleMedia.on("setup",function(){}),simpleMedia.setup({title:"Video demo",html:templates.controls.render({})}); \ No newline at end of file
+var Hogan={};!function(t){function n(t,n,e){var s;return n&&"object"==typeof n&&(void 0!==n[t]?s=n[t]:e&&n.get&&"function"==typeof n.get&&(s=n.get(t))),s}function e(t,n,e,s,a,r){function i(){}function o(){}i.prototype=t,o.prototype=t.subs;var l,u=new i;u.subs=new o,u.subsText={},u.buf="",s=s||{},u.stackSubs=s,u.subsText=r;for(l in n)s[l]||(s[l]=n[l]);for(l in s)u.subs[l]=s[l];a=a||{},u.stackPartials=a;for(l in e)a[l]||(a[l]=e[l]);for(l in a)u.partials[l]=a[l];return u}function s(t){return String(null===t||void 0===t?"":t)}function a(t){return t=s(t),c.test(t)?t.replace(r,"&amp;").replace(i,"&lt;").replace(o,"&gt;").replace(l,"&#39;").replace(u,"&quot;"):t}t.Template=function(t,n,e,s){t=t||{},this.r=t.code||this.r,this.c=e,this.options=s||{},this.text=n||"",this.partials=t.partials||{},this.subs=t.subs||{},this.buf=""},t.Template.prototype={r:function(){return""},v:a,t:s,render:function(t,n,e){return this.ri([t],n||{},e)},ri:function(t,n,e){return this.r(t,n,e)},ep:function(t,n){var s=this.partials[t],a=n[s.name];if(s.instance&&s.base==a)return s.instance;if("string"==typeof a){if(!this.c)throw new Error("No compiler available.");a=this.c.compile(a,this.options)}if(!a)return null;if(this.partials[t].base=a,s.subs){n.stackText||(n.stackText={});for(key in s.subs)n.stackText[key]||(n.stackText[key]=void 0!==this.activeSub&&n.stackText[this.activeSub]?n.stackText[this.activeSub]:this.text);a=e(a,s.subs,s.partials,this.stackSubs,this.stackPartials,n.stackText)}return this.partials[t].instance=a,a},rp:function(t,n,e,s){var a=this.ep(t,e);return a?a.ri(n,e,s):""},rs:function(t,n,e){var s=t[t.length-1];if(!p(s))return e(t,n,this),void 0;for(var a=0;a<s.length;a++)t.push(s[a]),e(t,n,this),t.pop()},s:function(t,n,e,s,a,r,i){var o;return p(t)&&0===t.length?!1:("function"==typeof t&&(t=this.ms(t,n,e,s,a,r,i)),o=!!t,!s&&o&&n&&n.push("object"==typeof t?t:n[n.length-1]),o)},d:function(t,e,s,a){var r,i=t.split("."),o=this.f(i[0],e,s,a),l=this.options.modelGet,u=null;if("."===t&&p(e[e.length-2]))o=e[e.length-1];else for(var c=1;c<i.length;c++)r=n(i[c],o,l),void 0!==r?(u=o,o=r):o="";return a&&!o?!1:(a||"function"!=typeof o||(e.push(u),o=this.mv(o,e,s),e.pop()),o)},f:function(t,e,s,a){for(var r=!1,i=null,o=!1,l=this.options.modelGet,u=e.length-1;u>=0;u--)if(i=e[u],r=n(t,i,l),void 0!==r){o=!0;break}return o?(a||"function"!=typeof r||(r=this.mv(r,e,s)),r):a?!1:""},ls:function(t,n,e,a,r){var i=this.options.delimiters;return this.options.delimiters=r,this.b(this.ct(s(t.call(n,a)),n,e)),this.options.delimiters=i,!1},ct:function(t,n,e){if(this.options.disableLambda)throw new Error("Lambda features disabled.");return this.c.compile(t,this.options).render(n,e)},b:function(t){this.buf+=t},fl:function(){var t=this.buf;return this.buf="",t},ms:function(t,n,e,s,a,r,i){var o,l=n[n.length-1],u=t.call(l);return"function"==typeof u?s?!0:(o=this.activeSub&&this.subsText&&this.subsText[this.activeSub]?this.subsText[this.activeSub]:this.text,this.ls(u,l,e,o.substring(a,r),i)):u},mv:function(t,n,e){var a=n[n.length-1],r=t.call(a);return"function"==typeof r?this.ct(s(r.call(a)),a,e):r},sub:function(t,n,e,s){var a=this.subs[t];a&&(this.activeSub=t,a(n,e,this,s),this.activeSub=!1)}};var r=/&/g,i=/</g,o=/>/g,l=/\'/g,u=/\"/g,c=/[&<>\"\']/,p=Array.isArray||function(t){return"[object Array]"===Object.prototype.toString.call(t)}}("undefined"!=typeof exports?exports:Hogan),function(t){function n(t){"}"===t.n.substr(t.n.length-1)&&(t.n=t.n.substring(0,t.n.length-1))}function e(t){return t.trim?t.trim():t.replace(/^\s*|\s*$/g,"")}function s(t,n,e){if(n.charAt(e)!=t.charAt(0))return!1;for(var s=1,a=t.length;a>s;s++)if(n.charAt(e+s)!=t.charAt(s))return!1;return!0}function a(n,e,s,o){var l=[],u=null,c=null,p=null;for(c=s[s.length-1];n.length>0;){if(p=n.shift(),c&&"<"==c.tag&&!(p.tag in k))throw new Error("Illegal content in < super tag.");if(t.tags[p.tag]<=t.tags.$||r(p,o))s.push(p),p.nodes=a(n,p.tag,s,o);else{if("/"==p.tag){if(0===s.length)throw new Error("Closing tag without opener: /"+p.n);if(u=s.pop(),p.n!=u.n&&!i(p.n,u.n,o))throw new Error("Nesting error: "+u.n+" vs. "+p.n);return u.end=p.i,l}"\n"==p.tag&&(p.last=0==n.length||"\n"==n[0].tag)}l.push(p)}if(s.length>0)throw new Error("missing closing tag: "+s.pop().n);return l}function r(t,n){for(var e=0,s=n.length;s>e;e++)if(n[e].o==t.n)return t.tag="#",!0}function i(t,n,e){for(var s=0,a=e.length;a>s;s++)if(e[s].c==t&&e[s].o==n)return!0}function o(t){var n=[];for(var e in t)n.push('"'+u(e)+'": function(c,p,t,i) {'+t[e]+"}");return"{ "+n.join(",")+" }"}function l(t){var n=[];for(var e in t.partials)n.push('"'+u(e)+'":{name:"'+u(t.partials[e].name)+'", '+l(t.partials[e])+"}");return"partials: {"+n.join(",")+"}, subs: "+o(t.subs)}function u(t){return t.replace(m,"\\\\").replace(g,'\\"').replace(d,"\\n").replace(v,"\\r").replace(y,"\\u2028").replace(x,"\\u2029")}function c(t){return~t.indexOf(".")?"d":"f"}function p(t,n){var e="<"+(n.prefix||""),s=e+t.n+w++;return n.partials[s]={name:t.n,partials:{}},n.code+='t.b(t.rp("'+u(s)+'",c,p,"'+(t.indent||"")+'"));',s}function b(t,n){n.code+="t.b(t.t(t."+c(t.n)+'("'+u(t.n)+'",c,p,0)));'}function f(t){return"t.b("+t+");"}var h=/\S/,g=/\"/g,d=/\n/g,v=/\r/g,m=/\\/g,y=/\u2028/,x=/\u2029/;t.tags={"#":1,"^":2,"<":3,$:4,"/":5,"!":6,">":7,"=":8,_v:9,"{":10,"&":11,_t:12},t.scan=function(a,r){function i(){m.length>0&&(y.push({tag:"_t",text:new String(m)}),m="")}function o(){for(var n=!0,e=w;e<y.length;e++)if(n=t.tags[y[e].tag]<t.tags._v||"_t"==y[e].tag&&null===y[e].text.match(h),!n)return!1;return n}function l(t,n){if(i(),t&&o())for(var e,s=w;s<y.length;s++)y[s].text&&((e=y[s+1])&&">"==e.tag&&(e.indent=y[s].text.toString()),y.splice(s,1));else n||y.push({tag:"\n"});x=!1,w=y.length}function u(t,n){var s="="+S,a=t.indexOf(s,n),r=e(t.substring(t.indexOf("=",n)+1,a)).split(" ");return T=r[0],S=r[r.length-1],a+s.length-1}var c=a.length,p=0,b=1,f=2,g=p,d=null,v=null,m="",y=[],x=!1,k=0,w=0,T="{{",S="}}";for(r&&(r=r.split(" "),T=r[0],S=r[1]),k=0;c>k;k++)g==p?s(T,a,k)?(--k,i(),g=b):"\n"==a.charAt(k)?l(x):m+=a.charAt(k):g==b?(k+=T.length-1,v=t.tags[a.charAt(k+1)],d=v?a.charAt(k+1):"_v","="==d?(k=u(a,k),g=p):(v&&k++,g=f),x=k):s(S,a,k)?(y.push({tag:d,n:e(m),otag:T,ctag:S,i:"/"==d?x-T.length:k+S.length}),m="",k+=S.length-1,g=p,"{"==d&&("}}"==S?k++:n(y[y.length-1]))):m+=a.charAt(k);return l(x,!0),y};var k={_t:!0,"\n":!0,$:!0,"/":!0};t.stringify=function(n){return"{code: function (c,p,i) { "+t.wrapMain(n.code)+" },"+l(n)+"}"};var w=0;t.generate=function(n,e,s){w=0;var a={code:"",subs:{},partials:{}};return t.walk(n,a),s.asString?this.stringify(a,e,s):this.makeTemplate(a,e,s)},t.wrapMain=function(t){return'var t=this;t.b(i=i||"");'+t+"return t.fl();"},t.template=t.Template,t.makeTemplate=function(t,n,e){var s=this.makePartials(t);return s.code=new Function("c","p","i",this.wrapMain(t.code)),new this.template(s,n,this,e)},t.makePartials=function(t){var n,e={subs:{},partials:t.partials,name:t.name};for(n in e.partials)e.partials[n]=this.makePartials(e.partials[n]);for(n in t.subs)e.subs[n]=new Function("c","p","t","i",t.subs[n]);return e},t.codegen={"#":function(n,e){e.code+="if(t.s(t."+c(n.n)+'("'+u(n.n)+'",c,p,1),c,p,0,'+n.i+","+n.end+',"'+n.otag+" "+n.ctag+'")){t.rs(c,p,function(c,p,t){',t.walk(n.nodes,e),e.code+="});c.pop();}"},"^":function(n,e){e.code+="if(!t.s(t."+c(n.n)+'("'+u(n.n)+'",c,p,1),c,p,1,0,0,"")){',t.walk(n.nodes,e),e.code+="};"},">":p,"<":function(n,e){var s={partials:{},code:"",subs:{},inPartial:!0};t.walk(n.nodes,s);var a=e.partials[p(n,e)];a.subs=s.subs,a.partials=s.partials},$:function(n,e){var s={subs:{},code:"",partials:e.partials,prefix:n.n};t.walk(n.nodes,s),e.subs[n.n]=s.code,e.inPartial||(e.code+='t.sub("'+u(n.n)+'",c,p,i);')},"\n":function(t,n){n.code+=f('"\\n"'+(t.last?"":" + i"))},_v:function(t,n){n.code+="t.b(t.v(t."+c(t.n)+'("'+u(t.n)+'",c,p,0)));'},_t:function(t,n){n.code+=f('"'+u(t.text)+'"')},"{":b,"&":b},t.walk=function(n,e){for(var s,a=0,r=n.length;r>a;a++)s=t.codegen[n[a].tag],s&&s(n[a],e);return e},t.parse=function(t,n,e){return e=e||{},a(t,"",[],e.sectionTags||[])},t.cache={},t.cacheKey=function(t,n){return[t,!!n.asString,!!n.disableLambda,n.delimiters,!!n.modelGet].join("||")},t.compile=function(n,e){e=e||{};var s=t.cacheKey(n,e),a=this.cache[s];if(a){var r=a.partials;for(var i in r)delete r[i].instance;return a}return a=this.generate(this.parse(this.scan(n,e.delimiters),n,e),n,e),this.cache[s]=a}}("undefined"!=typeof exports?exports:Hogan);var Mustache=function(t){function n(n,e,s,a){var r=this.f(n,e,s,0),i=e;return r&&(i=i.concat(r)),t.Template.prototype.rp.call(this,n,i,s,a)}var e=function(e,s,a){this.rp=n,t.Template.call(this,e,s,a)};e.prototype=t.Template.prototype;var s,a=function(){this.cache={},this.generate=function(t,n){return new e(new Function("c","p","i",t),n,s)}};return a.prototype=t,s=new a,{to_html:function(t,n,e,a){var r=s.compile(t),i=r.render(n,e);return a?(a(i),void 0):i}}}(Hogan),templates={};templates.controls=new Hogan.Template({code:function(t,n,e){var s=this;return s.b(e=e||""),s.b('<div class="player-controls">'),s.b("\n"+e),s.b(' <progress class="player-progress" max="100" value="0">'),s.b("\n"+e),s.b(" <span>0</span>% played"),s.b("\n"+e),s.b(" </progress>"),s.b("\n"+e),s.b(' <span class="player-controls-playback">'),s.b("\n"+e),s.b(' <button data-player="restart">'),s.b("\n"+e),s.b(' <svg><use xlink:href="#icon-refresh"></use></svg>'),s.b("\n"+e),s.b(' <span class="sr-only">Restart</span>'),s.b("\n"+e),s.b(" </button>"),s.b("\n"+e),s.b(' <button data-player="rewind">'),s.b("\n"+e),s.b(' <svg><use xlink:href="#icon-rewind"></use></svg>'),s.b("\n"+e),s.b(' <span class="sr-only">Rewind <span class="player-seek-time">10</span> seconds</span>'),s.b("\n"+e),s.b(" </button>"),s.b("\n"+e),s.b(' <button aria-label="{aria-label}" data-player="play">'),s.b("\n"+e),s.b(' <svg><use xlink:href="#icon-play"></use></svg>'),s.b("\n"+e),s.b(' <span class="sr-only">Play</span>'),s.b("\n"+e),s.b(" </button>"),s.b("\n"+e),s.b(' <button data-player="pause">'),s.b("\n"+e),s.b(' <svg><use xlink:href="#icon-pause"></use></svg>'),s.b("\n"+e),s.b(' <span class="sr-only">Pause</span>'),s.b("\n"+e),s.b(" </button>"),s.b("\n"+e),s.b(' <button data-player="fast-forward">'),s.b("\n"+e),s.b(' <svg><use xlink:href="#icon-fast-forward"></use></svg>'),s.b("\n"+e),s.b(' <span class="sr-only">Fast forward <span class="player-seek-time">10</span> seconds</span>'),s.b("\n"+e),s.b(" </button>"),s.b("\n"+e),s.b(' <span class="player-time">'),s.b("\n"+e),s.b(' <span class="sr-only">Time</span>'),s.b("\n"+e),s.b(' <span class="player-duration">00:00</span>'),s.b("\n"+e),s.b(" </span>"),s.b("\n"+e),s.b(" </span>"),s.b("\n"+e),s.b(' <span class="player-controls-sound">'),s.b("\n"+e),s.b(' <input class="inverted sr-only" id="mute{id}" type="checkbox" data-player="mute">'),s.b("\n"+e),s.b(' <label id="mute{id}" for="mute{id}">'),s.b("\n"+e),s.b(' <svg class="icon-muted"><use xlink:href="#icon-muted"></use></svg>'),s.b("\n"+e),s.b(' <svg><use xlink:href="#icon-sound"></use></svg>'),s.b("\n"+e),s.b(' <span class="sr-only">Mute</span>'),s.b("\n"+e),s.b(" </label>"),s.b("\n"),s.b("\n"+e),s.b(' <label for="volume{id}" class="sr-only">Volume:</label>'),s.b("\n"+e),s.b(' <input id="volume{id}" class="player-volume" type="range" min="0" max="10" value="5" data-player="volume">'),s.b("\n"),s.b("\n"+e),s.b(' <input class="sr-only" id="captions{id}" type="checkbox" data-player="captions">'),s.b("\n"+e),s.b(' <label for="captions{id}">'),s.b("\n"+e),s.b(' <svg><use xlink:href="#icon-bubble"></use></svg>'),s.b("\n"+e),s.b(' <span class="sr-only">Captions</span>'),s.b("\n"+e),s.b(" </label>"),s.b("\n"),s.b("\n"+e),s.b(' <button data-player="fullscreen">'),s.b("\n"+e),s.b(' <svg class="icon-exit-fullscreen"><use xlink:href="#icon-collapse"></use></svg>'),s.b("\n"+e),s.b(' <svg><use xlink:href="#icon-expand"></use></svg>'),s.b("\n"+e),s.b(' <span class="sr-only">Toggle fullscreen</span>'),s.b("\n"+e),s.b(" </button>"),s.b("\n"+e),s.b(" </span>"),s.b("\n"+e),s.b("</div>"),s.fl()},partials:{},subs:{}}),simpleMedia.on("setup",function(){});var players=simpleMedia.setup({title:"Video demo",html:templates.controls.render({})});console.log(players); \ No newline at end of file
diff --git a/dist/js/simple-media.js b/dist/js/simple-media.js
index a4352aee..6b434154 100644
--- a/dist/js/simple-media.js
+++ b/dist/js/simple-media.js
@@ -1 +1 @@
-!function(e){"use strict";function t(){var e,t,n,r=navigator.userAgent,s=navigator.appName,a=""+parseFloat(navigator.appVersion),o=parseInt(navigator.appVersion,10);return-1!==navigator.appVersion.indexOf("Windows NT")&&-1!==navigator.appVersion.indexOf("rv:11")?(s="IE",a="11;"):-1!==(t=r.indexOf("MSIE"))?(s="IE",a=r.substring(t+5)):-1!==(t=r.indexOf("Chrome"))?(s="Chrome",a=r.substring(t+7)):-1!==(t=r.indexOf("Safari"))?(s="Safari",a=r.substring(t+7),-1!==(t=r.indexOf("Version"))&&(a=r.substring(t+8))):-1!==(t=r.indexOf("Firefox"))?(s="Firefox",a=r.substring(t+8)):(e=r.lastIndexOf(" ")+1)<(t=r.lastIndexOf("/"))&&(s=r.substring(e,t),a=r.substring(t+1),s.toLowerCase()==s.toUpperCase()&&(s=navigator.appName)),-1!==(n=a.indexOf(";"))&&(a=a.substring(0,n)),-1!==(n=a.indexOf(" "))&&(a=a.substring(0,n)),o=parseInt(""+a,10),isNaN(o)&&(a=""+parseFloat(navigator.appVersion),o=parseInt(navigator.appVersion,10)),[s,o]}function n(e){var t=[];return t=e.split(" --> "),s(t[0])}function r(e){var t=[];return t=e.split(" --> "),s(t[1])}function s(e){if(null===e||void 0===e)return 0;var t,n=[],r=[];return n=e.split(","),r=n[0].split(":"),t=Math.floor(60*r[0]*60)+Math.floor(60*r[1])+Math.floor(r[2])}function a(e){for(e.subcount=0;r(e.captions[e.subcount][0])<e.media.currentTime.toFixed(1);)if(e.subcount++,e.subcount>e.captions.length-1){e.subcount=e.captions.length-1;break}}function o(e){e.container.className+=" "+C.classes.captions.enabled,C.captions.defaultActive&&(e.container.className+=" "+C.classes.captions.active,e.buttons.captions.setAttribute("checked","checked"))}function i(e,t,n){return e.replace(new RegExp(t.replace(/([.*+?^=!:${}()|\[\]\/\\])/g,"\\$1"),"g"),n)}function c(e,t){e.length||(e=[e]);for(var n=e.length-1;n>=0;n--){var r=n>0?t.cloneNode(!0):t,s=e[n],a=s.parentNode,o=s.nextSibling;r.appendChild(s),o?a.insertBefore(r,o):a.appendChild(r)}}function l(e){var t=p().isFullScreen()?{x:0,y:0}:u(e.currentTarget);return{x:e.clientX-t.x,y:e.clientY-t.y}}function u(e){for(var t=0,n=0;e;)t+=e.offsetLeft-e.scrollLeft+e.clientLeft,n+=e.offsetTop-e.scrollTop+e.clientTop,e=e.offsetParent;return{x:t,y:n}}function d(e,t){for(var n in t)t[n]&&t[n].constructor&&t[n].constructor===Object?(e[n]=e[n]||{},d(e[n],t[n])):e[n]=t[n];return e}function p(){var e={supportsFullScreen:!1,isFullScreen:function(){return!1},requestFullScreen:function(){},cancelFullScreen:function(){},fullScreenEventName:"",element:null,prefix:""},t="webkit moz o ms khtml".split(" ");if("undefined"!=typeof document.cancelFullScreen)e.supportsFullScreen=!0;else for(var n=0,r=t.length;r>n;n++)if(e.prefix=t[n],"undefined"!=typeof document[e.prefix+"CancelFullScreen"]){e.supportsFullScreen=!0;break}return"webkit"===e.prefix&&navigator.userAgent.match(/Version\/[\d\.]+.*Safari/)&&(e.supportsFullScreen=!1),e.supportsFullScreen&&(e.fullScreenEventName=e.prefix+"fullscreenchange",e.isFullScreen=function(){switch(this.prefix){case"":return document.fullScreen;case"webkit":return document.webkitIsFullScreen;default:return document[this.prefix+"FullScreen"]}},e.requestFullScreen=function(e){return""===this.prefix?e.requestFullScreen():e[this.prefix+"RequestFullScreen"]("webkit"===this.prefix?e.ALLOW_KEYBOARD_INPUT:null)},e.cancelFullScreen=function(){return""===this.prefix?document.cancelFullScreen():document[this.prefix+"CancelFullScreen"]()},e.element=function(){return""===this.prefix?document.fullscreenElement:document[this.prefix+"FullscreenElement"]}),e}function f(e){var t,n=A[e]||[],r=n.length;for(t=0;r>t;t++)n[t].apply(I.media,[])}function m(){C.debug&&console.log("Injecting custom controls");var e=C.html;e=i(e,"{aria-label}",C.playAriaLabel),e=i(e,"{id}",I.random),I.container.insertAdjacentHTML("beforeend",e)}function b(e){return I.container.querySelectorAll(e)}function v(e){return b(e)[0]}function g(){I.controls=v(C.selectors.controls),I.buttons={},I.buttons.play=v(C.selectors.buttons.play),I.buttons.pause=v(C.selectors.buttons.pause),I.buttons.restart=v(C.selectors.buttons.restart),I.buttons.rewind=v(C.selectors.buttons.rewind),I.buttons.forward=v(C.selectors.buttons.forward),I.buttons.mute=v(C.selectors.buttons.mute),I.buttons.captions=v(C.selectors.buttons.captions),I.buttons.fullscreen=v(C.selectors.buttons.fullscreen),I.progress={},I.progress.bar=v(C.selectors.progress),I.progress.text=I.progress.bar.getElementsByTagName("span")[0],I.volume=v(C.selectors.buttons.volume),I.duration=v(C.selectors.duration),I.seekTime=b(C.selectors.seekTime)}function x(){I.media.play(),I.container.className=I.container.className.replace(C.classes.stopped,C.classes.playing)}function T(){I.media.pause(),I.container.className=I.container.className.replace(C.classes.playing,C.classes.stopped)}function h(){I.media.currentTime=0,I.isTextTracks||(I.subcount=0),x()}function y(){I.volume.value=C.volume,I.media.volume=parseFloat(C.volume/10),k()}function k(){0===I.media.volume||I.media.muted?I.container.className+=" "+C.classes.muted:I.container.className=I.container.className.replace(C.classes.muted,"")}function w(){if(I.media=I.container.querySelectorAll("audio, video")[0],!I.media)return console.error("No audio or video element found!"),!1;if(null===I.media.getAttribute("autoplay")&&(I.container.className+=" "+C.classes.stopped),I.media.removeAttribute("controls"),I.type="video"==I.media.tagName.toLowerCase()?"video":"audio","video"===I.type){var e=document.createElement("div");e.setAttribute("class",C.classes.videoContainer),c(I.media,e),I.videoContainer=e}}function N(){if("video"===I.type){I.videoContainer.insertAdjacentHTML("afterbegin","<div class='"+C.selectors.captions.replace(".","")+"'></div>"),I.captionsContainer=v(C.selectors.captions),I.isTextTracks=!1,I.media.textTracks&&(I.isTextTracks=!0);for(var e,t="",s=I.media.childNodes,a=0;a<s.length;a++)"track"===s[a].nodeName.toLowerCase()&&(e=s[a].getAttribute("kind"),"captions"===e&&(t=s[a].getAttribute("src")));if(I.captionExists=!0,""===t?(I.captionExists=!1,C.debug&&console.log("No caption track found.")):C.debug&&console.log("Caption track found; URI: "+t),I.captionExists){var i,c,l={};if("IE"===I.browserName&&10===I.browserMajorVersion||"IE"===I.browserName&&11===I.browserMajorVersion||"Firefox"===I.browserName&&I.browserMajorVersion>=31||"Safari"===I.browserName&&I.browserMajorVersion>=7)for(C.debug&&console.log("Detected IE 10/11 or Firefox 31+ or Safari 7+"),I.isTextTracks=!1,l={},i=I.media.textTracks,c=0;c<i.length;c++)l=I.media.textTracks[c],l.mode="hidden";if(I.isTextTracks)for(C.debug&&console.log("textTracks supported"),o(I),l={},i=I.media.textTracks,c=0;c<i.length;c++)l=I.media.textTracks[c],l.mode="hidden","captions"===l.kind&&l.addEventListener("cuechange",function(){this.activeCues[0]&&this.activeCues[0].hasOwnProperty("text")&&(I.captionsContainer.innerHTML=this.activeCues[0].text)},!1);else if(C.debug&&console.log("textTracks not supported so rendering captions manually"),o(I),I.currentCaption="",I.subcount=0,I.captions=[],I.media.addEventListener("timeupdate",function(){I.media.currentTime.toFixed(1)>n(I.captions[I.subcount][0])&&I.media.currentTime.toFixed(1)<r(I.captions[I.subcount][0])&&(I.currentCaption=I.captions[I.subcount][1]),I.media.currentTime.toFixed(1)>r(I.captions[I.subcount][0])&&I.subcount<I.captions.length-1&&I.subcount++,I.captionsContainer.innerHTML=I.currentCaption},!1),""!==t){var u;window.XMLHttpRequest?u=new XMLHttpRequest:window.ActiveXObject&&(u=new ActiveXObject("Microsoft.XMLHTTP")),u.onreadystatechange=function(){if(4===u.readyState)if(200===u.status){C.debug&&console.log("xhr = 200"),I.captions=[];var e,t=[],n=u.responseText;t=n.split("\n\n");for(var r=0;r<t.length;r++)e=t[r],I.captions[r]=[],I.captions[r]=e.split("\n");I.captions.shift(),C.debug&&console.log("Successfully loaded the caption file via ajax.")}else C.debug&&console.log("There was a problem loading the caption file via ajax.")},u.open("get",t,!0),u.send()}"Safari"===I.browserName&&7===I.browserMajorVersion&&(C.debug&&console.log("Safari 7 detected; removing track from DOM"),i=I.media.getElementsByTagName("track"),I.media.removeChild(i[0]))}else I.container.className=I.container.className.replace(C.classes.captions.enabled,"")}}function S(){I.seekTime[0].innerHTML=C.seekInterval,I.seekTime[1].innerHTML=C.seekInterval}function F(){"video"===I.type&&C.fullscreen.enabled&&(C.debug&&console.log(M.supportsFullScreen?"Fullscreen supported":"No fullscreen supported"),M.supportsFullScreen?(C.debug&&console.log("Fullscreen enabled"),I.container.className+=" "+C.classes.fullscreen.enabled):C.debug&&console.warn("Fullscreen not supported"))}function L(){I.buttons.fullscreen.addEventListener("click",function(){M.isFullScreen()?M.cancelFullScreen():M.requestFullScreen(I.container)},!1),"video"===I.type&&C.clickToPause&&I.videoContainer.addEventListener("click",function(){I.media.paused?x():I.media.ended?h():T()},!1),I.buttons.play.addEventListener("click",function(){x(),I.buttons.pause.focus()},!1),I.buttons.pause.addEventListener("click",function(){T(),I.buttons.play.focus()},!1),I.buttons.restart.addEventListener("click",h,!1),I.buttons.rewind.addEventListener("click",function(){var e=I.media.currentTime-C.seekInterval;I.media.currentTime=0>e?0:e,I.isTextTracks||"video"!==I.type||a(I)},!1),I.buttons.forward.addEventListener("click",function(){var e=I.media.currentTime+C.seekInterval;I.media.currentTime=e>I.media.duration?I.media.duration:e,I.isTextTracks||"video"!==I.type||a(I)},!1),I.volume.addEventListener("change",function(){C.volume=this.value,y()},!1),I.buttons.mute.addEventListener("click",function(){I.media.muted=I.media.muted===!0?!1:!0,k()},!1),I.media.addEventListener("timeupdate",function(){I.secs=parseInt(I.media.currentTime%60),I.mins=parseInt(I.media.currentTime/60%60),I.secs=("0"+I.secs).slice(-2),I.mins=("0"+I.mins).slice(-2),I.duration.innerHTML=I.mins+":"+I.secs},!1),I.media.addEventListener("timeupdate",function(){I.percent=100/I.media.duration*I.media.currentTime,I.percent>0&&(I.progress.bar.value=I.percent,I.progress.text.innerHTML=I.percent)},!1),I.progress.bar.addEventListener("click",function(e){I.pos=l(e).x/this.offsetWidth,I.media.currentTime=I.pos*I.media.duration,I.isTextTracks||"video"!==I.type||a(I)}),I.buttons.captions.addEventListener("click",function(){this.checked?I.container.className+=" "+C.classes.captions.active:I.container.className=I.container.className.replace(C.classes.captions.active,"")},!1),I.media.addEventListener("ended",function(){"video"===I.type&&(I.captionsContainer.innerHTML=""),I.container.className=I.container.className.replace(C.classes.playing,C.classes.stopped)})}function E(e){I.container=e,w(),I.random=Math.floor(1e4*Math.random()),m(),g(),y(),F(),N(),S(),L()}var M,C,I={},A={},O={enabled:!0,debug:!1,seekInterval:10,volume:5,clickToPause:!0,selectors:{container:".player",controls:".player-controls",buttons:{play:"[data-player='play']",pause:"[data-player='pause']",restart:"[data-player='restart']",rewind:"[data-player='rewind']",forward:"[data-player='fast-forward']",mute:"[data-player='mute']",volume:"[data-player='volume']",captions:"[data-player='captions']",fullscreen:"[data-player='fullscreen']"},progress:".player-progress",captions:".player-captions",duration:".player-duration",seekTime:".player-seek-time"},classes:{videoContainer:"player-video",stopped:"stopped",playing:"playing",muted:"muted",captions:{active:"captions-active",enabled:"captions-enabled"},fullscreen:{enabled:"fullscreen-enabled"}},captions:{defaultActive:!0},fullscreen:{enabled:!0}};e.on=function(e,t){A[e]||(A[e]=[]),A[e].push(t)},e.setup=function(e){if(C=d(O,e),!C.enabled)return!1;if(M=p(),I.browserInfo=t(),I.browserName=I.browserInfo[0],I.browserMajorVersion=I.browserInfo[1],C.debug&&console.log(I.browserName+" "+I.browserMajorVersion),"IE"===I.browserName&&(8===I.browserMajorVersion||9===I.browserMajorVersion))return C.debug&&console.error("Browser not suppported."),!1;C.playAriaLabel="undefined"!=typeof C.title&&C.title.length?"Play "+C.title:"Play";var n=document.querySelector(C.selectors.container);return null===n?(C.debug&&console.error("Selector "+C.selectors.container+" not found!"),!1):(E(n),f("setup"),void 0)}}(this.simpleMedia=this.simpleMedia||{}); \ No newline at end of file
+!function(e){"use strict";function t(){var e,t,n,r=navigator.userAgent,o=navigator.appName,s=""+parseFloat(navigator.appVersion),i=parseInt(navigator.appVersion,10);return-1!==navigator.appVersion.indexOf("Windows NT")&&-1!==navigator.appVersion.indexOf("rv:11")?(o="IE",s="11;"):-1!==(t=r.indexOf("MSIE"))?(o="IE",s=r.substring(t+5)):-1!==(t=r.indexOf("Chrome"))?(o="Chrome",s=r.substring(t+7)):-1!==(t=r.indexOf("Safari"))?(o="Safari",s=r.substring(t+7),-1!==(t=r.indexOf("Version"))&&(s=r.substring(t+8))):-1!==(t=r.indexOf("Firefox"))?(o="Firefox",s=r.substring(t+8)):(e=r.lastIndexOf(" ")+1)<(t=r.lastIndexOf("/"))&&(o=r.substring(e,t),s=r.substring(t+1),o.toLowerCase()==o.toUpperCase()&&(o=navigator.appName)),-1!==(n=s.indexOf(";"))&&(s=s.substring(0,n)),-1!==(n=s.indexOf(" "))&&(s=s.substring(0,n)),i=parseInt(""+s,10),isNaN(i)&&(s=""+parseFloat(navigator.appVersion),i=parseInt(navigator.appVersion,10)),[o,i]}function n(e,t,n){return e.replace(new RegExp(t.replace(/([.*+?^=!:${}()|\[\]\/\\])/g,"\\$1"),"g"),n)}function r(e,t){e.length||(e=[e]);for(var n=e.length-1;n>=0;n--){var r=n>0?t.cloneNode(!0):t,o=e[n],s=o.parentNode,i=o.nextSibling;r.appendChild(o),i?s.insertBefore(r,i):s.appendChild(r)}}function o(e,t,n){if(e)if(e.classList)e.classList[n?"add":"remove"](t);else{var r=(" "+e.className+" ").replace(/\s+/g," ").replace(" "+t+" ","");e.className=r+(n?" "+t:"")}}function s(e,t,n){e.addEventListener(t,n,!1)}function i(e){var t=u().isFullScreen()?{x:0,y:0}:a(e.currentTarget);return{x:e.clientX-t.x,y:e.clientY-t.y}}function a(e){for(var t=0,n=0;e;)t+=e.offsetLeft-e.scrollLeft+e.clientLeft,n+=e.offsetTop-e.scrollTop+e.clientTop,e=e.offsetParent;return{x:t,y:n}}function c(e,t){for(var n in t)t[n]&&t[n].constructor&&t[n].constructor===Object?(e[n]=e[n]||{},c(e[n],t[n])):e[n]=t[n];return e}function u(){var e={supportsFullScreen:!1,isFullScreen:function(){return!1},requestFullScreen:function(){},cancelFullScreen:function(){},fullScreenEventName:"",element:null,prefix:""},t="webkit moz o ms khtml".split(" ");if("undefined"!=typeof document.cancelFullScreen)e.supportsFullScreen=!0;else for(var n=0,r=t.length;r>n;n++)if(e.prefix=t[n],"undefined"!=typeof document[e.prefix+"CancelFullScreen"]){e.supportsFullScreen=!0;break}return"webkit"===e.prefix&&navigator.userAgent.match(/Version\/[\d\.]+.*Safari/)&&(e.supportsFullScreen=!1),e.supportsFullScreen&&(e.fullScreenEventName=e.prefix+"fullscreenchange",e.isFullScreen=function(){switch(this.prefix){case"":return document.fullScreen;case"webkit":return document.webkitIsFullScreen;default:return document[this.prefix+"FullScreen"]}},e.requestFullScreen=function(e){return""===this.prefix?e.requestFullScreen():e[this.prefix+"RequestFullScreen"]("webkit"===this.prefix?e.ALLOW_KEYBOARD_INPUT:null)},e.cancelFullScreen=function(){return""===this.prefix?document.cancelFullScreen():document[this.prefix+"CancelFullScreen"]()},e.element=function(){return""===this.prefix?document.fullscreenElement:document[this.prefix+"FullscreenElement"]}),e}function l(e,t){var n,r=m[e]||[],o=r.length;for(n=0;o>n;n++)r[n].apply(t,[])}function d(e){function a(){for(j.subcount=0;d(j.captions[j.subcount][0])<j.media.currentTime.toFixed(1);)if(j.subcount++,j.subcount>j.captions.length-1){j.subcount=j.captions.length-1;break}}function c(){o(j.container,f.classes.captions.enabled,!0),f.captions.defaultActive&&(o(j.container,f.classes.captions.active,!0),j.buttons.captions.setAttribute("checked","checked"))}function l(e){var t=[];return t=e.split(" --> "),m(t[0])}function d(e){var t=[];return t=e.split(" --> "),m(t[1])}function m(e){if(null===e||void 0===e)return 0;var t,n=[],r=[];return n=e.split(","),r=n[0].split(":"),t=Math.floor(60*r[0]*60)+Math.floor(60*r[1])+Math.floor(r[2])}function b(e){return j.container.querySelectorAll(e)}function g(e){return b(e)[0]}function v(){f.debug&&console.log("Injecting custom controls");var e=f.html;e=n(e,"{aria-label}",f.playAriaLabel),e=n(e,"{id}",j.random),j.container.insertAdjacentHTML("beforeend",e)}function h(){j.controls=g(f.selectors.controls),j.buttons={},j.buttons.play=g(f.selectors.buttons.play),j.buttons.pause=g(f.selectors.buttons.pause),j.buttons.restart=g(f.selectors.buttons.restart),j.buttons.rewind=g(f.selectors.buttons.rewind),j.buttons.forward=g(f.selectors.buttons.forward),j.buttons.mute=g(f.selectors.buttons.mute),j.buttons.captions=g(f.selectors.buttons.captions),j.buttons.fullscreen=g(f.selectors.buttons.fullscreen),j.progress={},j.progress.bar=g(f.selectors.progress),j.progress.text=j.progress.bar.getElementsByTagName("span")[0],j.volume=g(f.selectors.buttons.volume),j.duration=g(f.selectors.duration),j.seekTime=b(f.selectors.seekTime)}function x(){if(j.media=j.container.querySelectorAll("audio, video")[0],!j.media)return console.error("No audio or video element found!"),!1;if(o(j.container,f.classes.stopped,null===j.media.getAttribute("autoplay")),j.media.removeAttribute("controls"),j.type="video"==j.media.tagName.toLowerCase()?"video":"audio","video"===j.type){var e=document.createElement("div");e.setAttribute("class",f.classes.videoContainer),r(j.media,e),j.videoContainer=e}}function y(){if("video"===j.type){j.videoContainer.insertAdjacentHTML("afterbegin","<div class='"+f.selectors.captions.replace(".","")+"'></div>"),j.captionsContainer=g(f.selectors.captions),j.isTextTracks=!1,j.media.textTracks&&(j.isTextTracks=!0);for(var e,t="",n=j.media.childNodes,r=0;r<n.length;r++)"track"===n[r].nodeName.toLowerCase()&&(e=n[r].getAttribute("kind"),"captions"===e&&(t=n[r].getAttribute("src")));if(j.captionExists=!0,""===t?(j.captionExists=!1,f.debug&&console.log("No caption track found.")):f.debug&&console.log("Caption track found; URI: "+t),j.captionExists){var i,a,u={};if("IE"===j.browserName&&10===j.browserMajorVersion||"IE"===j.browserName&&11===j.browserMajorVersion||"Firefox"===j.browserName&&j.browserMajorVersion>=31||"Safari"===j.browserName&&j.browserMajorVersion>=7)for(f.debug&&console.log("Detected IE 10/11 or Firefox 31+ or Safari 7+"),j.isTextTracks=!1,u={},i=j.media.textTracks,a=0;a<i.length;a++)u=j.media.textTracks[a],u.mode="hidden";if(j.isTextTracks)for(f.debug&&console.log("textTracks supported"),c(j),u={},i=j.media.textTracks,a=0;a<i.length;a++)u=j.media.textTracks[a],u.mode="hidden","captions"===u.kind&&s(u,"cuechange",function(){this.activeCues[0]&&this.activeCues[0].hasOwnProperty("text")&&(j.captionsContainer.innerHTML=this.activeCues[0].text)});else if(f.debug&&console.log("textTracks not supported so rendering captions manually"),c(j),j.currentCaption="",j.subcount=0,j.captions=[],s(j.media,"timeupdate",function(){j.media.currentTime.toFixed(1)>l(j.captions[j.subcount][0])&&j.media.currentTime.toFixed(1)<d(j.captions[j.subcount][0])&&(j.currentCaption=j.captions[j.subcount][1]),j.media.currentTime.toFixed(1)>d(j.captions[j.subcount][0])&&j.subcount<j.captions.length-1&&j.subcount++,j.captionsContainer.innerHTML=j.currentCaption}),""!==t){var p;window.XMLHttpRequest?p=new XMLHttpRequest:window.ActiveXObject&&(p=new ActiveXObject("Microsoft.XMLHTTP")),p.onreadystatechange=function(){if(4===p.readyState)if(200===p.status){f.debug&&console.log("xhr = 200"),j.captions=[];var e,t=[],n=p.responseText;t=n.split("\n\n");for(var r=0;r<t.length;r++)e=t[r],j.captions[r]=[],j.captions[r]=e.split("\n");j.captions.shift(),f.debug&&console.log("Successfully loaded the caption file via ajax.")}else f.debug&&console.log("There was a problem loading the caption file via ajax.")},p.open("get",t,!0),p.send()}"Safari"===j.browserName&&7===j.browserMajorVersion&&(f.debug&&console.log("Safari 7 detected; removing track from DOM"),i=j.media.getElementsByTagName("track"),j.media.removeChild(i[0]))}else o(j.container,f.classes.captions.enabled)}}function T(){j.seekTime[0].innerHTML=f.seekInterval,j.seekTime[1].innerHTML=f.seekInterval}function k(){"video"===j.type&&f.fullscreen.enabled&&(f.debug&&console.log(p.supportsFullScreen?"Fullscreen supported":"No fullscreen supported"),p.supportsFullScreen?(f.debug&&console.log("Fullscreen enabled"),o(j.container,f.classes.fullscreen.enabled,!0)):f.debug&&console.warn("Fullscreen not supported"))}function w(){j.media.play(),o(j.container,f.classes.stopped),o(j.container,f.classes.playing,!0)}function S(){j.media.pause(),o(j.container,f.classes.playing),o(j.container,f.classes.stopped,!0)}function F(){j.media.currentTime=0,j.isTextTracks||(j.subcount=0),w()}function M(){var e=j.media.currentTime-f.seekInterval;j.media.currentTime=0>e?0:e,j.isTextTracks||"video"!==j.type||a(j)}function N(){var e=j.media.currentTime+f.seekInterval;j.media.currentTime=e>j.media.duration?j.media.duration:e,j.isTextTracks||"video"!==j.type||a(j)}function C(){p.isFullScreen()?p.cancelFullScreen():p.requestFullScreen(j.container)}function I(e){j.volume.value=e,j.media.volume=parseFloat(e/10),E()}function L(e){"undefined"==typeof active&&(e=!j.media.muted,j.buttons.mute.checked=e),j.media.muted=e,E()}function A(e){"undefined"==typeof e&&(e=-1===j.container.className.indexOf(f.classes.captions.active),j.buttons.captions.checked=e),e?o(j.container,f.classes.captions.active,!0):o(j.container,f.classes.captions.active)}function E(){o(j.container,f.classes.muted,0===j.media.volume||j.media.muted)}function O(){s(j.buttons.fullscreen,"click",C),"video"===j.type&&f.clickToPause&&s(j.videoContainer,"click",function(){j.media.paused?w():j.media.ended?F():S()}),s(j.buttons.play,"click",function(){w(),j.buttons.pause.focus()}),s(j.buttons.pause,"click",function(){S(),j.buttons.play.focus()}),s(j.buttons.restart,"click",F),s(j.buttons.rewind,"click",M),s(j.buttons.forward,"click",N),s(j.volume,"change",function(){I(this.value)}),s(j.buttons.mute,"change",function(){L(this.checked)}),s(j.media,"timeupdate",function(){j.secs=parseInt(j.media.currentTime%60),j.mins=parseInt(j.media.currentTime/60%60),j.secs=("0"+j.secs).slice(-2),j.mins=("0"+j.mins).slice(-2),j.duration.innerHTML=j.mins+":"+j.secs}),s(j.media,"timeupdate",function(){j.percent=100/j.media.duration*j.media.currentTime,j.percent>0&&(j.progress.bar.value=j.percent,j.progress.text.innerHTML=j.percent)}),s(j.progress.bar,"click",function(e){j.pos=i(e).x/this.offsetWidth,j.media.currentTime=j.pos*j.media.duration,j.isTextTracks||"video"!==j.type||a(j)}),s(j.buttons.captions,"click",function(){A(this.checked)}),s(j.media,"ended",function(){"video"===j.type&&(j.captionsContainer.innerHTML=""),o(j.container,f.classes.stopped,!0),o(j.container,f.classes.playing)})}function V(){return p=u(),j.browserInfo=t(),j.browserName=j.browserInfo[0],j.browserMajorVersion=j.browserInfo[1],f.debug&&console.log(j.browserName+" "+j.browserMajorVersion),"IE"!==j.browserName||8!==j.browserMajorVersion&&9!==j.browserMajorVersion?(f.playAriaLabel="undefined"!=typeof f.title&&f.title.length?"Play "+f.title:"Play",x(),j.random=Math.floor(1e4*Math.random()),v(),h(),I(f.volume),k(),y(),T(),O(),void 0):(f.debug&&console.error("Browser not suppported."),!1)}var j=this;return j.container=e,V(),{media:j.media,play:w,pause:S,restart:F,rewind:M,forward:N,setVolume:I,toggleMute:L,toggleCaptions:A}}var p,f,m={},b={enabled:!0,debug:!1,seekInterval:10,volume:5,clickToPause:!0,selectors:{container:".player",controls:".player-controls",buttons:{play:"[data-player='play']",pause:"[data-player='pause']",restart:"[data-player='restart']",rewind:"[data-player='rewind']",forward:"[data-player='fast-forward']",mute:"[data-player='mute']",volume:"[data-player='volume']",captions:"[data-player='captions']",fullscreen:"[data-player='fullscreen']"},progress:".player-progress",captions:".player-captions",duration:".player-duration",seekTime:".player-seek-time"},classes:{videoContainer:"player-video",stopped:"stopped",playing:"playing",muted:"muted",captions:{active:"captions-active",enabled:"captions-enabled"},fullscreen:{enabled:"fullscreen-enabled"}},captions:{defaultActive:!0},fullscreen:{enabled:!0}};e.on=function(e,t){m[e]||(m[e]=[]),m[e].push(t)},e.setup=function(e){if(f=c(b,e),!f.enabled)return!1;for(var t=[],n=document.querySelectorAll(f.selectors.container),r=n.length-1;r>=0;r--){var o=n[r],s=new d(o);t.push(s),o.player=s}return l("setup",t),t}}(this.simpleMedia=this.simpleMedia||{}); \ No newline at end of file
diff --git a/docs/audio.html b/docs/audio.html
deleted file mode 100644
index bf48615b..00000000
--- a/docs/audio.html
+++ /dev/null
@@ -1,41 +0,0 @@
-<!doctype html>
-<html lang="en" class="example-audio">
- <head>
- <meta charset="utf-8" />
- <title>Simple Media</title>
- <meta name="description" content="Custom HTML5 video controls and WebVTT captions.">
- <meta name="author" content="Sam Potts">
- <meta name="viewport" content="width=device-width, initial-scale=1">
-
- <!-- Styles -->
- <link rel="stylesheet" href="../dist/css/simple-media.css">
- <link rel="stylesheet" href="../dist/css/docs.css">
- </head>
- <body>
- <header>
- <h1>Simple Media</h1>
- <p>A simple HTML5 media player</p>
- </header>
-
- <div class="player">
- <audio controls>
- <!-- Audio files -->
- <source src="../media/Covox_-_Switchblade_Squadron.mp3" type="audio/mp3">
-
- <!-- Fallback for browsers that don't support the <audio> element -->
- <div>
- <a href="../media/Covox_-_Switchblade_Squadron.mp3">Download it</a>
- </div>
- </video>
- </div>
-
- <!-- Load SVG defs -->
- <script>(function(d,p){var a=new XMLHttpRequest(),b=d.body; a.open("GET",p,!0);a.send();a.onload=function(){var c=d.createElement("div");c.style.display="none";c.innerHTML=a.responseText;b.insertBefore(c,b.childNodes[0])}})(document,"../dist/svg/sprite.svg");</script>
-
- <!-- Core player -->
- <script src="../dist/js/simple-media.js"></script>
-
- <!-- Docs setup -->
- <script src="../dist/js/docs.js"></script>
- </body>
-</html> \ No newline at end of file
diff --git a/docs/index.html b/docs/index.html
new file mode 100644
index 00000000..a47f3ed0
--- /dev/null
+++ b/docs/index.html
@@ -0,0 +1,67 @@
+<!doctype html>
+<html lang="en">
+ <head>
+ <meta charset="utf-8" />
+ <title>Simple Media</title>
+ <meta name="description" content="Custom HTML5 video controls and WebVTT captions.">
+ <meta name="author" content="Sam Potts">
+ <meta name="viewport" content="width=device-width, initial-scale=1">
+
+ <!-- Styles -->
+ <link rel="stylesheet" href="../dist/css/simple-media.css">
+ <link rel="stylesheet" href="../dist/css/docs.css">
+ </head>
+ <body>
+ <header>
+ <h1>Simple Media</h1>
+ <p>A simple HTML5 media player</p>
+ </header>
+
+ <section class="example-video">
+ <h2>Video</h2>
+ <p>Using the native &lt;video&gt; element.</p>
+ <div class="player">
+ <video poster="../media/poster_PayPal_Austin2.jpg" controls>
+ <!-- Video files -->
+ <source src="https://www.paypalobjects.com/webstatic/mktg/videos/PayPal_AustinSMB_baseline.mp4" type="video/mp4">
+ <source src="https://www.paypalobjects.com/webstatic/mktg/videos/PayPal_AustinSMB_baseline.webm" type="video/webm">
+
+ <!-- Text track file -->
+ <track kind="captions" label="English captions" src="../media/captions_PayPal_Austin_en.vtt" srclang="en" default>
+
+ <!-- Fallback for browsers that don't support the <video> element -->
+ <div>
+ <a href="https://www.paypalobjects.com/webstatic/mktg/videos/PayPal_AustinSMB_baseline.mp4">
+ <img src="../media/poster_PayPal_Austin2.jpg" width="640" height="360" alt="download video">
+ </a>
+ </div>
+ </video>
+ </div>
+ </section>
+
+ <section class="example-audio">
+ <h2>Audio</h2>
+ <p>Yep, it does &lt;audio&gt; too.</p>
+ <div class="player">
+ <audio controls>
+ <!-- Audio files -->
+ <source src="../media/Covox_-_Switchblade_Squadron.mp3" type="audio/mp3">
+
+ <!-- Fallback for browsers that don't support the <audio> element -->
+ <div>
+ <a href="../media/Covox_-_Switchblade_Squadron.mp3">Download it</a>
+ </div>
+ </audio>
+ </div>
+ </section>
+
+ <!-- Load SVG defs -->
+ <script>(function(d,p){var a=new XMLHttpRequest(),b=d.body; a.open("GET",p,!0);a.send();a.onload=function(){var c=d.createElement("div");c.style.display="none";c.innerHTML=a.responseText;b.insertBefore(c,b.childNodes[0])}})(document,"../dist/svg/sprite.svg");</script>
+
+ <!-- Core player -->
+ <script src="../dist/js/simple-media.js"></script>
+
+ <!-- Docs setup -->
+ <script src="../dist/js/docs.js"></script>
+ </body>
+</html> \ No newline at end of file
diff --git a/docs/video.html b/docs/video.html
deleted file mode 100644
index 9625f0e9..00000000
--- a/docs/video.html
+++ /dev/null
@@ -1,47 +0,0 @@
-<!doctype html>
-<html lang="en" class="example-video">
- <head>
- <meta charset="utf-8" />
- <title>Simple Media</title>
- <meta name="description" content="Custom HTML5 video controls and WebVTT captions.">
- <meta name="author" content="Sam Potts">
- <meta name="viewport" content="width=device-width, initial-scale=1">
-
- <!-- Styles -->
- <link rel="stylesheet" href="../dist/css/simple-media.css">
- <link rel="stylesheet" href="../dist/css/docs.css">
- </head>
- <body>
- <header>
- <h1>Simple Media</h1>
- <p>A simple HTML5 media player</p>
- </header>
-
- <div class="player">
- <video poster="../media/poster_PayPal_Austin2.jpg" controls>
- <!-- Video files -->
- <source src="https://www.paypalobjects.com/webstatic/mktg/videos/PayPal_AustinSMB_baseline.mp4" type="video/mp4">
- <source src="https://www.paypalobjects.com/webstatic/mktg/videos/PayPal_AustinSMB_baseline.webm" type="video/webm">
-
- <!-- Text track file -->
- <track kind="captions" label="English captions" src="../media/captions_PayPal_Austin_en.vtt" srclang="en" default>
-
- <!-- Fallback for browsers that don't support the <video> element -->
- <div>
- <a href="https://www.paypalobjects.com/webstatic/mktg/videos/PayPal_AustinSMB_baseline.mp4">
- <img src="../media/poster_PayPal_Austin2.jpg" width="640" height="360" alt="download video">
- </a>
- </div>
- </video>
- </div>
-
- <!-- Load SVG defs -->
- <script>(function(d,p){var a=new XMLHttpRequest(),b=d.body; a.open("GET",p,!0);a.send();a.onload=function(){var c=d.createElement("div");c.style.display="none";c.innerHTML=a.responseText;b.insertBefore(c,b.childNodes[0])}})(document,"../dist/svg/sprite.svg");</script>
-
- <!-- Core player -->
- <script src="../dist/js/simple-media.js"></script>
-
- <!-- Docs setup -->
- <script src="../dist/js/docs.js"></script>
- </body>
-</html> \ No newline at end of file