mirror of https://github.com/calzoneman/sync.git
Continue working on YouTube player
This commit is contained in:
parent
d77497aaa7
commit
ae899fd9be
|
@ -39,3 +39,81 @@ window.removeOld = (replace) ->
|
||||||
replace.insertBefore(old)
|
replace.insertBefore(old)
|
||||||
old.remove()
|
old.remove()
|
||||||
replace.attr('id', 'ytapiplayer')
|
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)
|
||||||
|
)
|
||||||
|
|
|
@ -2,7 +2,7 @@ class YouTubePlayer extends Player
|
||||||
constructor: (data) ->
|
constructor: (data) ->
|
||||||
@setMediaProperties(data)
|
@setMediaProperties(data)
|
||||||
@qualityRaceCondition = true
|
@qualityRaceCondition = true
|
||||||
@pauseSeekRaceCondition = true
|
@pauseSeekRaceCondition = false
|
||||||
|
|
||||||
waitUntilDefined(window, 'YT', =>
|
waitUntilDefined(window, 'YT', =>
|
||||||
removeOld()
|
removeOld()
|
||||||
|
@ -39,6 +39,7 @@ class YouTubePlayer extends Player
|
||||||
# until the first event has fired.
|
# until the first event has fired.
|
||||||
if @qualityRaceCondition
|
if @qualityRaceCondition
|
||||||
@qualityRaceCondition = false
|
@qualityRaceCondition = false
|
||||||
|
if USEROPTS.default_quality
|
||||||
@yt.setPlaybackQuality(USEROPTS.default_quality)
|
@yt.setPlaybackQuality(USEROPTS.default_quality)
|
||||||
|
|
||||||
# Similar to above, if you pause the video before the first PLAYING
|
# Similar to above, if you pause the video before the first PLAYING
|
||||||
|
@ -57,12 +58,12 @@ class YouTubePlayer extends Player
|
||||||
socket.emit('playNext')
|
socket.emit('playNext')
|
||||||
|
|
||||||
play: ->
|
play: ->
|
||||||
super()
|
@paused = false
|
||||||
if @yt
|
if @yt
|
||||||
@yt.playVideo()
|
@yt.playVideo()
|
||||||
|
|
||||||
pause: ->
|
pause: ->
|
||||||
super()
|
@paused = true
|
||||||
if @yt
|
if @yt
|
||||||
@yt.pauseVideo()
|
@yt.pauseVideo()
|
||||||
|
|
||||||
|
@ -81,10 +82,16 @@ class YouTubePlayer extends Player
|
||||||
getTime: (cb) ->
|
getTime: (cb) ->
|
||||||
if @yt
|
if @yt
|
||||||
cb(@yt.getCurrentTime())
|
cb(@yt.getCurrentTime())
|
||||||
|
else
|
||||||
|
cb(0)
|
||||||
|
|
||||||
getVolume: (cb) ->
|
getVolume: (cb) ->
|
||||||
if @yt
|
if @yt
|
||||||
if @yt.isMuted()
|
if @yt.isMuted()
|
||||||
return 0
|
cb(0)
|
||||||
else
|
else
|
||||||
return @yt.getVolume() / 100.0
|
cb(@yt.getVolume() / 100)
|
||||||
|
else
|
||||||
|
cb(VOLUME)
|
||||||
|
|
||||||
|
window.YouTubePlayer = YouTubePlayer
|
||||||
|
|
|
@ -1097,7 +1097,7 @@ setupCallbacks = function() {
|
||||||
Callbacks[key](data);
|
Callbacks[key](data);
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
if (SOCKET_DEBUG) {
|
if (SOCKET_DEBUG) {
|
||||||
console.log("EXCEPTION: " + e.stack);
|
console.log("EXCEPTION: " + e + "\n" + e.stack);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
(function() {
|
(function() {
|
||||||
var Player, VideoJSPlayer, YouTubePlayer,
|
var Player, TYPE_MAP, VideoJSPlayer, 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;
|
||||||
|
|
||||||
|
@ -58,7 +58,73 @@
|
||||||
old = $('#ytapiplayer');
|
old = $('#ytapiplayer');
|
||||||
replace.insertBefore(old);
|
replace.insertBefore(old);
|
||||||
old.remove();
|
old.remove();
|
||||||
return replace.attr('id', 'ytapiplayer');
|
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);
|
||||||
|
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) {
|
VideoJSPlayer = (function(superClass) {
|
||||||
|
@ -81,7 +147,7 @@
|
||||||
function YouTubePlayer(data) {
|
function YouTubePlayer(data) {
|
||||||
this.setMediaProperties(data);
|
this.setMediaProperties(data);
|
||||||
this.qualityRaceCondition = true;
|
this.qualityRaceCondition = true;
|
||||||
this.pauseSeekRaceCondition = true;
|
this.pauseSeekRaceCondition = false;
|
||||||
waitUntilDefined(window, 'YT', (function(_this) {
|
waitUntilDefined(window, 'YT', (function(_this) {
|
||||||
return function() {
|
return function() {
|
||||||
var wmode;
|
var wmode;
|
||||||
|
@ -89,6 +155,7 @@
|
||||||
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,
|
||||||
|
@ -124,8 +191,10 @@
|
||||||
YouTubePlayer.prototype.onStateChange = function(ev) {
|
YouTubePlayer.prototype.onStateChange = function(ev) {
|
||||||
if (this.qualityRaceCondition) {
|
if (this.qualityRaceCondition) {
|
||||||
this.qualityRaceCondition = false;
|
this.qualityRaceCondition = false;
|
||||||
|
if (USEROPTS.default_quality) {
|
||||||
this.yt.setPlaybackQuality(USEROPTS.default_quality);
|
this.yt.setPlaybackQuality(USEROPTS.default_quality);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
if (ev.data === YT.PlayerState.PLAYING && this.pauseSeekRaceCondition) {
|
if (ev.data === YT.PlayerState.PLAYING && this.pauseSeekRaceCondition) {
|
||||||
this.pause();
|
this.pause();
|
||||||
this.pauseSeekRaceCondition = false;
|
this.pauseSeekRaceCondition = false;
|
||||||
|
@ -142,14 +211,14 @@
|
||||||
};
|
};
|
||||||
|
|
||||||
YouTubePlayer.prototype.play = function() {
|
YouTubePlayer.prototype.play = function() {
|
||||||
YouTubePlayer.__super__.play.call(this);
|
this.paused = false;
|
||||||
if (this.yt) {
|
if (this.yt) {
|
||||||
return this.yt.playVideo();
|
return this.yt.playVideo();
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
YouTubePlayer.prototype.pause = function() {
|
YouTubePlayer.prototype.pause = function() {
|
||||||
YouTubePlayer.__super__.pause.call(this);
|
this.paused = true;
|
||||||
if (this.yt) {
|
if (this.yt) {
|
||||||
return this.yt.pauseVideo();
|
return this.yt.pauseVideo();
|
||||||
}
|
}
|
||||||
|
@ -173,16 +242,20 @@
|
||||||
YouTubePlayer.prototype.getTime = function(cb) {
|
YouTubePlayer.prototype.getTime = function(cb) {
|
||||||
if (this.yt) {
|
if (this.yt) {
|
||||||
return cb(this.yt.getCurrentTime());
|
return cb(this.yt.getCurrentTime());
|
||||||
|
} else {
|
||||||
|
return cb(0);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
YouTubePlayer.prototype.getVolume = function(cb) {
|
YouTubePlayer.prototype.getVolume = function(cb) {
|
||||||
if (this.yt) {
|
if (this.yt) {
|
||||||
if (this.yt.isMuted()) {
|
if (this.yt.isMuted()) {
|
||||||
return 0;
|
return cb(0);
|
||||||
} else {
|
} else {
|
||||||
return this.yt.getVolume() / 100.0;
|
return cb(this.yt.getVolume() / 100);
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
return cb(VOLUME);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -190,4 +263,6 @@
|
||||||
|
|
||||||
})(Player);
|
})(Player);
|
||||||
|
|
||||||
|
window.YouTubePlayer = YouTubePlayer;
|
||||||
|
|
||||||
}).call(this);
|
}).call(this);
|
||||||
|
|
Loading…
Reference in New Issue