Work on player rewrites

This commit is contained in:
calzoneman 2015-05-02 11:45:35 -05:00
parent fa3ddbe94a
commit 391ea264f5
6 changed files with 385 additions and 221 deletions

View File

@ -1,5 +1,8 @@
class Player window.Player = class Player
constructor: (data) -> constructor: (data) ->
if not (this instanceof Player)
return new Player(data)
@setMediaProperties(data) @setMediaProperties(data)
@paused = false @paused = false
@ -29,91 +32,3 @@ class Player
getVolume: (cb) -> getVolume: (cb) ->
cb(VOLUME) cb(VOLUME)
window.Player = Player
window.removeOld = (replace) ->
$('#sc_volume').remove()
replace ?= $('<div/>').addClass('embed-responsive-item')
old = $('#ytapiplayer')
replace.insertBefore(old)
old.remove()
replace.attr('id', 'ytapiplayer')
return replace
TYPE_MAP =
yt: 'YouTubePlayer'
window.loadMediaPlayer = (data) ->
if data.type of TYPE_MAP
ctor = window[TYPE_MAP[data.type]]
window.PLAYER = new ctor(data)
window.handleMediaUpdate = (data) ->
PLAYER = window.PLAYER
# Do not update if the current time is past the end of the video, unless
# the video has length 0 (which is a special case for livestreams)
if typeof PLAYER.mediaLength is 'number' and
PLAYER.mediaLength > 0 and
data.currentTime > PLAYER.mediaLength
return
# Negative currentTime indicates a lead-in for clients to load the video,
# but not play it yet (helps with initial buffering)
waiting = data.currentTime < 0
# Load a new video in the same player if the ID changed
if data.id and data.id != PLAYER.mediaId
if data.currentTime < 0
data.currentTime = 0
PLAYER.load(data)
PLAYER.play()
if waiting
console.log('waiting')
# YouTube player has a race condition that crashes the player if
# play(), seek(0), and pause() are called quickly without waiting
# for events to fire. Setting a flag variable that is checked in the
# event handler mitigates this.
if PLAYER.type is 'yt'
PLAYER.pauseSeekRaceCondition = true
else
PLAYER.seekTo(0)
PLAYER.pause()
else if PLAYER.type is 'yt'
PLAYER.pauseSeekRaceCondition = false
if CLIENT.leader or not USEROPTS.synch
return
if data.paused and not PLAYER.paused
PLAYER.seekTo(data.currentTime)
PLAYER.pause()
else if PLAYER.paused
PLAYER.play()
PLAYER.getTime((seconds) ->
time = data.currentTime
diff = (time - seconds) or time
accuracy = USEROPTS.sync_accuracy
# Dailymotion can't seek very accurately in Flash due to keyframe
# placement. Accuracy should not be set lower than 5 or the video
# may be very choppy.
if PLAYER.type is 'dm'
accuracy = Math.max(accuracy, 5)
if diff > accuracy
# The player is behind the correct time
PLAYER.seekTo(time)
else if diff < -accuracy
# The player is ahead of the correct time
# Don't seek all the way back, to account for possible buffering.
# However, do seek all the way back for Dailymotion due to the
# keyframe issue mentioned above.
if PLAYER.type isnt 'dm'
time += 1
PLAYER.seekTo(time)
)

89
player/update.coffee Normal file
View File

@ -0,0 +1,89 @@
TYPE_MAP =
yt: YouTubePlayer
vi: VimeoPlayer
window.loadMediaPlayer = (data) ->
if data.type of TYPE_MAP
console.log data
try
window.PLAYER = TYPE_MAP[data.type](data)
catch e
console.error e
console.log(window.PLAYER)
window.handleMediaUpdate = (data) ->
PLAYER = window.PLAYER
# Do not update if the current time is past the end of the video, unless
# the video has length 0 (which is a special case for livestreams)
if typeof PLAYER.mediaLength is 'number' and
PLAYER.mediaLength > 0 and
data.currentTime > PLAYER.mediaLength
return
# Negative currentTime indicates a lead-in for clients to load the video,
# but not play it yet (helps with initial buffering)
waiting = data.currentTime < 0
# Load a new video in the same player if the ID changed
if data.id and data.id != PLAYER.mediaId
if data.currentTime < 0
data.currentTime = 0
PLAYER.load(data)
PLAYER.play()
if waiting
# YouTube player has a race condition that crashes the player if
# play(), seek(0), and pause() are called quickly without waiting
# for events to fire. Setting a flag variable that is checked in the
# event handler mitigates this.
if PLAYER instanceof YouTubePlayer
PLAYER.pauseSeekRaceCondition = true
else
PLAYER.seekTo(0)
PLAYER.pause()
else if PLAYER instanceof YouTubePlayer
PLAYER.pauseSeekRaceCondition = false
if CLIENT.leader or not USEROPTS.synch
return
if data.paused and not PLAYER.paused
PLAYER.seekTo(data.currentTime)
PLAYER.pause()
else if PLAYER.paused
PLAYER.play()
PLAYER.getTime((seconds) ->
time = data.currentTime
diff = (time - seconds) or time
accuracy = USEROPTS.sync_accuracy
# Dailymotion can't seek very accurately in Flash due to keyframe
# placement. Accuracy should not be set lower than 5 or the video
# may be very choppy.
if PLAYER.mediaType is 'dm' # TODO: instanceof DailymotionPlayer
accuracy = Math.max(accuracy, 5)
if diff > accuracy
# The player is behind the correct time
PLAYER.seekTo(time)
else if diff < -accuracy
# The player is ahead of the correct time
# Don't seek all the way back, to account for possible buffering.
# However, do seek all the way back for Dailymotion due to the
# keyframe issue mentioned above.
if PLAYER.mediaType isnt 'dm' # TODO: instanceof DailymotionPlayer
time += 1
PLAYER.seekTo(time)
)
window.removeOld = (replace) ->
$('#sc_volume').remove()
replace ?= $('<div/>').addClass('embed-responsive-item')
old = $('#ytapiplayer')
replace.insertBefore(old)
old.remove()
replace.attr('id', 'ytapiplayer')
return replace

80
player/vimeo.coffee Normal file
View File

@ -0,0 +1,80 @@
window.VimeoPlayer = class VimeoPlayer extends Player
constructor: (data) ->
if not (this instanceof VimeoPlayer)
return new VimeoPlayer(data)
@load(data)
load: (data) ->
@setMediaProperties(data)
waitUntilDefined(window, '$f', =>
video = $('<iframe/>')
removeOld(video)
video.attr(
src: "https://player.vimeo.com/video/#{data.id}?api=1&player_id=ytapiplayer"
webkitallowfullscreen: true
mozallowfullscreen: true
allowfullscreen: true
)
if USEROPTS.wmode_transparent
video.attr('wmode', 'transparent')
$f(video[0]).addEvent('ready', =>
@vimeo = $f(video[0])
@play()
@vimeo.addEvent('finish', =>
if CLIENT.leader
socket.emit('playNext')
)
@vimeo.addEvent('pause', =>
@paused = true
if CLIENT.leader
sendVideoUpdate()
)
@vimeo.addEvent('play', =>
@paused = false
if CLIENT.leader
sendVideoUpdate()
)
@setVolume(VOLUME)
)
)
play: ->
@paused = false
if @vimeo
@vimeo.api('play')
pause: ->
@paused = true
if @vimeo
@vimeo.api('pause')
seekTo: (time) ->
if @vimeo
@vimeo.api('seekTo', time)
setVolume: (volume) ->
if @vimeo
@vimeo.api('setVolume', volume)
getTime: (cb) ->
if @vimeo
@vimeo.api('getCurrentTime', (time) ->
# I will never understand why Vimeo returns current time as a string
cb(parseFloat(time))
)
else
cb(0)
getVolume: (cb) ->
if @vimeo
@vimeo.api('getVolume', cb)
else
cb(VOLUME)

View File

@ -1,5 +1,8 @@
class YouTubePlayer extends Player window.YouTubePlayer = class YouTubePlayer extends Player
constructor: (data) -> constructor: (data) ->
if not (this instanceof YouTubePlayer)
return new YouTubePlayer(data)
@setMediaProperties(data) @setMediaProperties(data)
@qualityRaceCondition = true @qualityRaceCondition = true
@pauseSeekRaceCondition = false @pauseSeekRaceCondition = false
@ -24,7 +27,7 @@ class YouTubePlayer extends Player
) )
load: (data) -> load: (data) ->
super(data) @setMediaProperties(data)
if @yt if @yt
@yt.loadVideoById(data.id, data.currentTime) @yt.loadVideoById(data.id, data.currentTime)
@qualityRaceCondition = true @qualityRaceCondition = true
@ -32,7 +35,7 @@ class YouTubePlayer extends Player
@yt.setPlaybackQuality(USEROPTS.default_quality) @yt.setPlaybackQuality(USEROPTS.default_quality)
onReady: -> onReady: ->
@yt.setVolume(VOLUME) @setVolume(VOLUME)
onStateChange: (ev) -> onStateChange: (ev) ->
# For some reason setting the quality doesn't work # For some reason setting the quality doesn't work
@ -93,5 +96,3 @@ class YouTubePlayer extends Player
cb(@yt.getVolume() / 100) cb(@yt.getVolume() / 100)
else else
cb(VOLUME) cb(VOLUME)
window.YouTubePlayer = YouTubePlayer

View File

@ -824,20 +824,27 @@ Callbacks = {
return; return;
} }
/* Failsafe */ // Failsafe
if (isNaN(VOLUME) || VOLUME > 1 || VOLUME < 0) { if (isNaN(VOLUME) || VOLUME > 1 || VOLUME < 0) {
VOLUME = 1; VOLUME = 1;
} }
var shouldResize = $("#ytapiplayer").html() === ""; // Persist the user's volume preference from the the player, if possible
if (PLAYER && typeof PLAYER.getVolume === "function") { if (PLAYER && typeof PLAYER.getVolume === "function") {
var name = PLAYER.__proto__.constructor.name;
PLAYER.getVolume(function (v) { PLAYER.getVolume(function (v) {
console.log(name, v)
if (typeof v === "number") { if (typeof v === "number") {
if (v < 0 || v > 1) { if (v < 0 || v > 1) {
// Dailymotion's API was wrong once and caused a huge
// headache. This alert is here to make debugging easier.
alert("Something went wrong with retrieving the volume. " + alert("Something went wrong with retrieving the volume. " +
"Please tell calzoneman the following: " + "Please tell calzoneman the following: " +
JSON.stringify({ v: v, t: PLAYER.type, i: PLAYER.videoId })); JSON.stringify({
v: v,
t: PLAYER.mediaType,
i: PLAYER.mediaId
}));
} else { } else {
VOLUME = v; VOLUME = v;
setOpt("volume", VOLUME); setOpt("volume", VOLUME);
@ -846,42 +853,21 @@ Callbacks = {
}); });
} }
if (CHANNEL.opts.allow_voteskip) // Reset voteskip since the video changed
if (CHANNEL.opts.allow_voteskip) {
$("#voteskip").attr("disabled", false); $("#voteskip").attr("disabled", false);
}
$("#currenttitle").text("Currently Playing: " + data.title); $("#currenttitle").text("Currently Playing: " + data.title);
if (data.type != "sc" && PLAYER.type == "sc") // TODO: fix this
// [](/goddamnitmango) setTimeout(function () {
fixSoundcloudShit(); if (data.type !== PLAYER.mediaType) {
loadMediaPlayer(data);
}
if (data.type != "jw" && PLAYER.type == "jw") { handleMediaUpdate(data);
// Is it so hard to not mess up my DOM? }, 100);
$("<div/>").attr("id", "ytapiplayer")
.insertBefore($("#ytapiplayer_wrapper"));
$("#ytapiplayer_wrapper").remove();
}
if (data.type === "fi") {
data.url = data.id;
}
if (NO_VIMEO && data.type === "vi" && data.meta.direct) {
data = vimeoSimulator2014(data);
}
/*
* Google Docs now uses the same simulator as Google+
*/
if (data.type === "gp" || data.type === "gd") {
data = googlePlusSimulator2014(data);
}
if (data.type != PLAYER.type) {
loadMediaPlayer(data);
}
handleMediaUpdate(data);
}, },
mediaUpdate: function(data) { mediaUpdate: function(data) {

View File

@ -1,10 +1,13 @@
(function() { (function() {
var Player, TYPE_MAP, VideoJSPlayer, YouTubePlayer, var Player, TYPE_MAP, VimeoPlayer, YouTubePlayer,
extend = function(child, parent) { for (var key in parent) { if (hasProp.call(parent, key)) child[key] = parent[key]; } function ctor() { this.constructor = child; } ctor.prototype = parent.prototype; child.prototype = new ctor(); child.__super__ = parent.prototype; return child; }, extend = function(child, parent) { for (var key in parent) { if (hasProp.call(parent, key)) child[key] = parent[key]; } function ctor() { this.constructor = child; } ctor.prototype = parent.prototype; child.prototype = new ctor(); child.__super__ = parent.prototype; return child; },
hasProp = {}.hasOwnProperty; hasProp = {}.hasOwnProperty;
Player = (function() { window.Player = Player = (function() {
function Player(data) { function Player(data) {
if (!(this instanceof Player)) {
return new Player(data);
}
this.setMediaProperties(data); this.setMediaProperties(data);
this.paused = false; this.paused = false;
} }
@ -47,104 +50,113 @@
})(); })();
window.Player = Player; window.VimeoPlayer = VimeoPlayer = (function(superClass) {
extend(VimeoPlayer, superClass);
window.removeOld = function(replace) { function VimeoPlayer(data) {
var old; if (!(this instanceof VimeoPlayer)) {
$('#sc_volume').remove(); return new VimeoPlayer(data);
if (replace == null) {
replace = $('<div/>').addClass('embed-responsive-item');
}
old = $('#ytapiplayer');
replace.insertBefore(old);
old.remove();
replace.attr('id', 'ytapiplayer');
return replace;
};
TYPE_MAP = {
yt: 'YouTubePlayer'
};
window.loadMediaPlayer = function(data) {
var ctor;
if (data.type in TYPE_MAP) {
ctor = window[TYPE_MAP[data.type]];
return window.PLAYER = new ctor(data);
}
};
window.handleMediaUpdate = function(data) {
var PLAYER, waiting;
PLAYER = window.PLAYER;
if (typeof PLAYER.mediaLength === 'number' && PLAYER.mediaLength > 0 && data.currentTime > PLAYER.mediaLength) {
return;
}
waiting = data.currentTime < 0;
if (data.id && data.id !== PLAYER.mediaId) {
if (data.currentTime < 0) {
data.currentTime = 0;
} }
PLAYER.load(data); this.load(data);
PLAYER.play();
} }
if (waiting) {
console.log('waiting');
if (PLAYER.type === 'yt') {
PLAYER.pauseSeekRaceCondition = true;
} else {
PLAYER.seekTo(0);
PLAYER.pause();
}
} else if (PLAYER.type === 'yt') {
PLAYER.pauseSeekRaceCondition = false;
}
if (CLIENT.leader || !USEROPTS.synch) {
return;
}
if (data.paused && !PLAYER.paused) {
PLAYER.seekTo(data.currentTime);
PLAYER.pause();
} else if (PLAYER.paused) {
PLAYER.play();
}
return PLAYER.getTime(function(seconds) {
var accuracy, diff, time;
time = data.currentTime;
diff = (time - seconds) || time;
accuracy = USEROPTS.sync_accuracy;
if (PLAYER.type === 'dm') {
accuracy = Math.max(accuracy, 5);
}
if (diff > accuracy) {
return PLAYER.seekTo(time);
} else if (diff < -accuracy) {
if (PLAYER.type !== 'dm') {
time += 1;
}
return PLAYER.seekTo(time);
}
});
};
VideoJSPlayer = (function(superClass) { VimeoPlayer.prototype.load = function(data) {
extend(VideoJSPlayer, superClass); this.setMediaProperties(data);
return waitUntilDefined(window, '$f', (function(_this) {
function VideoJSPlayer(data) {} return function() {
var video;
VideoJSPlayer.prototype.load = function(data) { video = $('<iframe/>');
var video; removeOld(video);
return video = $('<video/>').addClass('video-js vjs-default-skin embed-responsive-item'); video.attr({
src: "https://player.vimeo.com/video/" + data.id + "?api=1&player_id=ytapiplayer",
webkitallowfullscreen: true,
mozallowfullscreen: true,
allowfullscreen: true
});
if (USEROPTS.wmode_transparent) {
video.attr('wmode', 'transparent');
}
return $f(video[0]).addEvent('ready', function() {
_this.vimeo = $f(video[0]);
_this.play();
_this.vimeo.addEvent('finish', function() {
if (CLIENT.leader) {
return socket.emit('playNext');
}
});
_this.vimeo.addEvent('pause', function() {
_this.paused = true;
if (CLIENT.leader) {
return sendVideoUpdate();
}
});
_this.vimeo.addEvent('play', function() {
_this.paused = false;
if (CLIENT.leader) {
return sendVideoUpdate();
}
});
return _this.setVolume(VOLUME);
});
};
})(this));
}; };
return VideoJSPlayer; VimeoPlayer.prototype.play = function() {
this.paused = false;
if (this.vimeo) {
return this.vimeo.api('play');
}
};
VimeoPlayer.prototype.pause = function() {
this.paused = true;
if (this.vimeo) {
return this.vimeo.api('pause');
}
};
VimeoPlayer.prototype.seekTo = function(time) {
if (this.vimeo) {
return this.vimeo.api('seekTo', time);
}
};
VimeoPlayer.prototype.setVolume = function(volume) {
if (this.vimeo) {
return this.vimeo.api('setVolume', volume);
}
};
VimeoPlayer.prototype.getTime = function(cb) {
if (this.vimeo) {
return this.vimeo.api('getCurrentTime', function(time) {
return cb(parseFloat(time));
});
} else {
return cb(0);
}
};
VimeoPlayer.prototype.getVolume = function(cb) {
if (this.vimeo) {
return this.vimeo.api('getVolume', cb);
} else {
return cb(VOLUME);
}
};
return VimeoPlayer;
})(Player); })(Player);
YouTubePlayer = (function(superClass) { window.YouTubePlayer = YouTubePlayer = (function(superClass) {
extend(YouTubePlayer, superClass); extend(YouTubePlayer, superClass);
function YouTubePlayer(data) { function YouTubePlayer(data) {
if (!(this instanceof YouTubePlayer)) {
return new YouTubePlayer(data);
}
this.setMediaProperties(data); this.setMediaProperties(data);
this.qualityRaceCondition = true; this.qualityRaceCondition = true;
this.pauseSeekRaceCondition = false; this.pauseSeekRaceCondition = false;
@ -155,7 +167,6 @@
wmode = USEROPTS.wmode_transparent ? 'transparent' : 'opaque'; wmode = USEROPTS.wmode_transparent ? 'transparent' : 'opaque';
return _this.yt = new YT.Player('ytapiplayer', { return _this.yt = new YT.Player('ytapiplayer', {
videoId: data.id, videoId: data.id,
startSeconds: data.currentTime,
playerVars: { playerVars: {
autohide: 1, autohide: 1,
autoplay: 1, autoplay: 1,
@ -174,7 +185,7 @@
} }
YouTubePlayer.prototype.load = function(data) { YouTubePlayer.prototype.load = function(data) {
YouTubePlayer.__super__.load.call(this, data); this.setMediaProperties(data);
if (this.yt) { if (this.yt) {
this.yt.loadVideoById(data.id, data.currentTime); this.yt.loadVideoById(data.id, data.currentTime);
this.qualityRaceCondition = true; this.qualityRaceCondition = true;
@ -185,7 +196,7 @@
}; };
YouTubePlayer.prototype.onReady = function() { YouTubePlayer.prototype.onReady = function() {
return this.yt.setVolume(VOLUME); return this.setVolume(VOLUME);
}; };
YouTubePlayer.prototype.onStateChange = function(ev) { YouTubePlayer.prototype.onStateChange = function(ev) {
@ -263,6 +274,88 @@
})(Player); })(Player);
window.YouTubePlayer = YouTubePlayer; TYPE_MAP = {
yt: YouTubePlayer,
vi: VimeoPlayer
};
window.loadMediaPlayer = function(data) {
var e;
if (data.type in TYPE_MAP) {
console.log(data);
try {
window.PLAYER = TYPE_MAP[data.type](data);
} catch (_error) {
e = _error;
console.error(e);
}
return console.log(window.PLAYER);
}
};
window.handleMediaUpdate = function(data) {
var PLAYER, waiting;
PLAYER = window.PLAYER;
if (typeof PLAYER.mediaLength === 'number' && PLAYER.mediaLength > 0 && data.currentTime > PLAYER.mediaLength) {
return;
}
waiting = data.currentTime < 0;
if (data.id && data.id !== PLAYER.mediaId) {
if (data.currentTime < 0) {
data.currentTime = 0;
}
PLAYER.load(data);
PLAYER.play();
}
if (waiting) {
if (PLAYER instanceof YouTubePlayer) {
PLAYER.pauseSeekRaceCondition = true;
} else {
PLAYER.seekTo(0);
PLAYER.pause();
}
} else if (PLAYER instanceof YouTubePlayer) {
PLAYER.pauseSeekRaceCondition = false;
}
if (CLIENT.leader || !USEROPTS.synch) {
return;
}
if (data.paused && !PLAYER.paused) {
PLAYER.seekTo(data.currentTime);
PLAYER.pause();
} else if (PLAYER.paused) {
PLAYER.play();
}
return PLAYER.getTime(function(seconds) {
var accuracy, diff, time;
time = data.currentTime;
diff = (time - seconds) || time;
accuracy = USEROPTS.sync_accuracy;
if (PLAYER.mediaType === 'dm') {
accuracy = Math.max(accuracy, 5);
}
if (diff > accuracy) {
return PLAYER.seekTo(time);
} else if (diff < -accuracy) {
if (PLAYER.mediaType !== 'dm') {
time += 1;
}
return PLAYER.seekTo(time);
}
});
};
window.removeOld = function(replace) {
var old;
$('#sc_volume').remove();
if (replace == null) {
replace = $('<div/>').addClass('embed-responsive-item');
}
old = $('#ytapiplayer');
replace.insertBefore(old);
old.remove();
replace.attr('id', 'ytapiplayer');
return replace;
};
}).call(this); }).call(this);