Work on source update and caption fixes
This commit is contained in:
parent
c48afb7880
commit
4f47ec49b1
2
dist/plyr.js
vendored
2
dist/plyr.js
vendored
File diff suppressed because one or more lines are too long
315
src/js/plyr.js
315
src/js/plyr.js
@ -204,6 +204,36 @@
|
|||||||
// Return data
|
// Return data
|
||||||
return [browserName, majorVersion];
|
return [browserName, majorVersion];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Check for mime type support against a player instance
|
||||||
|
// Credits: http://diveintohtml5.info/everything.html
|
||||||
|
// Related: http://www.leanbackplayer.com/test/h5mt.html
|
||||||
|
function _support(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;
|
||||||
|
}
|
||||||
|
|
||||||
// Replace all
|
// Replace all
|
||||||
function _replaceAll(string, find, replace) {
|
function _replaceAll(string, find, replace) {
|
||||||
@ -242,6 +272,11 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Remove an element
|
||||||
|
function _remove(element) {
|
||||||
|
element.parentNode.removeChild(element);
|
||||||
|
}
|
||||||
|
|
||||||
// Toggle class on an element
|
// Toggle class on an element
|
||||||
function _toggleClass(element, name, state) {
|
function _toggleClass(element, name, state) {
|
||||||
if(element){
|
if(element){
|
||||||
@ -264,18 +299,6 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Set attributes
|
|
||||||
function _setAttributes(element, attributes) {
|
|
||||||
for(var key in attributes) {
|
|
||||||
element.setAttribute(key, attributes[key]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Prepend child
|
|
||||||
function _prependChild(parent, element) {
|
|
||||||
parent.insertBefore(element, parent.firstChild);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Bind event
|
// Bind event
|
||||||
function _on(element, events, callback) {
|
function _on(element, events, callback) {
|
||||||
_toggleHandler(element, events, callback, true);
|
_toggleHandler(element, events, callback, true);
|
||||||
@ -404,19 +427,45 @@
|
|||||||
player.container = container;
|
player.container = container;
|
||||||
|
|
||||||
// Captions functions
|
// Captions functions
|
||||||
// Credits: http://paypal.github.io/accessible-html5-video-player/
|
// 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") {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
// For "manual" captions, adjust caption position when play time changed (via rewind, clicking progress bar, etc.)
|
// Reset subcount
|
||||||
function _adjustManualCaptions() {
|
|
||||||
player.subcount = 0;
|
player.subcount = 0;
|
||||||
while (_timecodeMax(player.captions[player.subcount][0]) < player.media.currentTime.toFixed(1)) {
|
|
||||||
|
// 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++;
|
player.subcount++;
|
||||||
if (player.subcount > player.captions.length-1) {
|
if (player.subcount > player.captions.length-1) {
|
||||||
player.subcount = player.captions.length-1;
|
player.subcount = player.captions.length-1;
|
||||||
break;
|
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];
|
||||||
|
}
|
||||||
|
|
||||||
|
// 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;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Display captions container and button (for initialization)
|
// Display captions container and button (for initialization)
|
||||||
function _showCaptions() {
|
function _showCaptions() {
|
||||||
_toggleClass(player.container, config.classes.captions.enabled, true);
|
_toggleClass(player.container, config.classes.captions.enabled, true);
|
||||||
@ -426,6 +475,7 @@
|
|||||||
player.buttons.captions.setAttribute("checked", "checked");
|
player.buttons.captions.setAttribute("checked", "checked");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Utilities for caption time codes
|
// Utilities for caption time codes
|
||||||
function _timecodeMin(tc) {
|
function _timecodeMin(tc) {
|
||||||
var tcpair = [];
|
var tcpair = [];
|
||||||
@ -600,9 +650,9 @@
|
|||||||
player.captionsContainer = _getElement(config.selectors.captions);
|
player.captionsContainer = _getElement(config.selectors.captions);
|
||||||
|
|
||||||
// Determine if HTML5 textTracks is supported
|
// Determine if HTML5 textTracks is supported
|
||||||
player.isTextTracks = false;
|
player.usingTextTracks = false;
|
||||||
if (player.media.textTracks) {
|
if (player.media.textTracks) {
|
||||||
player.isTextTracks = true;
|
player.usingTextTracks = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get URL of caption file if exists
|
// Get URL of caption file if exists
|
||||||
@ -654,12 +704,12 @@
|
|||||||
_log("Detected IE 10/11 or Firefox 31+ or Safari 7+.");
|
_log("Detected IE 10/11 or Firefox 31+ or Safari 7+.");
|
||||||
|
|
||||||
// Set to false so skips to "manual" captioning
|
// Set to false so skips to "manual" captioning
|
||||||
player.isTextTracks = false;
|
player.usingTextTracks = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Rendering caption tracks
|
// Rendering caption tracks
|
||||||
// Native support required - http://caniuse.com/webvtt
|
// Native support required - http://caniuse.com/webvtt
|
||||||
if (player.isTextTracks) {
|
if (player.usingTextTracks) {
|
||||||
_log("TextTracks supported.");
|
_log("TextTracks supported.");
|
||||||
|
|
||||||
for (var y=0; y < tracks.length; y++) {
|
for (var y=0; y < tracks.length; y++) {
|
||||||
@ -685,21 +735,6 @@
|
|||||||
player.subcount = 0;
|
player.subcount = 0;
|
||||||
player.captions = [];
|
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 !== "") {
|
if (captionSrc !== "") {
|
||||||
// Create XMLHttpRequest Object
|
// Create XMLHttpRequest Object
|
||||||
var xhr = new XMLHttpRequest();
|
var xhr = new XMLHttpRequest();
|
||||||
@ -780,17 +815,6 @@
|
|||||||
player.media.pause();
|
player.media.pause();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Restart playback
|
|
||||||
function _restart() {
|
|
||||||
// Move to beginning
|
|
||||||
player.media.currentTime = 0;
|
|
||||||
|
|
||||||
// Special handling for "manual" captions
|
|
||||||
if (!player.isTextTracks) {
|
|
||||||
player.subcount = 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Rewind
|
// Rewind
|
||||||
function _rewind(seekTime) {
|
function _rewind(seekTime) {
|
||||||
// Use default if needed
|
// Use default if needed
|
||||||
@ -810,8 +834,8 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Seek to time
|
// Seek to time
|
||||||
|
// The parameter can be an event or a number
|
||||||
var _seek = function(input) {
|
var _seek = function(input) {
|
||||||
//var value = config.seekTime;
|
|
||||||
var targetTime = 0;
|
var targetTime = 0;
|
||||||
|
|
||||||
// If no event or time is passed, bail
|
// If no event or time is passed, bail
|
||||||
@ -826,19 +850,25 @@
|
|||||||
else if (input.type === "change" || input.type === "input") {
|
else if (input.type === "change" || input.type === "input") {
|
||||||
// It's the seek slider
|
// It's the seek slider
|
||||||
// Seek to the selected time
|
// Seek to the selected time
|
||||||
targetTime = ((this.value / this.max) * player.media.duration).toFixed(1);
|
targetTime = ((this.value / this.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
|
// Set the current time
|
||||||
player.media.currentTime = targetTime;
|
player.media.currentTime = targetTime.toFixed(1);
|
||||||
|
|
||||||
// Logging
|
// Logging
|
||||||
_log("Seeking to " + player.media.currentTime + " seconds");
|
_log("Seeking to " + player.media.currentTime + " seconds");
|
||||||
|
|
||||||
// Special handling for "manual" captions
|
// Special handling for "manual" captions
|
||||||
if (!player.isTextTracks && player.type === "video") {
|
_seekManualCaptions(targetTime);
|
||||||
_adjustManualCaptions(player);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check playing state
|
// Check playing state
|
||||||
@ -958,53 +988,52 @@
|
|||||||
|
|
||||||
// Update <progress> elements
|
// Update <progress> elements
|
||||||
function _updateProgress(event) {
|
function _updateProgress(event) {
|
||||||
var progress, text, value = 0;
|
var progress = player.progress.played.bar,
|
||||||
|
text = player.progress.played.text,
|
||||||
|
value = 0;
|
||||||
|
|
||||||
switch(event.type) {
|
if(event) {
|
||||||
// Video playing
|
switch(event.type) {
|
||||||
case "timeupdate":
|
// Video playing
|
||||||
case "seeking":
|
case "timeupdate":
|
||||||
progress = player.progress.played.bar;
|
case "seeking":
|
||||||
text = player.progress.played.text;
|
value = _getPercentage(player.media.currentTime, player.media.duration);
|
||||||
value = _getPercentage(player.media.currentTime, player.media.duration);
|
|
||||||
|
|
||||||
// Set seek range value only if it's a "natural" time event
|
// Set seek range value only if it's a "natural" time event
|
||||||
if(event.type == "timeupdate") {
|
if(event.type == "timeupdate") {
|
||||||
player.buttons.seek.value = value;
|
player.buttons.seek.value = value;
|
||||||
}
|
}
|
||||||
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
// Events from seek range
|
// Events from seek range
|
||||||
case "change":
|
case "change":
|
||||||
case "input":
|
case "input":
|
||||||
progress = player.progress.played.bar;
|
value = event.target.value;
|
||||||
text = player.progress.played.text;
|
break;
|
||||||
value = event.target.value;
|
|
||||||
break;
|
|
||||||
|
|
||||||
|
|
||||||
// Check buffer status
|
// Check buffer status
|
||||||
case "playing":
|
case "playing":
|
||||||
case "progress":
|
case "progress":
|
||||||
progress = player.progress.buffer.bar;
|
progress = player.progress.buffer.bar;
|
||||||
text = player.progress.buffer.text;
|
text = player.progress.buffer.text;
|
||||||
value = (function() {
|
value = (function() {
|
||||||
var buffered = player.media.buffered;
|
var buffered = player.media.buffered;
|
||||||
|
|
||||||
if(buffered.length) {
|
if(buffered.length) {
|
||||||
return _getPercentage(buffered.end(0), player.media.duration);
|
return _getPercentage(buffered.end(0), player.media.duration);
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
})();
|
})();
|
||||||
break;
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (progress && value > 0) {
|
// Set values
|
||||||
progress.value = value;
|
progress.value = value;
|
||||||
text.innerHTML = value;
|
text.innerHTML = value;
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Update the displayed play time
|
// Update the displayed play time
|
||||||
@ -1024,20 +1053,19 @@
|
|||||||
function _timeUpdate(event) {
|
function _timeUpdate(event) {
|
||||||
// Duration
|
// Duration
|
||||||
_updateTimeDisplay();
|
_updateTimeDisplay();
|
||||||
|
|
||||||
// Playing progress
|
// Playing progress
|
||||||
_updateProgress(event);
|
_updateProgress(event);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Remove an element
|
// Remove <source> children and src attribute
|
||||||
function _remove(element) {
|
|
||||||
element.parentNode.removeChild(element);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Remove sources
|
|
||||||
function _removeSources() {
|
function _removeSources() {
|
||||||
// Remove child <source> elements
|
// Find child <source> elements
|
||||||
var sources = player.media.querySelectorAll("source");
|
var sources = player.media.querySelectorAll("source");
|
||||||
|
|
||||||
|
// Remove each
|
||||||
for (var i = sources.length - 1; i >= 0; i--) {
|
for (var i = sources.length - 1; i >= 0; i--) {
|
||||||
|
_log(sources[i]);
|
||||||
_remove(sources[i]);
|
_remove(sources[i]);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1045,51 +1073,60 @@
|
|||||||
player.media.removeAttribute("src");
|
player.media.removeAttribute("src");
|
||||||
}
|
}
|
||||||
|
|
||||||
// Inject a source
|
// Set source
|
||||||
function _addSource(attributes) {
|
function _setSource(source) {
|
||||||
// Create a new <source>
|
if(source.type && source.src) {
|
||||||
var element = document.createElement("source");
|
// Check if it's supported first
|
||||||
|
if(_support(player, source.type)) {
|
||||||
|
// Pause playback (webkit freaks out)
|
||||||
|
_pause();
|
||||||
|
|
||||||
// Set all passed attributes
|
// Update the UI
|
||||||
_setAttributes(element, attributes);
|
_checkPlaying();
|
||||||
|
|
||||||
// Inject the new source
|
// Remove current sources
|
||||||
_prependChild(player.media, element);
|
_removeSources();
|
||||||
|
|
||||||
|
// Set the src attribute
|
||||||
|
player.media.setAttribute("src", source.src);
|
||||||
|
|
||||||
|
// Restart
|
||||||
|
_seek();
|
||||||
|
|
||||||
|
// Reset time display
|
||||||
|
_timeUpdate();
|
||||||
|
|
||||||
|
// Play if autoplay attribute is present
|
||||||
|
if(player.media.getAttribute("autoplay") !== null) {
|
||||||
|
_play();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
_log("No support for: " + source.src + " [" + source.type + "]");
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Update source
|
// Update source
|
||||||
function _updateSource(sources) {
|
function _parseSource(sources) {
|
||||||
// Pause on update
|
// If a single source object is provided
|
||||||
// Play automatically if autoplay set or already playing
|
// ({ src: "//cdn.selz.com/plyr/1.0/movie.webm", type: "video/webm" })
|
||||||
|
if(typeof sources === "object" && sources.constructor !== Array) {
|
||||||
// Remove current sources
|
|
||||||
_removeSources();
|
|
||||||
|
|
||||||
// If a single source is provided
|
|
||||||
// ("path/to/src.mp4")
|
|
||||||
if(typeof sources === "string") {
|
|
||||||
// Set src attribute on the element
|
// Set src attribute on the element
|
||||||
player.media.setAttribute("src", sources);
|
_setSource(sources);
|
||||||
}
|
}
|
||||||
// Single source but using object to pass attributes
|
|
||||||
// ({ src: "path/to/src.mp4", type: "video/mp4" })
|
// An array of source objects
|
||||||
else if (typeof sources === "object") {
|
// Check if a source exists, use that or set the "src" attribute?
|
||||||
_addSource(sources);
|
// [{ src: "path/to/src.mp4", type: "video/mp4" },{ src: "path/to/src.webm", type: "video/webm" }]
|
||||||
}
|
|
||||||
// Array of source objects to pass attributes
|
|
||||||
// ([{ src: "path/to/src.mp4", type: "video/mp4" },{ src: "path/to/src.webm", type: "video/webm" }])
|
|
||||||
else if (sources.constructor === Array) {
|
else if (sources.constructor === Array) {
|
||||||
for (var key in sources) {
|
for (var index in sources) {
|
||||||
_addSource(sources[key]);
|
_setSource(sources[index]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// Not an object or an array
|
||||||
// Restart
|
else {
|
||||||
_restart();
|
_log("Bad source format...");
|
||||||
|
|
||||||
// Play if autoplay attribute is present
|
|
||||||
if(player.media.getAttribute("autoplay") !== null) {
|
|
||||||
_play();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1115,7 +1152,7 @@
|
|||||||
});
|
});
|
||||||
|
|
||||||
// Restart
|
// Restart
|
||||||
_on(player.buttons.restart, "click", _restart);
|
_on(player.buttons.restart, "click", _seek);
|
||||||
|
|
||||||
// Rewind
|
// Rewind
|
||||||
_on(player.buttons.rewind, "click", _rewind);
|
_on(player.buttons.rewind, "click", _rewind);
|
||||||
@ -1147,7 +1184,7 @@
|
|||||||
_play();
|
_play();
|
||||||
}
|
}
|
||||||
else if(player.media.ended) {
|
else if(player.media.ended) {
|
||||||
_restart();
|
_seek();
|
||||||
_play();
|
_play();
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
@ -1159,6 +1196,9 @@
|
|||||||
// Time change on media
|
// Time change on media
|
||||||
_on(player.media, "timeupdate seeking", _timeUpdate);
|
_on(player.media, "timeupdate seeking", _timeUpdate);
|
||||||
|
|
||||||
|
// Update manual captions
|
||||||
|
_on(player.media, "timeupdate", _seekManualCaptions);
|
||||||
|
|
||||||
// Seek
|
// Seek
|
||||||
_on(player.buttons.seek, "change input", _seek);
|
_on(player.buttons.seek, "change input", _seek);
|
||||||
|
|
||||||
@ -1243,15 +1283,16 @@
|
|||||||
media: player.media,
|
media: player.media,
|
||||||
play: _play,
|
play: _play,
|
||||||
pause: _pause,
|
pause: _pause,
|
||||||
restart: _restart,
|
restart: _seek,
|
||||||
rewind: _rewind,
|
rewind: _rewind,
|
||||||
forward: _forward,
|
forward: _forward,
|
||||||
seek: _seek,
|
seek: _seek,
|
||||||
setVolume: _setVolume,
|
setVolume: _setVolume,
|
||||||
toggleMute: _toggleMute,
|
toggleMute: _toggleMute,
|
||||||
toggleCaptions: _toggleCaptions,
|
toggleCaptions: _toggleCaptions,
|
||||||
source: _updateSource,
|
source: _parseSource,
|
||||||
poster: _updatePoster
|
poster: _updatePoster,
|
||||||
|
support: _support
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user