2090 lines
74 KiB
JavaScript
2090 lines
74 KiB
JavaScript
// ==========================================================================
|
|
// Plyr
|
|
// plyr.js v1.3.6
|
|
// https://github.com/selz/plyr
|
|
// License: The MIT License (MIT)
|
|
// ==========================================================================
|
|
// Credits: http://paypal.github.io/accessible-html5-video-player/
|
|
// ==========================================================================
|
|
|
|
(function (api) {
|
|
'use strict';
|
|
/*global YT*/
|
|
|
|
// Globals
|
|
var fullscreen, config, callbacks = {
|
|
youtube: []
|
|
};
|
|
|
|
// Default config
|
|
var defaults = {
|
|
enabled: true,
|
|
debug: false,
|
|
seekTime: 10,
|
|
volume: 5,
|
|
click: true,
|
|
tooltips: true,
|
|
displayDuration: true,
|
|
iconPrefix: 'icon',
|
|
selectors: {
|
|
container: '.player',
|
|
controls: '.player-controls',
|
|
labels: '[data-player] .sr-only, label .sr-only',
|
|
buttons: {
|
|
seek: '[data-player="seek"]',
|
|
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: {
|
|
container: '.player-progress',
|
|
buffer: '.player-progress-buffer',
|
|
played: '.player-progress-played'
|
|
},
|
|
captions: '.player-captions',
|
|
currentTime: '.player-current-time',
|
|
duration: '.player-duration'
|
|
},
|
|
classes: {
|
|
videoWrapper: 'player-video-wrapper',
|
|
embedWrapper: 'player-video-embed',
|
|
type: 'player-{0}',
|
|
stopped: 'stopped',
|
|
playing: 'playing',
|
|
muted: 'muted',
|
|
loading: 'loading',
|
|
tooltip: 'player-tooltip',
|
|
hidden: 'sr-only',
|
|
hover: 'player-hover',
|
|
captions: {
|
|
enabled: 'captions-enabled',
|
|
active: 'captions-active'
|
|
},
|
|
fullscreen: {
|
|
enabled: 'fullscreen-enabled',
|
|
active: 'fullscreen-active',
|
|
hideControls: 'fullscreen-hide-controls'
|
|
}
|
|
},
|
|
captions: {
|
|
defaultActive: false
|
|
},
|
|
fullscreen: {
|
|
enabled: true,
|
|
fallback: true,
|
|
hideControls: true
|
|
},
|
|
storage: {
|
|
enabled: true,
|
|
key: 'plyr_volume'
|
|
},
|
|
controls: ['restart', 'rewind', 'play', 'fast-forward', 'current-time', 'duration', 'mute', 'volume', /*'captions',*/ 'fullscreen'],
|
|
i18n: {
|
|
restart: '重新播放',
|
|
rewind: '后退{seektime}秒',
|
|
play: '播放',
|
|
pause: '暂停',
|
|
forward: '快进{seektime}秒',
|
|
played: '播放中',
|
|
buffered: '缓冲中',
|
|
currentTime: '当前时间',
|
|
duration: '持续时间',
|
|
volume: '音量',
|
|
toggleMute: '静音',
|
|
toggleCaptions: '字幕',
|
|
toggleFullscreen: '全屏'
|
|
}
|
|
};
|
|
|
|
// Build the default HTML
|
|
function _buildControls() {
|
|
// Open and add the progress and seek elements
|
|
var html = [
|
|
'<div class="player-controls">',
|
|
'<div class="player-progress">',
|
|
'<label for="seek{id}" class="sr-only">Seek</label>',
|
|
'<input id="seek{id}" class="player-progress-seek" type="range" min="0" max="100" step="0.5" value="0" data-player="seek">',
|
|
'<progress class="player-progress-played" max="100" value="0">',
|
|
'<span>0</span>% ' + config.i18n.played,
|
|
'</progress>',
|
|
'<progress class="player-progress-buffer" max="100" value="0">',
|
|
'<span>0</span>% ' + config.i18n.buffered,
|
|
'</progress>',
|
|
'</div>',
|
|
'<span class="player-controls-left">'];
|
|
|
|
// Restart button
|
|
if (_inArray(config.controls, 'restart')) {
|
|
html.push(
|
|
'<button type="button" data-player="restart">',
|
|
'<svg><use xlink:href="#' + config.iconPrefix + '-restart" /></svg>',
|
|
'<span class="sr-only">' + config.i18n.restart + '</span>',
|
|
'</button>'
|
|
);
|
|
}
|
|
|
|
// Rewind button
|
|
if (_inArray(config.controls, 'rewind')) {
|
|
html.push(
|
|
'<button type="button" data-player="rewind">',
|
|
'<svg><use xlink:href="#' + config.iconPrefix + '-rewind" /></svg>',
|
|
'<span class="sr-only">' + config.i18n.rewind + '</span>',
|
|
'</button>'
|
|
);
|
|
}
|
|
|
|
// Play/pause button
|
|
if (_inArray(config.controls, 'play')) {
|
|
html.push(
|
|
'<button type="button" data-player="play">',
|
|
'<svg><use xlink:href="#' + config.iconPrefix + '-play" /></svg>',
|
|
'<span class="sr-only">' + config.i18n.play + '</span>',
|
|
'</button>',
|
|
'<button type="button" data-player="pause">',
|
|
'<svg><use xlink:href="#' + config.iconPrefix + '-pause" /></svg>',
|
|
'<span class="sr-only">' + config.i18n.pause + '</span>',
|
|
'</button>'
|
|
);
|
|
}
|
|
|
|
// Fast forward button
|
|
if (_inArray(config.controls, 'fast-forward')) {
|
|
html.push(
|
|
'<button type="button" data-player="fast-forward">',
|
|
'<svg><use xlink:href="#' + config.iconPrefix + '-fast-forward" /></svg>',
|
|
'<span class="sr-only">' + config.i18n.forward + '</span>',
|
|
'</button>'
|
|
);
|
|
}
|
|
|
|
// Media current time display
|
|
if (_inArray(config.controls, 'current-time')) {
|
|
html.push(
|
|
'<span class="player-time">',
|
|
'<span class="sr-only">' + config.i18n.currentTime + '</span>',
|
|
'<span class="player-current-time">00:00</span>',
|
|
'</span>'
|
|
);
|
|
}
|
|
|
|
// Media duration display
|
|
if (_inArray(config.controls, 'duration')) {
|
|
html.push(
|
|
'<span class="player-time">',
|
|
'<span class="sr-only">' + config.i18n.duration + '</span>',
|
|
'<span class="player-duration">00:00</span>',
|
|
'</span>'
|
|
);
|
|
}
|
|
|
|
// Close left controls
|
|
html.push(
|
|
'</span>',
|
|
'<span class="player-controls-right">'
|
|
);
|
|
|
|
// Toggle mute button
|
|
if (_inArray(config.controls, 'mute')) {
|
|
html.push(
|
|
'<button type="button" data-player="mute">',
|
|
'<svg class="icon-muted"><use xlink:href="#' + config.iconPrefix + '-muted" /></svg>',
|
|
'<svg><use xlink:href="#' + config.iconPrefix + '-volume" /></svg>',
|
|
'<span class="sr-only">' + config.i18n.toggleMute + '</span>',
|
|
'</button>'
|
|
);
|
|
}
|
|
|
|
// Volume range control
|
|
if (_inArray(config.controls, 'volume')) {
|
|
html.push(
|
|
'<label for="volume{id}" class="sr-only">' + config.i18n.volume + '</label>',
|
|
'<input id="volume{id}" class="player-volume" type="range" min="0" max="10" value="5" data-player="volume">'
|
|
);
|
|
}
|
|
|
|
// Toggle captions button
|
|
if (_inArray(config.controls, 'captions')) {
|
|
html.push(
|
|
'<button type="button" data-player="captions">',
|
|
'<svg class="icon-captions-on"><use xlink:href="#' + config.iconPrefix + '-captions-on" /></svg>',
|
|
'<svg><use xlink:href="#' + config.iconPrefix + '-captions-off" /></svg>',
|
|
'<span class="sr-only">' + config.i18n.toggleCaptions + '</span>',
|
|
'</button>'
|
|
);
|
|
}
|
|
|
|
// Toggle fullscreen button
|
|
if (_inArray(config.controls, 'fullscreen')) {
|
|
html.push(
|
|
'<button type="button" data-player="fullscreen">',
|
|
'<svg class="icon-exit-fullscreen"><use xlink:href="#' + config.iconPrefix + '-exit-fullscreen" /></svg>',
|
|
'<svg><use xlink:href="#' + config.iconPrefix + '-enter-fullscreen" /></svg>',
|
|
'<span class="sr-only">' + config.i18n.toggleFullscreen + '</span>',
|
|
'</button>'
|
|
);
|
|
}
|
|
|
|
// Close everything
|
|
html.push(
|
|
'</span>',
|
|
'</div>'
|
|
);
|
|
|
|
return html.join('');
|
|
}
|
|
|
|
// Debugging
|
|
function _log(text, error) {
|
|
if (config.debug && window.console) {
|
|
console[(error ? 'error' : 'log')](text);
|
|
}
|
|
}
|
|
|
|
// Credits: http://paypal.github.io/accessible-html5-video-player/
|
|
// Unfortunately, due to mixed support, UA sniffing is required
|
|
function _browserSniff() {
|
|
var nAgt = navigator.userAgent,
|
|
name = navigator.appName,
|
|
fullVersion = '' + parseFloat(navigator.appVersion),
|
|
majorVersion = parseInt(navigator.appVersion, 10),
|
|
nameOffset,
|
|
verOffset,
|
|
ix;
|
|
|
|
// MSIE 11
|
|
if ((navigator.appVersion.indexOf('Windows NT') !== -1) && (navigator.appVersion.indexOf('rv:11') !== -1)) {
|
|
name = 'IE';
|
|
fullVersion = '11;';
|
|
}
|
|
// MSIE
|
|
else if ((verOffset = nAgt.indexOf('MSIE')) !== -1) {
|
|
name = 'IE';
|
|
fullVersion = nAgt.substring(verOffset + 5);
|
|
}
|
|
// Chrome
|
|
else if ((verOffset = nAgt.indexOf('Chrome')) !== -1) {
|
|
name = 'Chrome';
|
|
fullVersion = nAgt.substring(verOffset + 7);
|
|
}
|
|
// Safari
|
|
else if ((verOffset = nAgt.indexOf('Safari')) !== -1) {
|
|
name = 'Safari';
|
|
fullVersion = nAgt.substring(verOffset + 7);
|
|
if ((verOffset = nAgt.indexOf('Version')) !== -1) {
|
|
fullVersion = nAgt.substring(verOffset + 8);
|
|
}
|
|
}
|
|
// Firefox
|
|
else if ((verOffset = nAgt.indexOf('Firefox')) !== -1) {
|
|
name = 'Firefox';
|
|
fullVersion = nAgt.substring(verOffset + 8);
|
|
}
|
|
// In most other browsers, 'name/version' is at the end of userAgent
|
|
else if ((nameOffset = nAgt.lastIndexOf(' ') + 1) < (verOffset = nAgt.lastIndexOf('/'))) {
|
|
name = nAgt.substring(nameOffset, verOffset);
|
|
fullVersion = nAgt.substring(verOffset + 1);
|
|
|
|
if (name.toLowerCase() == name.toUpperCase()) {
|
|
name = navigator.appName;
|
|
}
|
|
}
|
|
// Trim the fullVersion string at semicolon/space if present
|
|
if ((ix = fullVersion.indexOf(';')) !== -1) {
|
|
fullVersion = fullVersion.substring(0, ix);
|
|
}
|
|
if ((ix = fullVersion.indexOf(' ')) !== -1) {
|
|
fullVersion = fullVersion.substring(0, ix);
|
|
}
|
|
// Get major version
|
|
majorVersion = parseInt('' + fullVersion, 10);
|
|
if (isNaN(majorVersion)) {
|
|
fullVersion = '' + parseFloat(navigator.appVersion);
|
|
majorVersion = parseInt(navigator.appVersion, 10);
|
|
}
|
|
|
|
// Return data
|
|
return {
|
|
name: name,
|
|
version: majorVersion,
|
|
ios: /(iPad|iPhone|iPod)/g.test(navigator.platform)
|
|
};
|
|
}
|
|
|
|
// Check for mime type support against a player instance
|
|
// Credits: http://diveintohtml5.info/everything.html
|
|
// Related: http://www.leanbackplayer.com/test/h5mt.html
|
|
function _supportMime(player, mimeType) {
|
|
var media = player.media;
|
|
|
|
// Only check video types for video players
|
|
if (player.type == 'video') {
|
|
// Check type
|
|
switch (mimeType) {
|
|
case 'video/webm':
|
|
return !!(media.canPlayType && media.canPlayType('video/webm; codecs="vp8, vorbis"').replace(/no/, ''));
|
|
case 'video/mp4':
|
|
return !!(media.canPlayType && media.canPlayType('video/mp4; codecs="avc1.42E01E, mp4a.40.2"').replace(/no/, ''));
|
|
case 'video/ogg':
|
|
return !!(media.canPlayType && media.canPlayType('video/ogg; codecs="theora"').replace(/no/, ''));
|
|
}
|
|
}
|
|
|
|
// Only check audio types for audio players
|
|
else if (player.type == 'audio') {
|
|
// Check type
|
|
switch (mimeType) {
|
|
case 'audio/mpeg':
|
|
return !!(media.canPlayType && media.canPlayType('audio/mpeg;').replace(/no/, ''));
|
|
case 'audio/ogg':
|
|
return !!(media.canPlayType && media.canPlayType('audio/ogg; codecs="vorbis"').replace(/no/, ''));
|
|
case 'audio/wav':
|
|
return !!(media.canPlayType && media.canPlayType('audio/wav; codecs="1"').replace(/no/, ''));
|
|
}
|
|
}
|
|
|
|
// If we got this far, we're stuffed
|
|
return false;
|
|
}
|
|
|
|
// Inject a script
|
|
function _injectScript(source) {
|
|
if (document.querySelectorAll('script[src="' + source + '"]').length) {
|
|
return;
|
|
}
|
|
|
|
var tag = document.createElement('script');
|
|
tag.src = source;
|
|
var firstScriptTag = document.getElementsByTagName('script')[0];
|
|
firstScriptTag.parentNode.insertBefore(tag, firstScriptTag);
|
|
}
|
|
|
|
// Element exists in an array
|
|
function _inArray(haystack, needle) {
|
|
return Array.prototype.indexOf && (haystack.indexOf(needle) != -1);
|
|
}
|
|
|
|
// Replace all
|
|
function _replaceAll(string, find, replace) {
|
|
return string.replace(new RegExp(find.replace(/([.*+?\^=!:${}()|\[\]\/\\])/g, '\\$1'), 'g'), replace);
|
|
}
|
|
|
|
// Wrap an element
|
|
function _wrap(elements, wrapper) {
|
|
// Convert `elements` to an array, if necessary.
|
|
if (!elements.length) {
|
|
elements = [elements];
|
|
}
|
|
|
|
// Loops backwards to prevent having to clone the wrapper on the
|
|
// first element (see `child` below).
|
|
for (var i = elements.length - 1; i >= 0; i--) {
|
|
var child = (i > 0) ? wrapper.cloneNode(true) : wrapper;
|
|
var element = elements[i];
|
|
|
|
// Cache the current parent and sibling.
|
|
var parent = element.parentNode;
|
|
var sibling = element.nextSibling;
|
|
|
|
// Wrap the element (is automatically removed from its current
|
|
// parent).
|
|
child.appendChild(element);
|
|
|
|
// If the element had a sibling, insert the wrapper before
|
|
// the sibling to maintain the HTML structure; otherwise, just
|
|
// append it to the parent.
|
|
if (sibling) {
|
|
parent.insertBefore(child, sibling);
|
|
} else {
|
|
parent.appendChild(child);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Unwrap an element
|
|
// http://plainjs.com/javascript/manipulation/unwrap-a-dom-element-35/
|
|
function _unwrap(wrapper) {
|
|
// Get the element's parent node
|
|
var parent = wrapper.parentNode;
|
|
|
|
// Move all children out of the element
|
|
while (wrapper.firstChild) {
|
|
parent.insertBefore(wrapper.firstChild, wrapper);
|
|
}
|
|
|
|
// Remove the empty element
|
|
parent.removeChild(wrapper);
|
|
}
|
|
|
|
// Remove an element
|
|
function _remove(element) {
|
|
element.parentNode.removeChild(element);
|
|
}
|
|
|
|
// Prepend child
|
|
function _prependChild(parent, element) {
|
|
parent.insertBefore(element, parent.firstChild);
|
|
}
|
|
|
|
// Set attributes
|
|
function _setAttributes(element, attributes) {
|
|
for (var key in attributes) {
|
|
element.setAttribute(key, attributes[key]);
|
|
}
|
|
}
|
|
|
|
// 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 : '');
|
|
}
|
|
}
|
|
}
|
|
|
|
// Toggle event
|
|
function _toggleHandler(element, events, callback, toggle) {
|
|
var eventList = events.split(' ');
|
|
|
|
// If a nodelist is passed, call itself on each node
|
|
if (element instanceof NodeList) {
|
|
for (var x = 0; x < element.length; x++) {
|
|
if (element[x] instanceof Node) {
|
|
_toggleHandler(element[x], arguments[1], arguments[2], arguments[3]);
|
|
}
|
|
}
|
|
return;
|
|
}
|
|
|
|
// If a single node is passed, bind the event listener
|
|
for (var i = 0; i < eventList.length; i++) {
|
|
element[toggle ? 'addEventListener' : 'removeEventListener'](eventList[i], callback, false);
|
|
}
|
|
}
|
|
|
|
// Bind event
|
|
function _on(element, events, callback) {
|
|
if (element) {
|
|
_toggleHandler(element, events, callback, true);
|
|
}
|
|
}
|
|
|
|
// Unbind event
|
|
function _off(element, events, callback) {
|
|
if (element) {
|
|
_toggleHandler(element, events, callback, false);
|
|
}
|
|
}
|
|
|
|
// Trigger event
|
|
function _triggerEvent(element, event) {
|
|
// Create faux event
|
|
var fauxEvent = document.createEvent('MouseEvents');
|
|
|
|
// Set the event type
|
|
fauxEvent.initEvent(event, true, true);
|
|
|
|
// Dispatch the event
|
|
element.dispatchEvent(fauxEvent);
|
|
}
|
|
|
|
// Toggle aria-pressed state on a toggle button
|
|
function _toggleState(target, state) {
|
|
// Get state
|
|
state = (typeof state === 'boolean' ? state : !target.getAttribute('aria-pressed'));
|
|
|
|
// Set the attribute on target
|
|
target.setAttribute('aria-pressed', state);
|
|
|
|
return state;
|
|
}
|
|
|
|
// Get percentage
|
|
function _getPercentage(current, max) {
|
|
if (current === 0 || max === 0 || isNaN(current) || isNaN(max)) {
|
|
return 0;
|
|
}
|
|
return ((current / max) * 100).toFixed(2);
|
|
}
|
|
|
|
// Deep extend/merge two Objects
|
|
// http://andrewdupont.net/2009/08/28/deep-extending-objects-in-javascript/
|
|
// Removed call to arguments.callee (used explicit function name instead)
|
|
function _extend(destination, source) {
|
|
for (var property in source) {
|
|
if (source[property] && source[property].constructor && source[property].constructor === Object) {
|
|
destination[property] = destination[property] || {};
|
|
_extend(destination[property], source[property]);
|
|
} else {
|
|
destination[property] = source[property];
|
|
}
|
|
}
|
|
return destination;
|
|
}
|
|
|
|
// Fullscreen API
|
|
function _fullscreen() {
|
|
var fullscreen = {
|
|
supportsFullScreen: false,
|
|
isFullScreen: function () {
|
|
return false;
|
|
},
|
|
requestFullScreen: function () {},
|
|
cancelFullScreen: function () {},
|
|
fullScreenEventName: '',
|
|
element: null,
|
|
prefix: ''
|
|
},
|
|
browserPrefixes = 'webkit moz o ms khtml'.split(' ');
|
|
|
|
// Check for native support
|
|
if (typeof document.cancelFullScreen !== 'undefined') {
|
|
fullscreen.supportsFullScreen = true;
|
|
} else {
|
|
// Check for fullscreen support by vendor prefix
|
|
for (var i = 0, il = browserPrefixes.length; i < il; i++) {
|
|
fullscreen.prefix = browserPrefixes[i];
|
|
|
|
if (typeof document[fullscreen.prefix + 'CancelFullScreen'] !== 'undefined') {
|
|
fullscreen.supportsFullScreen = true;
|
|
break;
|
|
}
|
|
// Special case for MS (when isn't it?)
|
|
else if (typeof document.msExitFullscreen !== 'undefined' && document.msFullscreenEnabled) {
|
|
fullscreen.prefix = 'ms';
|
|
fullscreen.supportsFullScreen = true;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Update methods to do something useful
|
|
if (fullscreen.supportsFullScreen) {
|
|
// Yet again Microsoft awesomeness,
|
|
// Sometimes the prefix is 'ms', sometimes 'MS' to keep you on your toes
|
|
fullscreen.fullScreenEventName = (fullscreen.prefix == 'ms' ? 'MSFullscreenChange' : fullscreen.prefix + 'fullscreenchange');
|
|
|
|
fullscreen.isFullScreen = function (element) {
|
|
if (typeof element === 'undefined') {
|
|
element = document.body;
|
|
}
|
|
switch (this.prefix) {
|
|
case '':
|
|
return document.fullscreenElement == element;
|
|
case 'moz':
|
|
return document.mozFullScreenElement == element;
|
|
default:
|
|
return document[this.prefix + 'FullscreenElement'] == element;
|
|
}
|
|
};
|
|
fullscreen.requestFullScreen = function (element) {
|
|
if (typeof element === 'undefined') {
|
|
element = document.body;
|
|
}
|
|
return (this.prefix === '') ? element.requestFullScreen() : element[this.prefix + (this.prefix == 'ms' ? 'RequestFullscreen' : 'RequestFullScreen')]();
|
|
};
|
|
fullscreen.cancelFullScreen = function () {
|
|
return (this.prefix === '') ? document.cancelFullScreen() : document[this.prefix + (this.prefix == 'ms' ? 'ExitFullscreen' : 'CancelFullScreen')]();
|
|
};
|
|
fullscreen.element = function () {
|
|
return (this.prefix === '') ? document.fullscreenElement : document[this.prefix + 'FullscreenElement'];
|
|
};
|
|
}
|
|
|
|
return fullscreen;
|
|
}
|
|
|
|
// Local storage
|
|
function _storage() {
|
|
var storage = {
|
|
supported: (function () {
|
|
try {
|
|
return 'localStorage' in window && window.localStorage !== null;
|
|
} catch (e) {
|
|
return false;
|
|
}
|
|
})()
|
|
};
|
|
return storage;
|
|
}
|
|
|
|
// Player instance
|
|
function Plyr(container) {
|
|
var player = this;
|
|
player.container = container;
|
|
|
|
// Captions functions
|
|
// Seek the manual caption time and update UI
|
|
function _seekManualCaptions(time) {
|
|
// If it's not video, or we're using textTracks, bail.
|
|
if (player.usingTextTracks || player.type !== 'video' || !player.supported.full) {
|
|
return;
|
|
}
|
|
|
|
// Reset subcount
|
|
player.subcount = 0;
|
|
|
|
// Check time is a number, if not use currentTime
|
|
// IE has a bug where currentTime doesn't go to 0
|
|
// https://twitter.com/Sam_Potts/status/573715746506731521
|
|
time = typeof time === 'number' ? time : player.media.currentTime;
|
|
|
|
while (_timecodeMax(player.captions[player.subcount][0]) < time.toFixed(1)) {
|
|
player.subcount++;
|
|
if (player.subcount > player.captions.length - 1) {
|
|
player.subcount = player.captions.length - 1;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// 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];
|
|
|
|
// Trim caption text
|
|
var content = player.currentCaption.trim();
|
|
|
|
// Render the caption (only if changed)
|
|
if (player.captionsContainer.innerHTML != content) {
|
|
// Empty caption
|
|
// Otherwise NVDA reads it twice
|
|
player.captionsContainer.innerHTML = '';
|
|
|
|
// Set new caption text
|
|
player.captionsContainer.innerHTML = content;
|
|
}
|
|
} else {
|
|
player.captionsContainer.innerHTML = '';
|
|
}
|
|
}
|
|
|
|
// Display captions container and button (for initialization)
|
|
function _showCaptions() {
|
|
// If there's no caption toggle, bail
|
|
if (!player.buttons.captions) {
|
|
return;
|
|
}
|
|
|
|
_toggleClass(player.container, config.classes.captions.enabled, true);
|
|
|
|
if (config.captions.defaultActive) {
|
|
_toggleClass(player.container, config.classes.captions.active, true);
|
|
_toggleState(player.buttons.captions, true);
|
|
}
|
|
}
|
|
|
|
// 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 a single element
|
|
function _getElement(selector) {
|
|
return _getElements(selector)[0];
|
|
}
|
|
|
|
// Determine if we're in an iframe
|
|
function _inFrame() {
|
|
try {
|
|
return window.self !== window.top;
|
|
} catch (e) {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
// Insert controls
|
|
function _injectControls() {
|
|
// Make a copy of the html
|
|
var html = config.html;
|
|
|
|
// Insert custom video controls
|
|
_log('Injecting custom controls.');
|
|
|
|
// If no controls are specified, create default
|
|
if (!html) {
|
|
html = _buildControls();
|
|
}
|
|
|
|
// Replace seek time instances
|
|
html = _replaceAll(html, '{seektime}', config.seekTime);
|
|
|
|
// Replace all id references with random numbers
|
|
html = _replaceAll(html, '{id}', Math.floor(Math.random() * (10000)));
|
|
|
|
// Inject into the container
|
|
player.container.insertAdjacentHTML('beforeend', html);
|
|
|
|
// Setup tooltips
|
|
if (config.tooltips) {
|
|
var labels = _getElements(config.selectors.labels);
|
|
|
|
for (var i = labels.length - 1; i >= 0; i--) {
|
|
var label = labels[i];
|
|
|
|
_toggleClass(label, config.classes.hidden, false);
|
|
_toggleClass(label, config.classes.tooltip, true);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Find the UI controls and store references
|
|
function _findElements() {
|
|
try {
|
|
player.controls = _getElement(config.selectors.controls);
|
|
|
|
// Buttons
|
|
player.buttons = {};
|
|
player.buttons.seek = _getElement(config.selectors.buttons.seek);
|
|
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.fullscreen = _getElement(config.selectors.buttons.fullscreen);
|
|
|
|
// Inputs
|
|
player.buttons.mute = _getElement(config.selectors.buttons.mute);
|
|
player.buttons.captions = _getElement(config.selectors.buttons.captions);
|
|
player.checkboxes = _getElements('[type="checkbox"]');
|
|
|
|
// Progress
|
|
player.progress = {};
|
|
player.progress.container = _getElement(config.selectors.progress.container);
|
|
|
|
// Progress - Buffering
|
|
player.progress.buffer = {};
|
|
player.progress.buffer.bar = _getElement(config.selectors.progress.buffer);
|
|
player.progress.buffer.text = player.progress.buffer.bar && player.progress.buffer.bar.getElementsByTagName('span')[0];
|
|
|
|
// Progress - Played
|
|
player.progress.played = {};
|
|
player.progress.played.bar = _getElement(config.selectors.progress.played);
|
|
player.progress.played.text = player.progress.played.bar && player.progress.played.bar.getElementsByTagName('span')[0];
|
|
|
|
// Volume
|
|
player.volume = _getElement(config.selectors.buttons.volume);
|
|
|
|
// Timing
|
|
player.duration = _getElement(config.selectors.duration);
|
|
player.currentTime = _getElement(config.selectors.currentTime);
|
|
player.seekTime = _getElements(config.selectors.seekTime);
|
|
|
|
return true;
|
|
} catch (e) {
|
|
_log('It looks like there\'s a problem with your controls html. Bailing.', true);
|
|
|
|
// Restore native video controls
|
|
player.media.setAttribute('controls', '');
|
|
|
|
return false;
|
|
}
|
|
}
|
|
|
|
// Setup aria attribute for play
|
|
function _setupPlayAria() {
|
|
// If there's no play button, bail
|
|
if (!player.buttons.play) {
|
|
return;
|
|
}
|
|
|
|
// Find the current text
|
|
var label = player.buttons.play.innerText || config.i18n.play;
|
|
|
|
// If there's a media title set, use that for the label
|
|
if (typeof (config.title) !== 'undefined' && config.title.length) {
|
|
label += ', ' + config.title;
|
|
}
|
|
|
|
player.buttons.play.setAttribute('aria-label', label);
|
|
}
|
|
|
|
// Setup media
|
|
function _setupMedia() {
|
|
// If there's no media, bail
|
|
if (!player.media) {
|
|
_log('No audio or video element found!', true);
|
|
return false;
|
|
}
|
|
|
|
if (player.supported.full) {
|
|
// Remove native video controls
|
|
player.media.removeAttribute('controls');
|
|
|
|
// Add type class
|
|
_toggleClass(player.container, config.classes.type.replace('{0}', player.type), true);
|
|
|
|
// If there's no autoplay attribute, assume the video is stopped and add state class
|
|
_toggleClass(player.container, config.classes.stopped, (player.media.getAttribute('autoplay') === null));
|
|
|
|
// Add iOS class
|
|
if (player.browser.ios) {
|
|
_toggleClass(player.container, 'ios', true);
|
|
}
|
|
|
|
// Inject the player wrapper
|
|
if (player.type === 'video') {
|
|
// Create the wrapper div
|
|
var wrapper = document.createElement('div');
|
|
wrapper.setAttribute('class', config.classes.videoWrapper);
|
|
|
|
// Wrap the video in a container
|
|
_wrap(player.media, wrapper);
|
|
|
|
// Cache the container
|
|
player.videoContainer = wrapper;
|
|
}
|
|
}
|
|
|
|
// YouTube
|
|
if (player.type == 'youtube') {
|
|
_setupYouTube(player.media.getAttribute('data-video-id'));
|
|
}
|
|
|
|
// Autoplay
|
|
if (player.media.getAttribute('autoplay') !== null) {
|
|
_play();
|
|
}
|
|
}
|
|
|
|
// Setup YouTube
|
|
function _setupYouTube(id) {
|
|
// Remove old containers
|
|
var containers = _getElements('[id^="youtube"]');
|
|
for (var i = containers.length - 1; i >= 0; i--) {
|
|
_remove(containers[i]);
|
|
}
|
|
|
|
// Create the YouTube container
|
|
var container = document.createElement('div');
|
|
container.setAttribute('id', 'youtube-' + Math.floor(Math.random() * (10000)));
|
|
player.media.appendChild(container);
|
|
|
|
// Add embed class for responsive
|
|
_toggleClass(player.media, config.classes.videoWrapper, true);
|
|
_toggleClass(player.media, config.classes.embedWrapper, true);
|
|
|
|
if (typeof YT === 'object') {
|
|
_YTReady(id, container);
|
|
} else {
|
|
// Load the API
|
|
_injectScript('https://www.youtube.com/iframe_api');
|
|
|
|
// Add callback to queue
|
|
callbacks.youtube.push(function () {
|
|
_YTReady(id, container);
|
|
});
|
|
|
|
// Setup callback for the API
|
|
window.onYouTubeIframeAPIReady = function () {
|
|
for (var i = callbacks.youtube.length - 1; i >= 0; i--) {
|
|
// Fire callback
|
|
callbacks.youtube[i]();
|
|
|
|
// Remove from queue
|
|
callbacks.youtube.splice(i, 1);
|
|
}
|
|
};
|
|
}
|
|
}
|
|
|
|
// Handle API ready
|
|
function _YTReady(id, container) {
|
|
_log('YouTube API Ready');
|
|
|
|
// Setup timers object
|
|
// We have to poll YouTube for updates
|
|
if (!('timer' in player)) {
|
|
player.timer = {};
|
|
}
|
|
|
|
// Setup instance
|
|
// https://developers.google.com/youtube/iframe_api_reference
|
|
player.embed = new YT.Player(container.id, {
|
|
videoId: id,
|
|
playerVars: {
|
|
autoplay: 0,
|
|
controls: (player.supported.full ? 0 : 1),
|
|
rel: 0,
|
|
showinfo: 0,
|
|
iv_load_policy: 3,
|
|
cc_load_policy: (config.captions.defaultActive ? 1 : 0),
|
|
cc_lang_pref: 'en',
|
|
wmode: 'transparent',
|
|
modestbranding: 1,
|
|
disablekb: 1
|
|
},
|
|
events: {
|
|
'onReady': function (event) {
|
|
// Get the instance
|
|
var instance = event.target;
|
|
|
|
// Create a faux HTML5 API using the YouTube API
|
|
player.media.play = function () {
|
|
instance.playVideo();
|
|
};
|
|
player.media.pause = function () {
|
|
instance.pauseVideo();
|
|
};
|
|
player.media.stop = function () {
|
|
instance.stopVideo();
|
|
};
|
|
player.media.duration = instance.getDuration();
|
|
player.media.paused = true;
|
|
player.media.currentTime = instance.getCurrentTime();
|
|
player.media.muted = instance.isMuted();
|
|
|
|
// Trigger timeupdate
|
|
_triggerEvent(player.media, 'timeupdate');
|
|
|
|
// Reset timer
|
|
window.clearInterval(player.timer.buffering);
|
|
|
|
// Setup buffering
|
|
player.timer.buffering = window.setInterval(function () {
|
|
// Get loaded % from YouTube
|
|
player.media.buffered = instance.getVideoLoadedFraction();
|
|
|
|
// Trigger progress
|
|
_triggerEvent(player.media, 'progress');
|
|
|
|
// Bail if we're at 100%
|
|
if (player.media.buffered === 1) {
|
|
window.clearInterval(player.timer.buffering);
|
|
}
|
|
}, 200);
|
|
|
|
if (player.supported.full) {
|
|
// Only setup controls once
|
|
if (!player.container.querySelectorAll(config.selectors.controls).length) {
|
|
_setupInterface();
|
|
}
|
|
|
|
// Display duration if available
|
|
if (config.displayDuration) {
|
|
_displayDuration();
|
|
}
|
|
}
|
|
},
|
|
'onStateChange': function (event) {
|
|
// Get the instance
|
|
var instance = event.target;
|
|
|
|
// Reset timer
|
|
window.clearInterval(player.timer.playing);
|
|
|
|
// Handle events
|
|
// -1 Unstarted
|
|
// 0 Ended
|
|
// 1 Playing
|
|
// 2 Paused
|
|
// 3 Buffering
|
|
// 5 Video cued
|
|
switch (event.data) {
|
|
case 0:
|
|
player.media.paused = true;
|
|
_triggerEvent(player.media, 'ended');
|
|
break;
|
|
|
|
case 1:
|
|
player.media.paused = false;
|
|
_triggerEvent(player.media, 'play');
|
|
|
|
// Poll to get playback progress
|
|
player.timer.playing = window.setInterval(function () {
|
|
// Set the current time
|
|
player.media.currentTime = instance.getCurrentTime();
|
|
|
|
// Trigger timeupdate
|
|
_triggerEvent(player.media, 'timeupdate');
|
|
}, 200);
|
|
|
|
break;
|
|
|
|
case 2:
|
|
player.media.paused = true;
|
|
_triggerEvent(player.media, 'pause');
|
|
}
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
// Setup captions
|
|
function _setupCaptions() {
|
|
if (player.type === 'video') {
|
|
// Inject the container
|
|
player.videoContainer.insertAdjacentHTML('afterbegin', '<div class="' + config.selectors.captions.replace('.', '') + '"><span></span></div>');
|
|
|
|
// Cache selector
|
|
player.captionsContainer = _getElement(config.selectors.captions).querySelector('span');
|
|
|
|
// Determine if HTML5 textTracks is supported
|
|
player.usingTextTracks = false;
|
|
if (player.media.textTracks) {
|
|
player.usingTextTracks = true;
|
|
}
|
|
|
|
// Get URL of caption file if exists
|
|
var captionSrc = '',
|
|
kind,
|
|
children = player.media.childNodes;
|
|
|
|
for (var i = 0; i < children.length; i++) {
|
|
if (children[i].nodeName.toLowerCase() === 'track') {
|
|
kind = children[i].kind;
|
|
if (kind === 'captions' || kind === 'subtitles') {
|
|
captionSrc = children[i].getAttribute('src');
|
|
}
|
|
}
|
|
}
|
|
|
|
// Record if caption file exists or not
|
|
player.captionExists = true;
|
|
if (captionSrc === '') {
|
|
player.captionExists = false;
|
|
_log('No caption track found.');
|
|
} else {
|
|
_log('Caption track found; URI: ' + captionSrc);
|
|
}
|
|
|
|
// 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 {
|
|
// Turn off native caption rendering to avoid double captions
|
|
// This doesn't seem to work in Safari 7+, so the <track> elements are removed from the dom below
|
|
var tracks = player.media.textTracks;
|
|
for (var x = 0; x < tracks.length; x++) {
|
|
tracks[x].mode = 'hidden';
|
|
}
|
|
|
|
// Enable UI
|
|
_showCaptions(player);
|
|
|
|
// Disable unsupported browsers than report false positive
|
|
if ((player.browser.name === 'IE' && player.browser.version >= 10) ||
|
|
(player.browser.name === 'Firefox' && player.browser.version >= 31) ||
|
|
(player.browser.name === 'Chrome' && player.browser.version >= 43) ||
|
|
(player.browser.name === 'Safari' && player.browser.version >= 7)) {
|
|
// Debugging
|
|
_log('Detected unsupported browser for HTML5 captions. Using fallback.');
|
|
|
|
// Set to false so skips to 'manual' captioning
|
|
player.usingTextTracks = false;
|
|
}
|
|
|
|
// Rendering caption tracks
|
|
// Native support required - http://caniuse.com/webvtt
|
|
if (player.usingTextTracks) {
|
|
_log('TextTracks supported.');
|
|
|
|
for (var y = 0; y < tracks.length; y++) {
|
|
var track = tracks[y];
|
|
|
|
if (track.kind === 'captions' || track.kind === 'subtitles') {
|
|
_on(track, 'cuechange', function () {
|
|
// Clear container
|
|
player.captionsContainer.innerHTML = '';
|
|
|
|
// Display a cue, if there is one
|
|
if (this.activeCues[0] && this.activeCues[0].hasOwnProperty('text')) {
|
|
player.captionsContainer.appendChild(this.activeCues[0].getCueAsHTML().trim());
|
|
}
|
|
});
|
|
}
|
|
}
|
|
}
|
|
// Caption tracks not natively supported
|
|
else {
|
|
_log('TextTracks not supported so rendering captions manually.');
|
|
|
|
// Render captions from array at appropriate time
|
|
player.currentCaption = '';
|
|
player.captions = [];
|
|
|
|
if (captionSrc !== '') {
|
|
// Create XMLHttpRequest Object
|
|
var xhr = new XMLHttpRequest();
|
|
|
|
xhr.onreadystatechange = function () {
|
|
if (xhr.readyState === 4) {
|
|
if (xhr.status === 200) {
|
|
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();
|
|
|
|
_log('Successfully loaded the caption file via AJAX.');
|
|
} else {
|
|
_log('There was a problem loading the caption file via AJAX.', true);
|
|
}
|
|
}
|
|
};
|
|
|
|
xhr.open('get', captionSrc, true);
|
|
|
|
xhr.send();
|
|
}
|
|
}
|
|
|
|
// If Safari 7+, removing track from DOM [see 'turn off native caption rendering' above]
|
|
if (player.browser.name === 'Safari' && player.browser.version >= 7) {
|
|
_log('Safari 7+ detected; removing track from DOM.');
|
|
|
|
// Find all <track> elements
|
|
tracks = player.media.getElementsByTagName('track');
|
|
|
|
// Loop through and remove one by one
|
|
for (var t = 0; t < tracks.length; t++) {
|
|
player.media.removeChild(tracks[t]);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Setup fullscreen
|
|
function _setupFullscreen() {
|
|
if (player.type != 'audio' && config.fullscreen.enabled) {
|
|
// Check for native support
|
|
var nativeSupport = fullscreen.supportsFullScreen;
|
|
|
|
if (nativeSupport || (config.fullscreen.fallback && !_inFrame())) {
|
|
_log((nativeSupport ? 'Native' : 'Fallback') + ' fullscreen enabled.');
|
|
|
|
// Add styling hook
|
|
_toggleClass(player.container, config.classes.fullscreen.enabled, true);
|
|
} else {
|
|
_log('Fullscreen not supported and fallback disabled.');
|
|
}
|
|
|
|
// Toggle state
|
|
_toggleState(player.buttons.fullscreen, false);
|
|
|
|
// Set control hide class hook
|
|
if (config.fullscreen.hideControls) {
|
|
_toggleClass(player.container, config.classes.fullscreen.hideControls, true);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Play media
|
|
function _play() {
|
|
player.media.play();
|
|
}
|
|
|
|
// Pause media
|
|
function _pause() {
|
|
player.media.pause();
|
|
}
|
|
|
|
// Toggle playback
|
|
function _togglePlay(toggle) {
|
|
// Play
|
|
if (toggle === true) {
|
|
_play();
|
|
}
|
|
// Pause
|
|
else if (toggle === false) {
|
|
_pause();
|
|
}
|
|
// True toggle
|
|
else {
|
|
player.media[player.media.paused ? 'play' : 'pause']();
|
|
}
|
|
}
|
|
|
|
// Rewind
|
|
function _rewind(seekTime) {
|
|
// Use default if needed
|
|
if (typeof seekTime !== 'number') {
|
|
seekTime = config.seekTime;
|
|
}
|
|
_seek(player.media.currentTime - seekTime);
|
|
}
|
|
|
|
// Fast forward
|
|
function _forward(seekTime) {
|
|
// Use default if needed
|
|
if (typeof seekTime !== 'number') {
|
|
seekTime = config.seekTime;
|
|
}
|
|
_seek(player.media.currentTime + seekTime);
|
|
}
|
|
|
|
// Seek to time
|
|
// The input parameter can be an event or a number
|
|
function _seek(input) {
|
|
var targetTime = 0,
|
|
paused = player.media.paused;
|
|
|
|
// Explicit position
|
|
if (typeof input === 'number') {
|
|
targetTime = input;
|
|
}
|
|
// Event
|
|
else if (typeof input === 'object' && (input.type === 'input' || input.type === 'change')) {
|
|
// It's the seek slider
|
|
// Seek to the selected time
|
|
targetTime = ((input.target.value / input.target.max) * player.media.duration);
|
|
}
|
|
|
|
// Normalise targetTime
|
|
if (targetTime < 0) {
|
|
targetTime = 0;
|
|
} else if (targetTime > player.media.duration) {
|
|
targetTime = player.media.duration;
|
|
}
|
|
|
|
// Set the current time
|
|
// Try/catch incase the media isn't set and we're calling seek() from source() and IE moans
|
|
try {
|
|
player.media.currentTime = targetTime.toFixed(1);
|
|
} catch (e) {}
|
|
|
|
// YouTube
|
|
if (player.type == 'youtube') {
|
|
player.embed.seekTo(targetTime);
|
|
|
|
if (paused) {
|
|
_pause();
|
|
}
|
|
|
|
// Trigger timeupdate
|
|
_triggerEvent(player.media, 'timeupdate');
|
|
}
|
|
|
|
// Logging
|
|
_log('Seeking to ' + player.media.currentTime + ' seconds');
|
|
|
|
// Special handling for 'manual' captions
|
|
_seekManualCaptions(targetTime);
|
|
}
|
|
|
|
// Check playing state
|
|
function _checkPlaying() {
|
|
_toggleClass(player.container, config.classes.playing, !player.media.paused);
|
|
_toggleClass(player.container, config.classes.stopped, player.media.paused);
|
|
}
|
|
|
|
// Toggle fullscreen
|
|
function _toggleFullscreen(event) {
|
|
// Check for native support
|
|
var nativeSupport = fullscreen.supportsFullScreen;
|
|
|
|
// If it's a fullscreen change event, it's probably a native close
|
|
if (event && event.type === fullscreen.fullScreenEventName) {
|
|
player.isFullscreen = fullscreen.isFullScreen(player.container);
|
|
}
|
|
// If there's native support, use it
|
|
else if (nativeSupport) {
|
|
// Request fullscreen
|
|
if (!fullscreen.isFullScreen(player.container)) {
|
|
fullscreen.requestFullScreen(player.container);
|
|
}
|
|
// Bail from fullscreen
|
|
else {
|
|
fullscreen.cancelFullScreen();
|
|
}
|
|
|
|
// Check if we're actually full screen (it could fail)
|
|
player.isFullscreen = fullscreen.isFullScreen(player.container);
|
|
} else {
|
|
// Otherwise, it's a simple toggle
|
|
player.isFullscreen = !player.isFullscreen;
|
|
|
|
// Bind/unbind escape key
|
|
if (player.isFullscreen) {
|
|
_on(document, 'keyup', _handleEscapeFullscreen);
|
|
document.body.style.overflow = 'hidden';
|
|
} else {
|
|
_off(document, 'keyup', _handleEscapeFullscreen);
|
|
document.body.style.overflow = '';
|
|
}
|
|
}
|
|
|
|
// Set class hook
|
|
_toggleClass(player.container, config.classes.fullscreen.active, player.isFullscreen);
|
|
|
|
// Set button state
|
|
_toggleState(player.buttons.fullscreen, player.isFullscreen);
|
|
|
|
// Toggle controls visibility based on mouse movement and location
|
|
var hoverTimer, isMouseOver = false;
|
|
|
|
// Show the player controls
|
|
function _showControls() {
|
|
// Set shown class
|
|
_toggleClass(player.container, config.classes.hover, true);
|
|
|
|
// Clear timer every movement
|
|
window.clearTimeout(hoverTimer);
|
|
|
|
// If the mouse is not over the controls, set a timeout to hide them
|
|
if (!isMouseOver) {
|
|
hoverTimer = window.setTimeout(function () {
|
|
_toggleClass(player.container, config.classes.hover, false);
|
|
}, 2000);
|
|
}
|
|
}
|
|
|
|
// Check mouse is over the controls
|
|
function _setMouseOver(event) {
|
|
isMouseOver = (event.type === 'mouseenter');
|
|
}
|
|
|
|
if (config.fullscreen.hideControls) {
|
|
// Hide on entering full screen
|
|
_toggleClass(player.controls, config.classes.hover, false);
|
|
|
|
// Keep an eye on the mouse location in relation to controls
|
|
_toggleHandler(player.controls, 'mouseenter mouseleave', _setMouseOver, player.isFullscreen);
|
|
|
|
// Show the controls on mouse move
|
|
_toggleHandler(player.container, 'mousemove', _showControls, player.isFullscreen);
|
|
}
|
|
}
|
|
|
|
// Bail from faux-fullscreen
|
|
function _handleEscapeFullscreen(event) {
|
|
// If it's a keypress and not escape, bail
|
|
if ((event.which || event.charCode || event.keyCode) === 27 && player.isFullscreen) {
|
|
_toggleFullscreen();
|
|
}
|
|
}
|
|
|
|
// Set volume
|
|
function _setVolume(volume) {
|
|
// Use default if no value specified
|
|
if (typeof volume === 'undefined') {
|
|
if (config.storage.enabled && _storage().supported) {
|
|
volume = window.localStorage[config.storage.key] || config.volume;
|
|
} else {
|
|
volume = config.volume;
|
|
}
|
|
}
|
|
|
|
// Maximum is 10
|
|
if (volume > 10) {
|
|
volume = 10;
|
|
}
|
|
// Minimum is 0
|
|
if (volume < 0) {
|
|
volume = 0;
|
|
}
|
|
|
|
// Set the player volume
|
|
player.media.volume = parseFloat(volume / 10);
|
|
|
|
// YouTube
|
|
if (player.type == 'youtube') {
|
|
player.embed.setVolume(player.media.volume * 100);
|
|
|
|
// Trigger timeupdate
|
|
_triggerEvent(player.media, 'volumechange');
|
|
}
|
|
|
|
// Toggle muted state
|
|
if (player.media.muted && volume > 0) {
|
|
_toggleMute();
|
|
}
|
|
}
|
|
|
|
// Mute
|
|
function _toggleMute(muted) {
|
|
// If the method is called without parameter, toggle based on current value
|
|
if (typeof muted !== 'boolean') {
|
|
muted = !player.media.muted;
|
|
}
|
|
|
|
// Set button state
|
|
_toggleState(player.buttons.mute, muted);
|
|
|
|
// Set mute on the player
|
|
player.media.muted = muted;
|
|
|
|
// YouTube
|
|
if (player.type === 'youtube') {
|
|
player.embed[player.media.muted ? 'mute' : 'unMute']();
|
|
|
|
// Trigger timeupdate
|
|
_triggerEvent(player.media, 'volumechange');
|
|
}
|
|
}
|
|
|
|
// Update volume UI and storage
|
|
function _updateVolume() {
|
|
// Get the current volume
|
|
var volume = player.media.muted ? 0 : (player.media.volume * 10);
|
|
|
|
// Update the <input type="range"> if present
|
|
if (player.supported.full && player.volume) {
|
|
player.volume.value = volume;
|
|
}
|
|
|
|
// Store the volume in storage
|
|
if (config.storage.enabled && _storage().supported) {
|
|
window.localStorage.setItem(config.storage.key, volume);
|
|
}
|
|
|
|
// Toggle class if muted
|
|
_toggleClass(player.container, config.classes.muted, (volume === 0));
|
|
|
|
// Update checkbox for mute state
|
|
if (player.supported.full && player.buttons.mute) {
|
|
_toggleState(player.buttons.mute, (volume === 0));
|
|
}
|
|
}
|
|
|
|
// Toggle captions
|
|
function _toggleCaptions(show) {
|
|
// If there's no full support, or there's no caption toggle
|
|
if (!player.supported.full || !player.buttons.captions) {
|
|
return;
|
|
}
|
|
|
|
// If the method is called without parameter, toggle based on current value
|
|
if (typeof show !== 'boolean') {
|
|
show = (player.container.className.indexOf(config.classes.captions.active) === -1);
|
|
}
|
|
|
|
// Toggle state
|
|
_toggleState(player.buttons.captions, show);
|
|
|
|
// Add class hook
|
|
_toggleClass(player.container, config.classes.captions.active, show);
|
|
}
|
|
|
|
// Check if media is loading
|
|
function _checkLoading(event) {
|
|
var loading = (event.type === 'waiting');
|
|
|
|
// Clear timer
|
|
clearTimeout(player.loadingTimer);
|
|
|
|
// Timer to prevent flicker when seeking
|
|
player.loadingTimer = setTimeout(function () {
|
|
_toggleClass(player.container, config.classes.loading, loading);
|
|
}, (loading ? 250 : 0));
|
|
}
|
|
|
|
// Update <progress> elements
|
|
function _updateProgress(event) {
|
|
var progress = player.progress.played.bar,
|
|
text = player.progress.played.text,
|
|
value = 0;
|
|
|
|
if (event) {
|
|
switch (event.type) {
|
|
// Video playing
|
|
case 'timeupdate':
|
|
case 'seeking':
|
|
value = _getPercentage(player.media.currentTime, player.media.duration);
|
|
|
|
// Set seek range value only if it's a 'natural' time event
|
|
if (event.type == 'timeupdate' && player.buttons.seek) {
|
|
player.buttons.seek.value = value;
|
|
}
|
|
|
|
break;
|
|
|
|
// Events from seek range
|
|
case 'change':
|
|
case 'input':
|
|
value = event.target.value;
|
|
break;
|
|
|
|
|
|
// Check buffer status
|
|
case 'playing':
|
|
case 'progress':
|
|
progress = player.progress.buffer.bar;
|
|
text = player.progress.buffer.text;
|
|
value = (function () {
|
|
var buffered = player.media.buffered;
|
|
|
|
// HTML5
|
|
if (buffered && buffered.length) {
|
|
return _getPercentage(buffered.end(0), player.media.duration);
|
|
}
|
|
// YouTube returns between 0 and 1
|
|
else if (typeof buffered === 'number') {
|
|
return (buffered * 100);
|
|
}
|
|
|
|
return 0;
|
|
})();
|
|
}
|
|
}
|
|
|
|
// Set values
|
|
if (progress) {
|
|
progress.value = value;
|
|
}
|
|
if (text) {
|
|
text.innerHTML = value;
|
|
}
|
|
}
|
|
|
|
// Update the displayed time
|
|
function _updateTimeDisplay(time, element) {
|
|
// Bail if there's no duration display
|
|
if (!element) {
|
|
return;
|
|
}
|
|
|
|
player.secs = parseInt(time % 60);
|
|
player.mins = parseInt((time / 60) % 60);
|
|
player.hours = parseInt(((time / 60) / 60) % 60);
|
|
|
|
// Do we need to display hours?
|
|
var displayHours = (parseInt(((player.media.duration / 60) / 60) % 60) > 0);
|
|
|
|
// 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
|
|
element.innerHTML = (displayHours ? player.hours + ':' : '') + player.mins + ':' + player.secs;
|
|
}
|
|
|
|
// Show the duration on metadataloaded
|
|
function _displayDuration() {
|
|
var duration = player.media.duration || 0;
|
|
|
|
// If there's only one time display, display duration there
|
|
if (!player.duration && config.displayDuration && player.media.paused) {
|
|
_updateTimeDisplay(duration, player.currentTime);
|
|
}
|
|
|
|
// If there's a duration element, update content
|
|
if (player.duration) {
|
|
_updateTimeDisplay(duration, player.duration);
|
|
}
|
|
}
|
|
|
|
// Handle time change event
|
|
function _timeUpdate(event) {
|
|
// Duration
|
|
_updateTimeDisplay(player.media.currentTime, player.currentTime);
|
|
|
|
// Playing progress
|
|
_updateProgress(event);
|
|
}
|
|
|
|
// Remove <source> children and src attribute
|
|
function _removeSources() {
|
|
// Find child <source> elements
|
|
var sources = player.media.querySelectorAll('source');
|
|
|
|
// Remove each
|
|
for (var i = sources.length - 1; i >= 0; i--) {
|
|
_remove(sources[i]);
|
|
}
|
|
|
|
// Remove src attribute
|
|
player.media.removeAttribute('src');
|
|
}
|
|
|
|
// Inject a source
|
|
function _addSource(attributes) {
|
|
if (attributes.src) {
|
|
// Create a new <source>
|
|
var element = document.createElement('source');
|
|
|
|
// Set all passed attributes
|
|
_setAttributes(element, attributes);
|
|
|
|
// Inject the new source
|
|
_prependChild(player.media, element);
|
|
}
|
|
}
|
|
|
|
// Update source
|
|
// Sources are not checked for support so be careful
|
|
function _parseSource(sources) {
|
|
// YouTube
|
|
if (player.type === 'youtube' && typeof sources === 'string') {
|
|
// Destroy YouTube instance
|
|
player.embed.destroy();
|
|
|
|
// Re-setup YouTube
|
|
// We don't use loadVideoBy[x] here since it has issues
|
|
_setupYouTube(sources);
|
|
|
|
// Update times
|
|
_timeUpdate();
|
|
|
|
// Bail
|
|
return;
|
|
}
|
|
|
|
// Pause playback (webkit freaks out)
|
|
_pause();
|
|
|
|
// Restart
|
|
_seek();
|
|
|
|
// Remove current sources
|
|
_removeSources();
|
|
|
|
// If a single source is passed
|
|
// .source('path/to/video.mp4')
|
|
if (typeof sources === 'string') {
|
|
_addSource({
|
|
src: sources
|
|
});
|
|
}
|
|
|
|
// An array of source objects
|
|
// Check if a source exists, use that or set the 'src' attribute?
|
|
// .source([{ src: 'path/to/video.mp4', type: 'video/mp4' },{ src: 'path/to/video.webm', type: 'video/webm' }])
|
|
else if (sources.constructor === Array) {
|
|
for (var index in sources) {
|
|
_addSource(sources[index]);
|
|
}
|
|
}
|
|
|
|
if (player.supported.full) {
|
|
// Reset time display
|
|
_timeUpdate();
|
|
|
|
// Update the UI
|
|
_checkPlaying();
|
|
}
|
|
|
|
// Re-load sources
|
|
player.media.load();
|
|
|
|
// Play if autoplay attribute is present
|
|
if (player.media.getAttribute('autoplay') !== null) {
|
|
_play();
|
|
}
|
|
}
|
|
|
|
// Update poster
|
|
function _updatePoster(source) {
|
|
if (player.type === 'video') {
|
|
player.media.setAttribute('poster', source);
|
|
}
|
|
}
|
|
|
|
// Listen for events
|
|
function _listeners() {
|
|
// IE doesn't support input event, so we fallback to change
|
|
var inputEvent = (player.browser.name == 'IE' ? 'change' : 'input');
|
|
|
|
// Detect tab focus
|
|
function checkFocus() {
|
|
var focused = document.activeElement;
|
|
if (!focused || focused == document.body) {
|
|
focused = null;
|
|
} else if (document.querySelector) {
|
|
focused = document.querySelector(':focus');
|
|
}
|
|
for (var button in player.buttons) {
|
|
var element = player.buttons[button];
|
|
|
|
_toggleClass(element, 'tab-focus', (element === focused));
|
|
}
|
|
}
|
|
_on(window, 'keyup', function (event) {
|
|
var code = (event.keyCode ? event.keyCode : event.which);
|
|
|
|
if (code == 9) {
|
|
checkFocus();
|
|
}
|
|
});
|
|
for (var button in player.buttons) {
|
|
var element = player.buttons[button];
|
|
|
|
_on(element, 'blur', function () {
|
|
_toggleClass(element, 'tab-focus', false);
|
|
});
|
|
}
|
|
|
|
// Play
|
|
_on(player.buttons.play, 'click', function () {
|
|
_play();
|
|
setTimeout(function () {
|
|
player.buttons.pause.focus();
|
|
}, 100);
|
|
});
|
|
|
|
// Pause
|
|
_on(player.buttons.pause, 'click', function () {
|
|
_pause();
|
|
setTimeout(function () {
|
|
player.buttons.play.focus();
|
|
}, 100);
|
|
});
|
|
|
|
// Restart
|
|
_on(player.buttons.restart, 'click', _seek);
|
|
|
|
// Rewind
|
|
_on(player.buttons.rewind, 'click', _rewind);
|
|
|
|
// Fast forward
|
|
_on(player.buttons.forward, 'click', _forward);
|
|
|
|
// Seek
|
|
_on(player.buttons.seek, inputEvent, _seek);
|
|
|
|
// Set volume
|
|
_on(player.volume, inputEvent, function () {
|
|
_setVolume(this.value);
|
|
});
|
|
|
|
// Mute
|
|
_on(player.buttons.mute, 'click', _toggleMute);
|
|
|
|
// Fullscreen
|
|
_on(player.buttons.fullscreen, 'click', _toggleFullscreen);
|
|
|
|
// Handle user exiting fullscreen by escaping etc
|
|
if (fullscreen.supportsFullScreen) {
|
|
_on(document, fullscreen.fullScreenEventName, _toggleFullscreen);
|
|
}
|
|
|
|
// Time change on media
|
|
_on(player.media, 'timeupdate seeking', _timeUpdate);
|
|
|
|
// Update manual captions
|
|
_on(player.media, 'timeupdate', _seekManualCaptions);
|
|
|
|
// Display duration
|
|
_on(player.media, 'loadedmetadata', _displayDuration);
|
|
|
|
// Captions
|
|
_on(player.buttons.captions, 'click', _toggleCaptions);
|
|
|
|
// Handle the media finishing
|
|
_on(player.media, 'ended', function () {
|
|
// Clear
|
|
if (player.type === 'video') {
|
|
player.captionsContainer.innerHTML = '';
|
|
}
|
|
|
|
// Reset UI
|
|
_checkPlaying();
|
|
});
|
|
|
|
// Check for buffer progress
|
|
_on(player.media, 'progress playing', _updateProgress);
|
|
|
|
// Handle native mute
|
|
_on(player.media, 'volumechange', _updateVolume);
|
|
|
|
// Handle native play/pause
|
|
_on(player.media, 'play pause', _checkPlaying);
|
|
|
|
// Loading
|
|
_on(player.media, 'waiting canplay seeked', _checkLoading);
|
|
|
|
// Click video
|
|
if (player.type === 'video' && config.click) {
|
|
_on(player.videoContainer, 'click', function () {
|
|
if (player.media.paused) {
|
|
_triggerEvent(player.buttons.play, 'click');
|
|
} else if (player.media.ended) {
|
|
_seek();
|
|
_triggerEvent(player.buttons.play, 'click');
|
|
} else {
|
|
_triggerEvent(player.buttons.pause, 'click');
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
// Destroy an instance
|
|
// Event listeners are removed when elements are removed
|
|
// http://stackoverflow.com/questions/12528049/if-a-dom-element-is-removed-are-its-listeners-also-removed-from-memory
|
|
function _destroy() {
|
|
// Bail if the element is not initialized
|
|
if (!player.init) {
|
|
return null;
|
|
}
|
|
|
|
// Reset container classname
|
|
player.container.setAttribute('class', config.selectors.container.replace('.', ''));
|
|
|
|
// Remove init flag
|
|
player.init = false;
|
|
|
|
// Remove controls
|
|
_remove(_getElement(config.selectors.controls));
|
|
|
|
// YouTube
|
|
if (player.type === 'youtube') {
|
|
player.embed.destroy();
|
|
return;
|
|
}
|
|
|
|
// If video, we need to remove some more
|
|
if (player.type === 'video') {
|
|
// Remove captions
|
|
_remove(_getElement(config.selectors.captions));
|
|
|
|
// Remove video wrapper
|
|
_unwrap(player.videoContainer);
|
|
}
|
|
|
|
// Restore native video controls
|
|
player.media.setAttribute('controls', '');
|
|
|
|
// Clone the media element to remove listeners
|
|
// http://stackoverflow.com/questions/19469881/javascript-remove-all-event-listeners-of-specific-type
|
|
var clone = player.media.cloneNode(true);
|
|
player.media.parentNode.replaceChild(clone, player.media);
|
|
}
|
|
|
|
// Setup a player
|
|
function _init() {
|
|
// Bail if the element is initialized
|
|
if (player.init) {
|
|
return null;
|
|
}
|
|
|
|
// Setup the fullscreen api
|
|
fullscreen = _fullscreen();
|
|
|
|
// Sniff out the browser
|
|
player.browser = _browserSniff();
|
|
|
|
// Get the media element
|
|
player.media = player.container.querySelectorAll('audio, video, div')[0];
|
|
|
|
// Set media type
|
|
var tagName = player.media.tagName.toLowerCase();
|
|
if (tagName === 'div') {
|
|
player.type = player.media.getAttribute('data-type');
|
|
} else {
|
|
player.type = tagName;
|
|
}
|
|
|
|
// Check for full support
|
|
player.supported = api.supported(player.type);
|
|
|
|
// If no native support, bail
|
|
if (!player.supported.basic) {
|
|
return false;
|
|
}
|
|
|
|
// Debug info
|
|
_log(player.browser.name + ' ' + player.browser.version);
|
|
|
|
// Setup media
|
|
_setupMedia();
|
|
|
|
// Setup interface
|
|
if (player.type == 'video' || player.type == 'audio') {
|
|
// Bail if no support
|
|
if (!player.supported.full) {
|
|
// Successful setup
|
|
player.init = true;
|
|
|
|
// Don't inject controls if no full support
|
|
return;
|
|
}
|
|
|
|
// Setup UI
|
|
_setupInterface();
|
|
|
|
// Display duration if available
|
|
if (config.displayDuration) {
|
|
_displayDuration();
|
|
}
|
|
|
|
// Set up aria-label for Play button with the title option
|
|
_setupPlayAria();
|
|
}
|
|
|
|
// Successful setup
|
|
player.init = true;
|
|
}
|
|
|
|
function _setupInterface() {
|
|
// Inject custom controls
|
|
_injectControls();
|
|
|
|
// Find the elements
|
|
if (!_findElements()) {
|
|
return false;
|
|
}
|
|
|
|
// Captions
|
|
_setupCaptions();
|
|
|
|
// Set volume
|
|
_setVolume();
|
|
_updateVolume();
|
|
|
|
// Setup fullscreen
|
|
_setupFullscreen();
|
|
|
|
// Listeners
|
|
_listeners();
|
|
}
|
|
|
|
// Initialize instance
|
|
_init();
|
|
|
|
// If init failed, return an empty object
|
|
if (!player.init) {
|
|
return {};
|
|
}
|
|
|
|
return {
|
|
media: player.media,
|
|
play: _play,
|
|
pause: _pause,
|
|
restart: _seek,
|
|
rewind: _rewind,
|
|
forward: _forward,
|
|
seek: _seek,
|
|
source: _parseSource,
|
|
poster: _updatePoster,
|
|
setVolume: _setVolume,
|
|
togglePlay: _togglePlay,
|
|
toggleMute: _toggleMute,
|
|
toggleCaptions: _toggleCaptions,
|
|
toggleFullscreen: _toggleFullscreen,
|
|
isFullscreen: function () {
|
|
return player.isFullscreen || false;
|
|
},
|
|
support: function (mimeType) {
|
|
return _supportMime(player, mimeType);
|
|
},
|
|
destroy: _destroy,
|
|
restore: _init
|
|
};
|
|
}
|
|
|
|
// Check for support
|
|
api.supported = function (type) {
|
|
var browser = _browserSniff(),
|
|
oldIE = (browser.name === 'IE' && browser.version <= 9),
|
|
iPhone = /iPhone|iPod/i.test(navigator.userAgent),
|
|
audio = !!document.createElement('audio').canPlayType,
|
|
video = !!document.createElement('video').canPlayType,
|
|
basic, full;
|
|
|
|
switch (type) {
|
|
case 'video':
|
|
basic = video;
|
|
full = (basic && (!oldIE && !iPhone));
|
|
break;
|
|
|
|
case 'audio':
|
|
basic = audio;
|
|
full = (basic && !oldIE);
|
|
break;
|
|
|
|
case 'youtube':
|
|
basic = true;
|
|
full = (!oldIE && !iPhone);
|
|
break;
|
|
|
|
default:
|
|
basic = (audio && video);
|
|
full = (basic && !oldIE);
|
|
}
|
|
|
|
return {
|
|
basic: basic,
|
|
full: full
|
|
};
|
|
};
|
|
|
|
// Expose setup function
|
|
api.setup = function (options) {
|
|
// Extend the default options with user specified
|
|
config = _extend(defaults, options);
|
|
|
|
// Bail if disabled or no basic support
|
|
// You may want to disable certain UAs etc
|
|
if (!config.enabled || !api.supported().basic) {
|
|
return false;
|
|
}
|
|
|
|
// Get the players
|
|
var elements = document.querySelectorAll(config.selectors.container),
|
|
players = [];
|
|
|
|
// Create a player instance for each element
|
|
for (var i = elements.length - 1; i >= 0; i--) {
|
|
// Get the current element
|
|
var element = elements[i];
|
|
|
|
// Setup a player instance and add to the element
|
|
if (typeof element.plyr === 'undefined') {
|
|
// Create new instance
|
|
var instance = new Plyr(element);
|
|
|
|
// Set plyr to false if setup failed
|
|
element.plyr = (Object.keys(instance).length ? instance : false);
|
|
|
|
// Callback
|
|
if (typeof config.onSetup === 'function') {
|
|
config.onSetup.apply(element.plyr);
|
|
}
|
|
}
|
|
|
|
// Add to return array even if it's already setup
|
|
players.push(element.plyr);
|
|
}
|
|
|
|
return players;
|
|
};
|
|
|
|
}(this.plyr = this.plyr || {}));
|