2014-02-22 22:52:30 -07:00
|
|
|
|
(function (window, chrome, console) {
|
|
|
|
|
|
|
|
|
|
// Based on https://github.com/googlecast/CastVideos-chrome/blob/master/CastVideos.js
|
2014-02-23 10:53:25 -07:00
|
|
|
|
|
2014-02-22 22:52:30 -07:00
|
|
|
|
/**
|
|
|
|
|
* Constants of states for Chromecast device
|
|
|
|
|
**/
|
|
|
|
|
var DEVICE_STATE = {
|
|
|
|
|
'IDLE': 0,
|
|
|
|
|
'ACTIVE': 1,
|
|
|
|
|
'WARNING': 2,
|
|
|
|
|
'ERROR': 3,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Constants of states for CastPlayer
|
|
|
|
|
**/
|
|
|
|
|
var PLAYER_STATE = {
|
|
|
|
|
'IDLE': 'IDLE',
|
|
|
|
|
'LOADING': 'LOADING',
|
|
|
|
|
'LOADED': 'LOADED',
|
|
|
|
|
'PLAYING': 'PLAYING',
|
|
|
|
|
'PAUSED': 'PAUSED',
|
|
|
|
|
'STOPPED': 'STOPPED',
|
|
|
|
|
'SEEKING': 'SEEKING',
|
|
|
|
|
'ERROR': 'ERROR'
|
|
|
|
|
};
|
|
|
|
|
|
2014-04-06 10:53:23 -07:00
|
|
|
|
var PlayerName = 'Chromecast';
|
|
|
|
|
|
2014-02-22 22:52:30 -07:00
|
|
|
|
var CastPlayer = function () {
|
|
|
|
|
|
|
|
|
|
/* device variables */
|
|
|
|
|
// @type {DEVICE_STATE} A state for device
|
|
|
|
|
this.deviceState = DEVICE_STATE.IDLE;
|
|
|
|
|
|
|
|
|
|
/* Cast player variables */
|
|
|
|
|
// @type {Object} a chrome.cast.media.Media object
|
|
|
|
|
this.currentMediaSession = null;
|
|
|
|
|
// @type {Number} volume
|
2014-04-15 20:49:49 -07:00
|
|
|
|
this.currentVolume = 1;
|
2014-04-06 10:53:23 -07:00
|
|
|
|
|
2014-02-22 22:52:30 -07:00
|
|
|
|
// @type {string} a chrome.cast.Session object
|
|
|
|
|
this.session = null;
|
|
|
|
|
// @type {PLAYER_STATE} A state for Cast media player
|
|
|
|
|
this.castPlayerState = PLAYER_STATE.IDLE;
|
|
|
|
|
|
|
|
|
|
// @type {Boolean} Fullscreen mode on/off
|
|
|
|
|
this.fullscreen = false;
|
|
|
|
|
|
|
|
|
|
/* Current media variables */
|
|
|
|
|
// @type {Boolean} Audio on and off
|
|
|
|
|
this.audio = true;
|
|
|
|
|
// @type {Number} A number for current media index
|
|
|
|
|
this.currentMediaIndex = 0;
|
|
|
|
|
// @type {Number} A number for current media time
|
|
|
|
|
this.currentMediaTime = 0;
|
|
|
|
|
// @type {Number} A number for current media duration
|
|
|
|
|
this.currentMediaDuration = -1;
|
|
|
|
|
// @type {Timer} A timer for tracking progress of media
|
|
|
|
|
this.timer = null;
|
|
|
|
|
// @type {Boolean} A boolean to stop timer update of progress when triggered by media status event
|
|
|
|
|
this.progressFlag = true;
|
|
|
|
|
// @type {Number} A number in milliseconds for minimal progress update
|
|
|
|
|
this.timerStep = 1000;
|
|
|
|
|
|
2014-02-23 20:27:13 -07:00
|
|
|
|
this.hasReceivers = false;
|
|
|
|
|
|
2014-04-09 16:58:09 -07:00
|
|
|
|
this.currentMediaOffset = 0;
|
|
|
|
|
|
|
|
|
|
// Progress bar element id
|
|
|
|
|
this.progressBar = "positionSlider";
|
|
|
|
|
|
|
|
|
|
// Timec display element id
|
|
|
|
|
this.duration = "currentTime";
|
|
|
|
|
|
|
|
|
|
// Playback display element id
|
|
|
|
|
this.playback = "playTime";
|
|
|
|
|
|
2014-04-26 21:46:12 -07:00
|
|
|
|
// bind once - commit 2ebffc2271da0bc5e8b13821586aee2a2e3c7753
|
|
|
|
|
this.errorHandler = this.onError.bind(this);
|
|
|
|
|
this.incrementMediaTimeHandler = this.incrementMediaTime.bind(this);
|
|
|
|
|
this.mediaStatusUpdateHandler = this.onMediaStatusUpdate.bind(this);
|
|
|
|
|
|
2014-02-22 22:52:30 -07:00
|
|
|
|
this.initializeCastPlayer();
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Initialize Cast media player
|
|
|
|
|
* Initializes the API. Note that either successCallback and errorCallback will be
|
|
|
|
|
* invoked once the API has finished initialization. The sessionListener and
|
|
|
|
|
* receiverListener may be invoked at any time afterwards, and possibly more than once.
|
|
|
|
|
*/
|
|
|
|
|
CastPlayer.prototype.initializeCastPlayer = function () {
|
|
|
|
|
|
2014-02-23 10:29:02 -07:00
|
|
|
|
if (!chrome) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
2014-02-23 10:53:25 -07:00
|
|
|
|
|
2014-02-22 22:52:30 -07:00
|
|
|
|
if (!chrome.cast || !chrome.cast.isAvailable) {
|
2014-03-30 16:26:16 -07:00
|
|
|
|
|
2014-02-23 10:29:02 -07:00
|
|
|
|
setTimeout(this.initializeCastPlayer.bind(this), 1000);
|
2014-02-22 22:52:30 -07:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-06 10:53:23 -07:00
|
|
|
|
// v1 Id AE4DA10A
|
|
|
|
|
// v2 Id 472F0435
|
2014-04-10 04:12:48 -07:00
|
|
|
|
var applicationID = '472F0435';
|
2014-02-22 22:52:30 -07:00
|
|
|
|
|
|
|
|
|
// request session
|
|
|
|
|
var sessionRequest = new chrome.cast.SessionRequest(applicationID);
|
|
|
|
|
var apiConfig = new chrome.cast.ApiConfig(sessionRequest,
|
|
|
|
|
this.sessionListener.bind(this),
|
|
|
|
|
this.receiverListener.bind(this));
|
|
|
|
|
|
2014-04-06 10:53:23 -07:00
|
|
|
|
console.log('chromecast.initialize');
|
2014-02-23 10:53:25 -07:00
|
|
|
|
|
2014-04-26 21:46:12 -07:00
|
|
|
|
chrome.cast.initialize(apiConfig, this.onInitSuccess.bind(this), this.errorHandler);
|
|
|
|
|
|
2014-02-22 22:52:30 -07:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Callback function for init success
|
|
|
|
|
*/
|
|
|
|
|
CastPlayer.prototype.onInitSuccess = function () {
|
2014-04-06 10:53:23 -07:00
|
|
|
|
this.isInitialized = true;
|
|
|
|
|
console.log("chromecast init success");
|
2014-02-22 22:52:30 -07:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Generic error callback function
|
|
|
|
|
*/
|
|
|
|
|
CastPlayer.prototype.onError = function () {
|
2014-03-29 12:23:47 -07:00
|
|
|
|
console.log("chromecast error");
|
2014-02-22 22:52:30 -07:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @param {!Object} e A new session
|
|
|
|
|
* This handles auto-join when a page is reloaded
|
|
|
|
|
* When active session is detected, playback will automatically
|
|
|
|
|
* join existing session and occur in Cast mode and media
|
|
|
|
|
* status gets synced up with current media of the session
|
|
|
|
|
*/
|
2014-02-23 10:53:25 -07:00
|
|
|
|
CastPlayer.prototype.sessionListener = function (e) {
|
2014-02-22 22:52:30 -07:00
|
|
|
|
this.session = e;
|
|
|
|
|
if (this.session) {
|
|
|
|
|
this.deviceState = DEVICE_STATE.ACTIVE;
|
2014-04-06 10:53:23 -07:00
|
|
|
|
MediaController.setActivePlayer(PlayerName);
|
2014-02-22 22:52:30 -07:00
|
|
|
|
if (this.session.media[0]) {
|
|
|
|
|
this.onMediaDiscovered('activeSession', this.session.media[0]);
|
|
|
|
|
}
|
2014-04-06 10:53:23 -07:00
|
|
|
|
|
2014-03-30 16:26:16 -07:00
|
|
|
|
this.session.addUpdateListener(this.sessionUpdateListener.bind(this));
|
2014-02-22 22:52:30 -07:00
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @param {string} e Receiver availability
|
|
|
|
|
* This indicates availability of receivers but
|
|
|
|
|
* does not provide a list of device IDs
|
|
|
|
|
*/
|
|
|
|
|
CastPlayer.prototype.receiverListener = function (e) {
|
2014-03-30 16:26:16 -07:00
|
|
|
|
|
2014-02-22 22:52:30 -07:00
|
|
|
|
if (e === 'available') {
|
2014-03-29 12:23:47 -07:00
|
|
|
|
console.log("chromecast receiver found");
|
2014-02-23 20:27:13 -07:00
|
|
|
|
this.hasReceivers = true;
|
2014-02-22 22:52:30 -07:00
|
|
|
|
}
|
|
|
|
|
else {
|
2014-03-29 12:23:47 -07:00
|
|
|
|
console.log("chromecast receiver list empty");
|
2014-02-23 20:27:13 -07:00
|
|
|
|
this.hasReceivers = false;
|
2014-02-22 22:52:30 -07:00
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2014-03-30 16:26:16 -07:00
|
|
|
|
/**
|
|
|
|
|
* session update listener
|
|
|
|
|
*/
|
|
|
|
|
CastPlayer.prototype.sessionUpdateListener = function (isAlive) {
|
|
|
|
|
if (!isAlive) {
|
|
|
|
|
this.session = null;
|
|
|
|
|
this.deviceState = DEVICE_STATE.IDLE;
|
|
|
|
|
this.castPlayerState = PLAYER_STATE.IDLE;
|
|
|
|
|
this.currentMediaSession = null;
|
|
|
|
|
clearInterval(this.timer);
|
|
|
|
|
|
2014-04-06 10:53:23 -07:00
|
|
|
|
MediaController.removeActivePlayer(PlayerName);
|
2014-03-30 16:26:16 -07:00
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2014-02-22 22:52:30 -07:00
|
|
|
|
/**
|
|
|
|
|
* Requests that a receiver application session be created or joined. By default, the SessionRequest
|
|
|
|
|
* passed to the API at initialization time is used; this may be overridden by passing a different
|
|
|
|
|
* session request in opt_sessionRequest.
|
|
|
|
|
*/
|
|
|
|
|
CastPlayer.prototype.launchApp = function () {
|
2014-04-06 10:53:23 -07:00
|
|
|
|
console.log("chromecast launching app...");
|
2014-02-22 22:52:30 -07:00
|
|
|
|
chrome.cast.requestSession(this.onRequestSessionSuccess.bind(this), this.onLaunchError.bind(this));
|
|
|
|
|
if (this.timer) {
|
|
|
|
|
clearInterval(this.timer);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Callback function for request session success
|
|
|
|
|
* @param {Object} e A chrome.cast.Session object
|
|
|
|
|
*/
|
|
|
|
|
CastPlayer.prototype.onRequestSessionSuccess = function (e) {
|
2014-04-06 10:53:23 -07:00
|
|
|
|
console.log("chromecast session success: " + e.sessionId);
|
2014-02-22 22:52:30 -07:00
|
|
|
|
this.session = e;
|
|
|
|
|
this.deviceState = DEVICE_STATE.ACTIVE;
|
2014-03-30 16:26:16 -07:00
|
|
|
|
this.session.addUpdateListener(this.sessionUpdateListener.bind(this));
|
2014-02-22 22:52:30 -07:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Callback function for launch error
|
|
|
|
|
*/
|
|
|
|
|
CastPlayer.prototype.onLaunchError = function () {
|
2014-04-06 10:53:23 -07:00
|
|
|
|
console.log("chromecast launch error");
|
2014-02-22 22:52:30 -07:00
|
|
|
|
this.deviceState = DEVICE_STATE.ERROR;
|
2014-02-23 13:35:58 -07:00
|
|
|
|
Dashboard.alert({
|
|
|
|
|
|
|
|
|
|
title: "Error Launching Chromecast",
|
2014-04-22 10:25:54 -07:00
|
|
|
|
message: "There was an error launching chromecast. Please ensure your device is connected to your wireless network."
|
2014-02-23 13:35:58 -07:00
|
|
|
|
|
|
|
|
|
});
|
2014-04-05 10:01:04 -07:00
|
|
|
|
|
2014-04-06 10:53:23 -07:00
|
|
|
|
MediaController.removeActivePlayer(PlayerName);
|
2014-02-22 22:52:30 -07:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Stops the running receiver application associated with the session.
|
|
|
|
|
*/
|
|
|
|
|
CastPlayer.prototype.stopApp = function () {
|
|
|
|
|
this.session.stop(this.onStopAppSuccess.bind(this, 'Session stopped'),
|
2014-04-26 21:46:12 -07:00
|
|
|
|
this.errorHandler);
|
2014-02-22 22:52:30 -07:00
|
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Callback function for stop app success
|
|
|
|
|
*/
|
|
|
|
|
CastPlayer.prototype.onStopAppSuccess = function (message) {
|
|
|
|
|
console.log(message);
|
|
|
|
|
this.deviceState = DEVICE_STATE.IDLE;
|
|
|
|
|
this.castPlayerState = PLAYER_STATE.IDLE;
|
|
|
|
|
this.currentMediaSession = null;
|
|
|
|
|
clearInterval(this.timer);
|
|
|
|
|
};
|
|
|
|
|
|
2014-04-15 20:49:49 -07:00
|
|
|
|
/**
|
|
|
|
|
* Loads media into a running receiver application
|
|
|
|
|
* @param {Number} mediaIndex An index number to indicate current media content
|
|
|
|
|
*/
|
|
|
|
|
CastPlayer.prototype.loadMedia = function (user, item, startTimeTicks, mediaSourceId, audioStreamIndex, subtitleStreamIndex) {
|
|
|
|
|
|
|
|
|
|
if (!this.session) {
|
|
|
|
|
console.log("no session");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
this.currentMediaOffset = startTimeTicks || 0;
|
|
|
|
|
|
|
|
|
|
var maxBitrate = 12000000;
|
|
|
|
|
var mediaInfo = getMediaSourceInfo(user, item, maxBitrate, mediaSourceId, audioStreamIndex, subtitleStreamIndex);
|
|
|
|
|
|
|
|
|
|
var streamUrl = getStreamUrl(item, mediaInfo, startTimeTicks, maxBitrate);
|
|
|
|
|
|
|
|
|
|
var castMediaInfo = new chrome.cast.media.MediaInfo(streamUrl);
|
|
|
|
|
|
|
|
|
|
castMediaInfo.customData = getCustomData(item, mediaInfo.mediaSource.Id, startTimeTicks);
|
|
|
|
|
castMediaInfo.metadata = getMetadata(item);
|
|
|
|
|
|
|
|
|
|
if (mediaInfo.streamContainer == 'm3u8') {
|
|
|
|
|
castMediaInfo.contentType = 'application/x-mpegURL';
|
|
|
|
|
} else {
|
|
|
|
|
castMediaInfo.contentType = item.MediaType.toLowerCase() + '/' + mediaInfo.streamContainer.toLowerCase();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
castMediaInfo.streamType = mediaInfo.isStatic ? chrome.cast.media.StreamType.BUFFERED : chrome.cast.media.StreamType.LIVE;
|
|
|
|
|
|
|
|
|
|
var request = new chrome.cast.media.LoadRequest(castMediaInfo);
|
|
|
|
|
request.autoplay = true;
|
2014-04-18 12:59:06 -07:00
|
|
|
|
request.currentTime = startTimeTicks ? startTimeTicks / 10000000 : 0;
|
2014-04-15 20:49:49 -07:00
|
|
|
|
|
|
|
|
|
this.castPlayerState = PLAYER_STATE.LOADING;
|
|
|
|
|
this.session.loadMedia(request,
|
|
|
|
|
this.onMediaDiscovered.bind(this, 'loadMedia'),
|
|
|
|
|
this.onLoadMediaError.bind(this));
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Callback function for loadMedia success
|
|
|
|
|
* @param {Object} mediaSession A new media object.
|
|
|
|
|
*/
|
|
|
|
|
CastPlayer.prototype.onMediaDiscovered = function (how, mediaSession) {
|
|
|
|
|
|
|
|
|
|
console.log("chromecast new media session ID:" + mediaSession.mediaSessionId + ' (' + how + ')');
|
|
|
|
|
this.currentMediaSession = mediaSession;
|
2014-04-18 12:59:06 -07:00
|
|
|
|
this.currentMediaTime = mediaSession.currentTime;
|
2014-04-15 20:49:49 -07:00
|
|
|
|
|
|
|
|
|
if (how == 'loadMedia') {
|
|
|
|
|
this.castPlayerState = PLAYER_STATE.PLAYING;
|
|
|
|
|
clearInterval(this.timer);
|
2014-04-26 21:46:12 -07:00
|
|
|
|
this.startProgressTimer();
|
2014-04-15 20:49:49 -07:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (how == 'activeSession') {
|
2014-04-18 12:59:06 -07:00
|
|
|
|
this.castPlayerState = mediaSession.playerState;
|
2014-04-15 20:49:49 -07:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (this.castPlayerState == PLAYER_STATE.PLAYING) {
|
|
|
|
|
// start progress timer
|
2014-04-26 21:46:12 -07:00
|
|
|
|
this.startProgressTimer();
|
2014-04-15 20:49:49 -07:00
|
|
|
|
}
|
|
|
|
|
|
2014-04-26 21:46:12 -07:00
|
|
|
|
this.currentMediaSession.addUpdateListener(this.mediaStatusUpdateHandler);
|
|
|
|
|
this.currentMediaDuration = mediaSession.media.duration * 10000000;
|
2014-04-15 20:49:49 -07:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Callback function when media load returns error
|
|
|
|
|
*/
|
|
|
|
|
CastPlayer.prototype.onLoadMediaError = function (e) {
|
|
|
|
|
console.log("chromecast media error");
|
|
|
|
|
this.castPlayerState = PLAYER_STATE.IDLE;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Callback function for media status update from receiver
|
|
|
|
|
* @param {!Boolean} e true/false
|
|
|
|
|
*/
|
|
|
|
|
CastPlayer.prototype.onMediaStatusUpdate = function (e) {
|
|
|
|
|
if (e == false) {
|
|
|
|
|
this.currentMediaTime = 0;
|
|
|
|
|
this.castPlayerState = PLAYER_STATE.IDLE;
|
|
|
|
|
}
|
|
|
|
|
console.log("chromecast updating media");
|
|
|
|
|
this.updateProgressBarByTimer();
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Helper function
|
|
|
|
|
* Increment media current position by 1 second
|
|
|
|
|
*/
|
|
|
|
|
CastPlayer.prototype.incrementMediaTime = function () {
|
|
|
|
|
if (this.castPlayerState == PLAYER_STATE.PLAYING) {
|
|
|
|
|
if (this.currentMediaTime < this.currentMediaDuration) {
|
|
|
|
|
this.currentMediaTime += 1;
|
|
|
|
|
this.updateProgressBarByTimer();
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
this.currentMediaTime = 0;
|
|
|
|
|
clearInterval(this.timer);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Play media in Cast mode
|
|
|
|
|
*/
|
|
|
|
|
CastPlayer.prototype.playMedia = function () {
|
|
|
|
|
|
|
|
|
|
if (!this.currentMediaSession) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
switch (this.castPlayerState) {
|
|
|
|
|
case PLAYER_STATE.LOADED:
|
|
|
|
|
case PLAYER_STATE.PAUSED:
|
|
|
|
|
this.currentMediaSession.play(null,
|
|
|
|
|
this.mediaCommandSuccessCallback.bind(this, "playing started for " + this.currentMediaSession.sessionId),
|
2014-04-26 21:46:12 -07:00
|
|
|
|
this.errorHandler);
|
|
|
|
|
this.currentMediaSession.addUpdateListener(this.mediaStatusUpdateHandler);
|
2014-04-15 20:49:49 -07:00
|
|
|
|
this.castPlayerState = PLAYER_STATE.PLAYING;
|
|
|
|
|
// start progress timer
|
|
|
|
|
clearInterval(this.timer);
|
2014-04-26 21:46:12 -07:00
|
|
|
|
this.startProgressTimer();
|
2014-04-15 20:49:49 -07:00
|
|
|
|
break;
|
|
|
|
|
case PLAYER_STATE.IDLE:
|
|
|
|
|
case PLAYER_STATE.LOADING:
|
|
|
|
|
case PLAYER_STATE.STOPPED:
|
|
|
|
|
this.loadMedia(this.currentMediaIndex);
|
2014-04-26 21:46:12 -07:00
|
|
|
|
this.currentMediaSession.addUpdateListener(this.mediaStatusUpdateHandler);
|
2014-04-15 20:49:49 -07:00
|
|
|
|
this.castPlayerState = PLAYER_STATE.PLAYING;
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Pause media playback in Cast mode
|
|
|
|
|
*/
|
|
|
|
|
CastPlayer.prototype.pauseMedia = function () {
|
|
|
|
|
|
|
|
|
|
if (!this.currentMediaSession) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (this.castPlayerState == PLAYER_STATE.PLAYING) {
|
|
|
|
|
this.castPlayerState = PLAYER_STATE.PAUSED;
|
|
|
|
|
this.currentMediaSession.pause(null,
|
|
|
|
|
this.mediaCommandSuccessCallback.bind(this, "paused " + this.currentMediaSession.sessionId),
|
2014-04-26 21:46:12 -07:00
|
|
|
|
this.errorHandler);
|
2014-04-15 20:49:49 -07:00
|
|
|
|
clearInterval(this.timer);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Stop CC playback
|
|
|
|
|
*/
|
|
|
|
|
CastPlayer.prototype.stopMedia = function () {
|
|
|
|
|
|
|
|
|
|
if (!this.currentMediaSession) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
this.currentMediaSession.stop(null,
|
|
|
|
|
this.mediaCommandSuccessCallback.bind(this, "stopped " + this.currentMediaSession.sessionId),
|
2014-04-26 21:46:12 -07:00
|
|
|
|
this.errorHandler);
|
2014-04-15 20:49:49 -07:00
|
|
|
|
this.castPlayerState = PLAYER_STATE.STOPPED;
|
|
|
|
|
clearInterval(this.timer);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Set media volume in Cast mode
|
|
|
|
|
* @param {Boolean} mute A boolean
|
|
|
|
|
*/
|
|
|
|
|
CastPlayer.prototype.setReceiverVolume = function (mute, vol) {
|
|
|
|
|
|
|
|
|
|
if (!this.currentMediaSession) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!mute) {
|
|
|
|
|
this.currentVolume = vol || 1;
|
|
|
|
|
this.session.setReceiverVolumeLevel(this.currentVolume,
|
|
|
|
|
this.mediaCommandSuccessCallback.bind(this),
|
2014-04-26 21:46:12 -07:00
|
|
|
|
this.errorHandler);
|
2014-04-15 20:49:49 -07:00
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
this.session.setReceiverMuted(true,
|
|
|
|
|
this.mediaCommandSuccessCallback.bind(this),
|
2014-04-26 21:46:12 -07:00
|
|
|
|
this.errorHandler);
|
2014-04-15 20:49:49 -07:00
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Toggle mute CC
|
|
|
|
|
*/
|
|
|
|
|
CastPlayer.prototype.toggleMute = function () {
|
|
|
|
|
if (this.audio == true) {
|
|
|
|
|
this.mute();
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
this.unMute();
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Mute CC
|
|
|
|
|
*/
|
|
|
|
|
CastPlayer.prototype.mute = function () {
|
|
|
|
|
this.audio = false;
|
|
|
|
|
this.setReceiverVolume(true);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Unmute CC
|
|
|
|
|
*/
|
|
|
|
|
CastPlayer.prototype.unMute = function () {
|
|
|
|
|
this.audio = true;
|
|
|
|
|
this.setReceiverVolume(false);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* media seek function in either Cast or local mode
|
|
|
|
|
* @param {Event} e An event object from seek
|
|
|
|
|
*/
|
|
|
|
|
CastPlayer.prototype.seekMedia = function (event) {
|
|
|
|
|
var pos = parseInt(event);
|
|
|
|
|
|
2014-04-18 12:59:06 -07:00
|
|
|
|
var curr = pos / 10000000;
|
2014-04-15 20:49:49 -07:00
|
|
|
|
|
|
|
|
|
if (this.castPlayerState != PLAYER_STATE.PLAYING && this.castPlayerState != PLAYER_STATE.PAUSED) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
this.currentMediaTime = curr;
|
|
|
|
|
console.log('Seeking ' + this.currentMediaSession.sessionId + ':' +
|
2014-04-18 12:59:06 -07:00
|
|
|
|
this.currentMediaSession.mediaSessionId + ' to ' + curr);
|
2014-04-15 20:49:49 -07:00
|
|
|
|
var request = new chrome.cast.media.SeekRequest();
|
|
|
|
|
request.currentTime = this.currentMediaTime;
|
|
|
|
|
this.currentMediaSession.seek(request,
|
|
|
|
|
this.onSeekSuccess.bind(this, 'media seek done'),
|
2014-04-26 21:46:12 -07:00
|
|
|
|
this.errorHandler);
|
2014-04-15 20:49:49 -07:00
|
|
|
|
this.castPlayerState = PLAYER_STATE.SEEKING;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Callback function for seek success
|
|
|
|
|
* @param {String} info A string that describe seek event
|
|
|
|
|
*/
|
|
|
|
|
CastPlayer.prototype.onSeekSuccess = function (info) {
|
|
|
|
|
console.log(info);
|
|
|
|
|
this.castPlayerState = PLAYER_STATE.PLAYING;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Callback function for media command success
|
|
|
|
|
*/
|
|
|
|
|
CastPlayer.prototype.mediaCommandSuccessCallback = function (info, e) {
|
|
|
|
|
console.log(info);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Update progress bar when there is a media status update
|
|
|
|
|
* @param {Object} e An media status update object
|
|
|
|
|
*/
|
|
|
|
|
CastPlayer.prototype.updateProgressBar = function (e) {
|
|
|
|
|
if (e.idleReason == 'FINISHED' && e.playerState == 'IDLE') {
|
|
|
|
|
clearInterval(this.timer);
|
|
|
|
|
this.castPlayerState = PLAYER_STATE.STOPPED;
|
|
|
|
|
if (e.idleReason == 'FINISHED') {
|
|
|
|
|
$(this).trigger("/playback/complete", e);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
var p = Number(e.currentTime / this.currentMediaSession.media.duration + 1).toFixed(3);
|
|
|
|
|
this.progressFlag = false;
|
|
|
|
|
setTimeout(this.setProgressFlag.bind(this), 1000); // don't update progress in 1 second
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Set progressFlag with a timeout of 1 second to avoid UI update
|
|
|
|
|
* until a media status update from receiver
|
|
|
|
|
*/
|
|
|
|
|
CastPlayer.prototype.setProgressFlag = function () {
|
|
|
|
|
this.progressFlag = true;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Update progress bar based on timer
|
|
|
|
|
*/
|
|
|
|
|
CastPlayer.prototype.updateProgressBarByTimer = function () {
|
2014-04-18 12:59:06 -07:00
|
|
|
|
if (!this.currentMediaTime) {
|
|
|
|
|
this.currentMediaDuration = this.session.media[0].currentTime;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!this.currentMediaDuration) {
|
|
|
|
|
this.currentMediaDuration = this.session.media[0].media.customData.runTimeTicks;
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-15 20:49:49 -07:00
|
|
|
|
var pp = 0;
|
|
|
|
|
if (this.currentMediaDuration > 0) {
|
|
|
|
|
pp = Number(this.currentMediaTime / this.currentMediaDuration).toFixed(3);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (this.progressFlag) {
|
|
|
|
|
// don't update progress if it's been updated on media status update event
|
|
|
|
|
$(this).trigger("/playback/update",
|
|
|
|
|
[{
|
|
|
|
|
positionTicks: this.currentMediaTime * 10000000,
|
2014-04-18 12:59:06 -07:00
|
|
|
|
runtimeTicks: this.currentMediaDuration
|
2014-04-15 20:49:49 -07:00
|
|
|
|
}]);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (pp > 100 || this.castPlayerState == PLAYER_STATE.IDLE) {
|
|
|
|
|
clearInterval(this.timer);
|
|
|
|
|
this.deviceState = DEVICE_STATE.IDLE;
|
|
|
|
|
this.castPlayerState = PLAYER_STATE.IDLE;
|
|
|
|
|
$(this).trigger("/playback/complete", true);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @param {function} A callback function for the fucntion to start timer
|
|
|
|
|
*/
|
2014-04-26 21:46:12 -07:00
|
|
|
|
CastPlayer.prototype.startProgressTimer = function() {
|
2014-04-15 20:49:49 -07:00
|
|
|
|
if(this.timer) {
|
|
|
|
|
clearInterval(this.timer);
|
|
|
|
|
this.timer = null;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// start progress timer
|
2014-04-26 21:46:12 -07:00
|
|
|
|
this.timer = setInterval(this.incrementMediaTimeHandler, this.timerStep);
|
2014-04-15 20:49:49 -07:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
var castPlayer = new CastPlayer();
|
|
|
|
|
|
2014-04-06 11:00:37 -07:00
|
|
|
|
function getCodecLimits() {
|
2014-04-15 20:49:49 -07:00
|
|
|
|
|
2014-04-06 11:00:37 -07:00
|
|
|
|
return {
|
2014-04-06 10:53:23 -07:00
|
|
|
|
|
2014-04-06 11:00:37 -07:00
|
|
|
|
maxVideoAudioChannels: 6,
|
|
|
|
|
maxAudioChannels: 2,
|
|
|
|
|
maxVideoLevel: 41,
|
|
|
|
|
maxWidth: 1920,
|
|
|
|
|
maxHeight: 1080,
|
|
|
|
|
maxSampleRate: 44100
|
2014-04-06 10:53:23 -07:00
|
|
|
|
|
2014-04-06 11:00:37 -07:00
|
|
|
|
};
|
2014-04-06 10:53:23 -07:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function canDirectStream(mediaType, mediaSource, maxBitrate) {
|
|
|
|
|
|
|
|
|
|
// If bitrate is unknown don't direct stream
|
|
|
|
|
if (!mediaSource.Bitrate || mediaSource.Bitrate > maxBitrate) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-06 11:00:37 -07:00
|
|
|
|
var codecLimits = getCodecLimits();
|
|
|
|
|
|
2014-04-06 10:53:23 -07:00
|
|
|
|
if (mediaType == "Audio") {
|
|
|
|
|
|
|
|
|
|
return ['mp3', 'aac'].indexOf(mediaSource.Container || '') != -1;
|
|
|
|
|
}
|
|
|
|
|
else if (mediaType == "Video") {
|
|
|
|
|
|
|
|
|
|
var videoStream = mediaSource.MediaStreams.filter(function (s) {
|
|
|
|
|
|
|
|
|
|
return s.Type == 'Video';
|
|
|
|
|
|
|
|
|
|
})[0];
|
|
|
|
|
|
|
|
|
|
if (!videoStream) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (['high', 'main', 'baseline'].indexOf((videoStream.Profile || '').toLowerCase()) == -1) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-06 11:00:37 -07:00
|
|
|
|
if (!videoStream.Level || videoStream.Level > codecLimits.maxVideoLevel) {
|
2014-04-06 10:53:23 -07:00
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-06 11:00:37 -07:00
|
|
|
|
if (!videoStream.Width || videoStream.Width > codecLimits.maxWidth) {
|
2014-04-06 10:53:23 -07:00
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-06 11:00:37 -07:00
|
|
|
|
if (!videoStream.Height || videoStream.Height > codecLimits.maxHeight) {
|
2014-04-06 10:53:23 -07:00
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return ['mp4'].indexOf(mediaSource.Container || '') != -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
throw new Error('Unrecognized MediaType');
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function canPlayAudioStreamDirect(audioStream, isVideo) {
|
|
|
|
|
|
|
|
|
|
var audioCodec = (audioStream.Codec || '').toLowerCase().replace('-', '');
|
|
|
|
|
|
|
|
|
|
if (audioCodec.indexOf('aac') == -1 &&
|
|
|
|
|
audioCodec.indexOf('mp3') == -1 &&
|
|
|
|
|
audioCodec.indexOf('mpeg') == -1) {
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-06 11:00:37 -07:00
|
|
|
|
var codecLimits = getCodecLimits();
|
|
|
|
|
|
|
|
|
|
var maxChannels = isVideo ? codecLimits.maxVideoAudioChannels : codecLimits.maxAudioChannels;
|
2014-04-06 10:53:23 -07:00
|
|
|
|
|
|
|
|
|
if (!audioStream.Channels || audioStream.Channels > maxChannels) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-06 11:00:37 -07:00
|
|
|
|
if (!audioStream.SampleRate || audioStream.SampleRate > codecLimits.maxSampleRate) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-06 10:53:23 -07:00
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function isSupportedCodec(mediaType, mediaSource) {
|
|
|
|
|
|
|
|
|
|
if (mediaType == "Audio") {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
else if (mediaType == "Video") {
|
|
|
|
|
|
|
|
|
|
return mediaSource.MediaStreams.filter(function (m) {
|
|
|
|
|
|
|
|
|
|
return m.Type == "Video" && (m.Codec || '').toLowerCase() == 'h264';
|
|
|
|
|
|
|
|
|
|
}).length > 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
throw new Error('Unrecognized MediaType');
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function getStreamByIndex(streams, type, index) {
|
|
|
|
|
return streams.filter(function (s) {
|
|
|
|
|
|
|
|
|
|
return s.Type == type && s.Index == index;
|
|
|
|
|
|
|
|
|
|
})[0];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function getDefaultAudioStream(mediaStreams, user) {
|
|
|
|
|
|
|
|
|
|
// Find all audio streams
|
|
|
|
|
var audioStreams = mediaStreams.filter(function (stream) {
|
|
|
|
|
return stream.Type == "Audio";
|
|
|
|
|
|
|
|
|
|
}).sort(function (a, b) {
|
|
|
|
|
|
|
|
|
|
var av = a.IsDefault ? 0 : 1;
|
|
|
|
|
var bv = b.IsDefault ? 0 : 1;
|
|
|
|
|
|
|
|
|
|
return av - bv;
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
if (user.Configuration.AudioLanguagePreference) {
|
|
|
|
|
|
|
|
|
|
for (var i = 0, length = audioStreams.length; i < length; i++) {
|
|
|
|
|
var mediaStream = audioStreams[i];
|
|
|
|
|
|
|
|
|
|
if (mediaStream.Language == user.Configuration.AudioLanguagePreference) {
|
|
|
|
|
return mediaStream.Index;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Just use the first audio stream
|
|
|
|
|
return audioStreams[0];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function getMediaSourceInfo(user, item, maxBitrate, mediaSourceId, audioStreamIndex, subtitleStreamIndex) {
|
|
|
|
|
|
2014-04-15 20:49:49 -07:00
|
|
|
|
var sources = item.MediaSources || [];
|
2014-04-06 10:53:23 -07:00
|
|
|
|
|
|
|
|
|
// If a specific stream was requested, filter the list
|
|
|
|
|
if (mediaSourceId) {
|
|
|
|
|
sources = sources.filter(function (m) {
|
|
|
|
|
|
|
|
|
|
return m.Id == mediaSourceId;
|
|
|
|
|
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Find first one that can be direct streamed
|
|
|
|
|
var source = sources.filter(function (m) {
|
|
|
|
|
|
|
|
|
|
var audioStreams = m.MediaStreams.filter(function (s) {
|
|
|
|
|
return s.Type == 'Audio';
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
var audioStream = mediaSourceId == m.Id && audioStreamIndex != null ? getStreamByIndex(audioStreams, 'Audio', audioStreamIndex) : getDefaultAudioStream(audioStreams, user);
|
|
|
|
|
|
|
|
|
|
if (!audioStream || !canPlayAudioStreamDirect(audioStream, item.MediaType == 'Video')) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var subtitleStream = mediaSourceId == m.Id && subtitleStreamIndex != null ? getStreamByIndex(m.MediaStreams, 'Subtitle', subtitleStreamIndex) : null;
|
|
|
|
|
|
|
|
|
|
if (subtitleStream) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return canDirectStream(item.MediaType, m, maxBitrate, audioStream);
|
|
|
|
|
|
|
|
|
|
})[0];
|
|
|
|
|
|
|
|
|
|
if (source) {
|
|
|
|
|
return {
|
|
|
|
|
mediaSource: source,
|
|
|
|
|
isStatic: true,
|
|
|
|
|
streamContainer: source.Container
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Find first one with supported codec
|
|
|
|
|
source = sources.filter(function (m) {
|
|
|
|
|
|
|
|
|
|
return isSupportedCodec(item.MediaType, m);
|
|
|
|
|
|
|
|
|
|
})[0];
|
|
|
|
|
|
|
|
|
|
// Default to first one
|
|
|
|
|
return {
|
|
|
|
|
mediaSource: source || sources[0],
|
|
|
|
|
isStatic: false,
|
|
|
|
|
streamContainer: item.MediaType == 'Audio' ? 'mp3' : 'm3u8'
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-08 11:32:03 -07:00
|
|
|
|
function getCustomData(item, mediaSourceId, startTimeTicks) {
|
2014-04-06 10:53:23 -07:00
|
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
|
|
|
|
|
serverAddress: ApiClient.serverAddress(),
|
|
|
|
|
itemId: item.Id,
|
|
|
|
|
userId: Dashboard.getCurrentUserId(),
|
|
|
|
|
deviceName: ApiClient.deviceName(),
|
|
|
|
|
//deviceId: ApiClient.deviceId(),
|
2014-04-18 12:59:06 -07:00
|
|
|
|
startTimeTicks: startTimeTicks || 0,
|
|
|
|
|
runTimeTicks: item.RunTimeTicks
|
2014-04-06 10:53:23 -07:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function getMetadata(item) {
|
|
|
|
|
|
|
|
|
|
var metadata = {};
|
|
|
|
|
|
|
|
|
|
if (item.Type == 'Episode') {
|
|
|
|
|
metadata = new chrome.cast.media.TvShowMediaMetadata();
|
|
|
|
|
metadata.type = chrome.cast.media.MetadataType.TV_SHOW;
|
|
|
|
|
|
|
|
|
|
metadata.episodeTitle = item.Name;
|
|
|
|
|
|
|
|
|
|
if (item.PremiereDate) {
|
|
|
|
|
metadata.originalAirdate = parseISO8601Date(item.PremiereDate).toISOString();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
metadata.seriesTitle = item.SeriesName;
|
|
|
|
|
|
|
|
|
|
if (item.IndexNumber != null) {
|
|
|
|
|
metadata.episode = metadata.episodeNumber = item.IndexNumber;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (item.ParentIndexNumber != null) {
|
|
|
|
|
metadata.season = metadata.seasonNumber = item.ParentIndexNumber;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
else if (item.Type == 'Photo') {
|
|
|
|
|
metadata = new chrome.cast.media.PhotoMediaMetadata();
|
|
|
|
|
metadata.type = chrome.cast.media.MetadataType.PHOTO;
|
|
|
|
|
|
|
|
|
|
if (item.PremiereDate) {
|
|
|
|
|
metadata.creationDateTime = parseISO8601Date(item.PremiereDate).toISOString();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
else if (item.MediaType == 'Audio') {
|
|
|
|
|
metadata = new chrome.cast.media.MusicTrackMediaMetadata();
|
|
|
|
|
metadata.type = chrome.cast.media.MetadataType.MUSIC_TRACK;
|
|
|
|
|
|
|
|
|
|
if (item.ProductionYear) {
|
|
|
|
|
metadata.releaseYear = item.ProductionYear;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (item.PremiereDate) {
|
|
|
|
|
metadata.releaseDate = parseISO8601Date(item.PremiereDate).toISOString();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
metadata.songName = item.Name;
|
2014-04-15 20:49:49 -07:00
|
|
|
|
metadata.artist = item.Artists & item.Artists.length ? item.Artists[0] : '';
|
|
|
|
|
metadata.albumArtist = item.AlbumArtist;
|
|
|
|
|
|
|
|
|
|
if (item.IndexNumber != null) {
|
|
|
|
|
metadata.trackNumber = item.IndexNumber;
|
2014-02-23 13:35:58 -07:00
|
|
|
|
}
|
2014-04-15 20:49:49 -07:00
|
|
|
|
|
|
|
|
|
if (item.ParentIndexNumber != null) {
|
|
|
|
|
metadata.discNumber = item.ParentIndexNumber;
|
2014-02-23 13:35:58 -07:00
|
|
|
|
}
|
|
|
|
|
|
2014-04-15 20:49:49 -07:00
|
|
|
|
var composer = (item.People || []).filter(function (p) {
|
|
|
|
|
return p.PersonType == 'Type';
|
|
|
|
|
})[0];
|
2014-04-06 10:53:23 -07:00
|
|
|
|
|
2014-04-15 20:49:49 -07:00
|
|
|
|
if (composer) {
|
|
|
|
|
metadata.composer = composer.Name;
|
|
|
|
|
}
|
2014-02-23 13:35:58 -07:00
|
|
|
|
}
|
|
|
|
|
|
2014-04-15 20:49:49 -07:00
|
|
|
|
else if (item.MediaType == 'Movie') {
|
|
|
|
|
metadata = new chrome.cast.media.MovieMediaMetadata();
|
|
|
|
|
metadata.type = chrome.cast.media.MetadataType.MOVIE;
|
2014-02-23 13:35:58 -07:00
|
|
|
|
|
2014-04-15 20:49:49 -07:00
|
|
|
|
if (item.ProductionYear) {
|
|
|
|
|
metadata.releaseYear = item.ProductionYear;
|
|
|
|
|
}
|
2014-04-06 10:53:23 -07:00
|
|
|
|
|
2014-04-15 20:49:49 -07:00
|
|
|
|
if (item.PremiereDate) {
|
|
|
|
|
metadata.releaseDate = parseISO8601Date(item.PremiereDate).toISOString();
|
|
|
|
|
}
|
2014-02-23 13:35:58 -07:00
|
|
|
|
}
|
|
|
|
|
|
2014-04-15 20:49:49 -07:00
|
|
|
|
else {
|
|
|
|
|
metadata = new chrome.cast.media.GenericMediaMetadata();
|
|
|
|
|
metadata.type = chrome.cast.media.MetadataType.GENERIC;
|
|
|
|
|
|
|
|
|
|
if (item.ProductionYear) {
|
|
|
|
|
metadata.releaseYear = item.ProductionYear;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (item.PremiereDate) {
|
|
|
|
|
metadata.releaseDate = parseISO8601Date(item.PremiereDate).toISOString();
|
|
|
|
|
}
|
2014-02-23 13:35:58 -07:00
|
|
|
|
}
|
|
|
|
|
|
2014-04-15 20:49:49 -07:00
|
|
|
|
metadata.title = item.Name;
|
2014-04-06 10:53:23 -07:00
|
|
|
|
|
2014-04-15 20:49:49 -07:00
|
|
|
|
if (item.Studios && item.Studios.length) {
|
|
|
|
|
metadata.Studio = item.Studios[0];
|
2014-02-23 13:35:58 -07:00
|
|
|
|
}
|
|
|
|
|
|
2014-04-15 20:49:49 -07:00
|
|
|
|
return metadata;
|
|
|
|
|
}
|
2014-02-23 13:35:58 -07:00
|
|
|
|
|
2014-04-15 20:49:49 -07:00
|
|
|
|
function getStreamUrl(item, mediaSourceInfo, startTimeTicks, maxBitrate) {
|
2014-04-06 10:53:23 -07:00
|
|
|
|
|
2014-04-15 20:49:49 -07:00
|
|
|
|
var url;
|
2014-02-23 13:35:58 -07:00
|
|
|
|
|
2014-04-15 20:49:49 -07:00
|
|
|
|
var codecLimits = getCodecLimits();
|
2014-02-23 13:35:58 -07:00
|
|
|
|
|
2014-04-15 20:49:49 -07:00
|
|
|
|
if (item.MediaType == 'Audio') {
|
2014-04-05 10:01:04 -07:00
|
|
|
|
|
2014-04-15 20:49:49 -07:00
|
|
|
|
url = ApiClient.serverAddress() + '/mediabrowser/audio/' + item.Id + '/stream.' + mediaSourceInfo.streamContainer + '?';
|
|
|
|
|
url += '&static=' + mediaSourceInfo.isStatic.toString();
|
|
|
|
|
url += '&maxaudiochannels=' + codecLimits.maxAudioChannels;
|
2014-04-05 10:01:04 -07:00
|
|
|
|
|
2014-04-15 20:49:49 -07:00
|
|
|
|
if (startTimeTicks) {
|
|
|
|
|
url += '&startTimeTicks=' + startTimeTicks.toString();
|
|
|
|
|
}
|
2014-02-23 13:35:58 -07:00
|
|
|
|
|
2014-04-15 20:49:49 -07:00
|
|
|
|
if (maxBitrate) {
|
|
|
|
|
url += '&audiobitrate=' + Math.min(maxBitrate, 320000).toString();
|
|
|
|
|
}
|
2014-02-23 13:35:58 -07:00
|
|
|
|
|
2014-04-15 20:49:49 -07:00
|
|
|
|
url += '&audiosamplerate=' + codecLimits.maxSampleRate;
|
|
|
|
|
url += '&mediasourceid=' + mediaSourceInfo.mediaSource.Id;
|
2014-04-10 08:49:46 -07:00
|
|
|
|
|
2014-04-15 20:49:49 -07:00
|
|
|
|
return url;
|
2014-02-23 13:35:58 -07:00
|
|
|
|
|
|
|
|
|
}
|
2014-04-15 20:49:49 -07:00
|
|
|
|
else if (item.MediaType == 'Video') {
|
2014-02-23 13:35:58 -07:00
|
|
|
|
|
2014-04-15 20:49:49 -07:00
|
|
|
|
url = ApiClient.serverAddress() + '/mediabrowser/videos/' + item.Id + '/stream.' + mediaSourceInfo.streamContainer + '?';
|
|
|
|
|
url += 'static=' + mediaSourceInfo.isStatic.toString();
|
|
|
|
|
url += '&maxaudiochannels=' + codecLimits.maxVideoAudioChannels;
|
2014-02-23 13:35:58 -07:00
|
|
|
|
|
2014-04-15 20:49:49 -07:00
|
|
|
|
if (startTimeTicks) {
|
|
|
|
|
url += '&startTimeTicks=' + startTimeTicks.toString();
|
|
|
|
|
}
|
2014-02-23 13:35:58 -07:00
|
|
|
|
|
2014-04-15 20:49:49 -07:00
|
|
|
|
if (maxBitrate) {
|
2014-02-23 13:35:58 -07:00
|
|
|
|
|
2014-04-15 20:49:49 -07:00
|
|
|
|
var audioRate = 768000;
|
|
|
|
|
url += '&audiobitrate=' + audioRate.toString();
|
|
|
|
|
url += '&videobitrate=' + (maxBitrate - audioRate).toString();
|
|
|
|
|
}
|
2014-02-23 13:35:58 -07:00
|
|
|
|
|
2014-04-15 20:49:49 -07:00
|
|
|
|
url += '&profile=high';
|
|
|
|
|
url += '&level=' + codecLimits.maxVideoLevel;
|
2014-02-23 13:35:58 -07:00
|
|
|
|
|
2014-04-15 20:49:49 -07:00
|
|
|
|
url += '&maxwidth=' + codecLimits.maxWidth;
|
|
|
|
|
url += '&maxheight=' + codecLimits.maxHeight;
|
2014-02-23 13:35:58 -07:00
|
|
|
|
|
2014-04-15 20:49:49 -07:00
|
|
|
|
url += '&videoCodec=h264';
|
|
|
|
|
url += '&audioCodec=aac,mp3';
|
2014-04-09 16:58:09 -07:00
|
|
|
|
|
2014-04-15 20:49:49 -07:00
|
|
|
|
url += '&audiosamplerate=' + codecLimits.maxSampleRate;
|
|
|
|
|
url += '&mediasourceid=' + mediaSourceInfo.mediaSource.Id;
|
2014-04-09 16:58:09 -07:00
|
|
|
|
|
2014-04-15 20:49:49 -07:00
|
|
|
|
return url;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
throw new Error('Unrecognized MediaType');
|
|
|
|
|
}
|
2014-02-23 13:35:58 -07:00
|
|
|
|
|
2014-04-11 08:36:25 -07:00
|
|
|
|
function chromecastPlayer() {
|
2014-02-23 13:35:58 -07:00
|
|
|
|
|
2014-04-05 10:01:04 -07:00
|
|
|
|
var self = this;
|
2014-02-23 13:35:58 -07:00
|
|
|
|
|
2014-04-12 10:27:53 -07:00
|
|
|
|
var currentPlaylistIndex;
|
2014-04-10 04:12:48 -07:00
|
|
|
|
|
2014-04-15 20:49:49 -07:00
|
|
|
|
var getItemFields = "MediaSources,Chapters";
|
|
|
|
|
|
2014-04-06 10:53:23 -07:00
|
|
|
|
self.name = PlayerName;
|
2014-02-23 13:35:58 -07:00
|
|
|
|
|
2014-04-09 16:58:09 -07:00
|
|
|
|
self.isPaused = false;
|
|
|
|
|
|
2014-04-15 21:52:35 -07:00
|
|
|
|
self.isMuted = false;
|
|
|
|
|
|
2014-04-10 04:12:48 -07:00
|
|
|
|
self.playlist = [];
|
2014-02-23 10:53:25 -07:00
|
|
|
|
|
2014-04-10 04:12:48 -07:00
|
|
|
|
self.playlistIndex = 0;
|
2014-04-09 16:58:09 -07:00
|
|
|
|
|
2014-04-15 20:49:49 -07:00
|
|
|
|
self.positionTicks = 0;
|
|
|
|
|
|
|
|
|
|
self.runtimeTicks = 0;
|
|
|
|
|
|
|
|
|
|
$(castPlayer).on("/playback/complete", function (e) {
|
|
|
|
|
if (self.playlistIndex < (self.playlist.items || []).length) {
|
|
|
|
|
self.play(self.playlist);
|
|
|
|
|
} else {
|
|
|
|
|
$(self).trigger("playbackstop.nowplayingbar");
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
$(castPlayer).on("/playback/update", function (e, data) {
|
2014-04-18 00:56:38 -07:00
|
|
|
|
|
2014-04-15 20:49:49 -07:00
|
|
|
|
self.positionTicks = data.positionTicks;
|
|
|
|
|
self.runtimeTicks = data.runtimeTicks;
|
2014-04-06 10:53:23 -07:00
|
|
|
|
|
2014-04-15 20:49:49 -07:00
|
|
|
|
var state = self.getPlayerStateInternal();
|
2014-04-09 16:58:09 -07:00
|
|
|
|
|
2014-04-15 20:49:49 -07:00
|
|
|
|
$(self).trigger("positionchange.nowplayingbar", [ state ]);
|
|
|
|
|
});
|
2014-04-09 16:58:09 -07:00
|
|
|
|
|
2014-04-10 04:12:48 -07:00
|
|
|
|
function getItemsForPlayback(query) {
|
|
|
|
|
var userId = Dashboard.getCurrentUserId();
|
|
|
|
|
|
|
|
|
|
query.Limit = query.Limit || 100;
|
|
|
|
|
query.Fields = getItemFields;
|
|
|
|
|
query.ExcludeLocationTypes = "Virtual";
|
2014-04-06 10:53:23 -07:00
|
|
|
|
|
2014-04-10 04:12:48 -07:00
|
|
|
|
return ApiClient.getItems(userId, query);
|
2014-04-11 08:36:25 -07:00
|
|
|
|
}
|
2014-04-10 04:12:48 -07:00
|
|
|
|
|
|
|
|
|
function queueItems (items) {
|
|
|
|
|
for (var i = 0, length = items.length; i < length; i++) {
|
|
|
|
|
self.playlist.push(items[i]);
|
|
|
|
|
}
|
2014-04-11 08:36:25 -07:00
|
|
|
|
}
|
2014-04-10 04:12:48 -07:00
|
|
|
|
|
|
|
|
|
function queueItemsNext(items) {
|
|
|
|
|
var insertIndex = 1;
|
|
|
|
|
for (var i = 0, length = items.length; i < length; i++) {
|
|
|
|
|
self.playlist.splice(insertIndex, 0, items[i]);
|
|
|
|
|
insertIndex++;
|
|
|
|
|
}
|
2014-04-11 08:36:25 -07:00
|
|
|
|
}
|
2014-04-10 04:12:48 -07:00
|
|
|
|
|
|
|
|
|
function translateItemsForPlayback(items) {
|
|
|
|
|
var deferred = $.Deferred();
|
|
|
|
|
var firstItem = items[0];
|
|
|
|
|
var promise;
|
|
|
|
|
if (firstItem.IsFolder) {
|
|
|
|
|
promise = self.getItemsForPlayback({
|
|
|
|
|
ParentId: firstItem.Id,
|
|
|
|
|
Filters: "IsNotFolder",
|
|
|
|
|
Recursive: true,
|
|
|
|
|
SortBy: "SortName",
|
|
|
|
|
MediaTypes: "Audio,Video"
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
else if (firstItem.Type == "MusicArtist") {
|
|
|
|
|
promise = self.getItemsForPlayback({
|
|
|
|
|
Artists: firstItem.Name,
|
|
|
|
|
Filters: "IsNotFolder",
|
|
|
|
|
Recursive: true,
|
|
|
|
|
SortBy: "SortName",
|
|
|
|
|
MediaTypes: "Audio"
|
2014-04-06 10:53:23 -07:00
|
|
|
|
});
|
2014-04-10 04:12:48 -07:00
|
|
|
|
}
|
|
|
|
|
else if (firstItem.Type == "MusicGenre") {
|
|
|
|
|
promise = self.getItemsForPlayback({
|
|
|
|
|
Genres: firstItem.Name,
|
|
|
|
|
Filters: "IsNotFolder",
|
|
|
|
|
Recursive: true,
|
|
|
|
|
SortBy: "SortName",
|
|
|
|
|
MediaTypes: "Audio"
|
|
|
|
|
});
|
|
|
|
|
}
|
2014-04-06 10:53:23 -07:00
|
|
|
|
|
2014-04-10 04:12:48 -07:00
|
|
|
|
if (promise) {
|
|
|
|
|
promise.done(function (result) {
|
|
|
|
|
deferred.resolveWith(null, [result.Items]);
|
|
|
|
|
});
|
2014-04-06 10:53:23 -07:00
|
|
|
|
} else {
|
2014-04-10 04:12:48 -07:00
|
|
|
|
deferred.resolveWith(null, [items]);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return deferred.promise();
|
2014-04-11 08:36:25 -07:00
|
|
|
|
}
|
2014-04-06 10:53:23 -07:00
|
|
|
|
|
2014-04-10 04:12:48 -07:00
|
|
|
|
self.play = function (options) {
|
|
|
|
|
if (self.isPaused) {
|
|
|
|
|
self.isPaused = !self.isPaused;
|
|
|
|
|
castPlayer.playMedia();
|
|
|
|
|
} else if (options.items) {
|
|
|
|
|
Dashboard.getCurrentUser().done(function (user) {
|
|
|
|
|
castPlayer.loadMedia(user, options.items[self.playlistIndex++], options.startPositionTicks);
|
|
|
|
|
});
|
|
|
|
|
} else {
|
2014-04-06 10:53:23 -07:00
|
|
|
|
var userId = Dashboard.getCurrentUserId();
|
|
|
|
|
|
|
|
|
|
var query = {};
|
|
|
|
|
query.Limit = query.Limit || 100;
|
2014-04-15 20:49:49 -07:00
|
|
|
|
query.Fields = getItemFields;
|
2014-04-06 10:53:23 -07:00
|
|
|
|
query.ExcludeLocationTypes = "Virtual";
|
|
|
|
|
query.Ids = options.ids.join(',');
|
|
|
|
|
|
|
|
|
|
ApiClient.getItems(userId, query).done(function (result) {
|
|
|
|
|
options.items = result.Items;
|
|
|
|
|
self.play(options);
|
|
|
|
|
});
|
|
|
|
|
}
|
2014-04-05 10:01:04 -07:00
|
|
|
|
};
|
2014-03-30 16:26:16 -07:00
|
|
|
|
|
2014-04-09 16:58:09 -07:00
|
|
|
|
self.unpause = function () {
|
|
|
|
|
self.isPaused = !self.isPaused;
|
|
|
|
|
castPlayer.playMedia();
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
self.pause = function () {
|
|
|
|
|
self.isPaused = true;
|
|
|
|
|
castPlayer.pauseMedia();
|
|
|
|
|
};
|
|
|
|
|
|
2014-04-05 10:01:04 -07:00
|
|
|
|
self.shuffle = function (id) {
|
2014-04-10 04:12:48 -07:00
|
|
|
|
var userId = Dashboard.getCurrentUserId();
|
|
|
|
|
ApiClient.getItem(userId, id).done(function (item) {
|
|
|
|
|
var query = {
|
|
|
|
|
UserId: userId,
|
|
|
|
|
Fields: getItemFields,
|
|
|
|
|
Limit: 50,
|
|
|
|
|
Filters: "IsNotFolder",
|
|
|
|
|
Recursive: true,
|
|
|
|
|
SortBy: "Random"
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
if (item.IsFolder) {
|
|
|
|
|
query.ParentId = id;
|
|
|
|
|
}
|
|
|
|
|
else if (item.Type == "MusicArtist") {
|
|
|
|
|
query.MediaTypes = "Audio";
|
|
|
|
|
query.Artists = item.Name;
|
|
|
|
|
}
|
|
|
|
|
else if (item.Type == "MusicGenre") {
|
|
|
|
|
query.MediaTypes = "Audio";
|
|
|
|
|
query.Genres = item.Name;
|
|
|
|
|
} else {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
self.getItemsForPlayback(query).done(function (result) {
|
|
|
|
|
self.playlist = { items: result.Items };
|
|
|
|
|
self.play(self.playlist);
|
|
|
|
|
});
|
|
|
|
|
});
|
2014-04-05 10:01:04 -07:00
|
|
|
|
};
|
2014-02-23 10:29:02 -07:00
|
|
|
|
|
2014-04-05 10:01:04 -07:00
|
|
|
|
self.instantMix = function (id) {
|
2014-04-10 04:12:48 -07:00
|
|
|
|
var userId = Dashboard.getCurrentUserId();
|
|
|
|
|
ApiClient.getItem(userId, id).done(function (item) {
|
|
|
|
|
var promise;
|
|
|
|
|
var mixLimit = 3;
|
|
|
|
|
|
|
|
|
|
if (item.Type == "MusicArtist") {
|
|
|
|
|
promise = ApiClient.getInstantMixFromArtist(name, {
|
|
|
|
|
UserId: userId,
|
|
|
|
|
Fields: getItemFields,
|
|
|
|
|
Limit: mixLimit
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
else if (item.Type == "MusicGenre") {
|
|
|
|
|
promise = ApiClient.getInstantMixFromMusicGenre(name, {
|
|
|
|
|
UserId: userId,
|
|
|
|
|
Fields: getItemFields,
|
|
|
|
|
Limit: mixLimit
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
else if (item.Type == "MusicAlbum") {
|
|
|
|
|
promise = ApiClient.getInstantMixFromAlbum(id, {
|
|
|
|
|
UserId: userId,
|
|
|
|
|
Fields: getItemFields,
|
|
|
|
|
Limit: mixLimit
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
else if (item.Type == "Audio") {
|
|
|
|
|
promise = ApiClient.getInstantMixFromSong(id, {
|
|
|
|
|
UserId: userId,
|
|
|
|
|
Fields: getItemFields,
|
|
|
|
|
Limit: mixLimit
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
promise.done(function (result) {
|
|
|
|
|
self.playlist = { items: result.Items };
|
|
|
|
|
self.play(self.playlist);
|
|
|
|
|
});
|
|
|
|
|
});
|
2014-04-05 10:01:04 -07:00
|
|
|
|
};
|
2014-02-22 22:52:30 -07:00
|
|
|
|
|
2014-04-10 04:12:48 -07:00
|
|
|
|
self.canQueueMediaType = function (mediaType) {
|
|
|
|
|
return mediaType == "Audio";
|
|
|
|
|
};
|
2014-04-05 10:01:04 -07:00
|
|
|
|
|
2014-04-10 04:12:48 -07:00
|
|
|
|
self.queue = function (options) {
|
|
|
|
|
Dashboard.getCurrentUser().done(function (user) {
|
|
|
|
|
if (options.items) {
|
|
|
|
|
translateItemsForPlayback(options.items).done(function (items) {
|
|
|
|
|
queueItems(items);
|
|
|
|
|
});
|
|
|
|
|
} else {
|
|
|
|
|
getItemsForPlayback({
|
|
|
|
|
Ids: options.ids.join(',')
|
|
|
|
|
}).done(function (result) {
|
|
|
|
|
translateItemsForPlayback(result.Items).done(function (items) {
|
|
|
|
|
queueItems(items);
|
|
|
|
|
});
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
});
|
2014-04-05 10:01:04 -07:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
self.queueNext = function (options) {
|
2014-04-10 04:12:48 -07:00
|
|
|
|
Dashboard.getCurrentUser().done(function (user) {
|
|
|
|
|
if (options.items) {
|
|
|
|
|
queueItemsNext(options.items);
|
|
|
|
|
} else {
|
|
|
|
|
self.getItemsForPlayback({
|
|
|
|
|
Ids: options.ids.join(',')
|
|
|
|
|
}).done(function (result) {
|
|
|
|
|
options.items = result.Items;
|
|
|
|
|
queueItemsNext(options.items);
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
});
|
2014-04-05 10:01:04 -07:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
self.stop = function () {
|
2014-04-10 04:12:48 -07:00
|
|
|
|
self.playlist = [];
|
|
|
|
|
self.playlistIndex = 0;
|
2014-04-09 16:58:09 -07:00
|
|
|
|
castPlayer.stopMedia();
|
2014-04-05 10:01:04 -07:00
|
|
|
|
};
|
|
|
|
|
|
2014-04-13 10:27:13 -07:00
|
|
|
|
self.displayContent = function (options) {
|
|
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
2014-04-06 10:53:23 -07:00
|
|
|
|
self.mute = function () {
|
2014-04-15 20:49:49 -07:00
|
|
|
|
self.isMuted = true;
|
2014-04-06 10:53:23 -07:00
|
|
|
|
castPlayer.mute();
|
2014-04-05 10:01:04 -07:00
|
|
|
|
};
|
|
|
|
|
|
2014-04-15 20:49:49 -07:00
|
|
|
|
self.unMute = function () {
|
|
|
|
|
self.isMuted = false;
|
2014-04-06 10:53:23 -07:00
|
|
|
|
castPlayer.unMute();
|
2014-04-05 10:01:04 -07:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
self.toggleMute = function () {
|
2014-04-06 10:53:23 -07:00
|
|
|
|
castPlayer.toggleMute();
|
2014-04-05 10:01:04 -07:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
self.getTargets = function () {
|
2014-04-13 10:27:13 -07:00
|
|
|
|
|
2014-04-05 10:01:04 -07:00
|
|
|
|
var targets = [];
|
2014-04-13 10:27:13 -07:00
|
|
|
|
|
|
|
|
|
if (castPlayer.hasReceivers) {
|
|
|
|
|
targets.push(self.getCurrentTargetInfo());
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-05 10:01:04 -07:00
|
|
|
|
return targets;
|
2014-04-13 10:27:13 -07:00
|
|
|
|
|
2014-04-05 10:01:04 -07:00
|
|
|
|
};
|
|
|
|
|
|
2014-04-06 10:53:23 -07:00
|
|
|
|
self.getCurrentTargetInfo = function () {
|
2014-04-13 10:27:13 -07:00
|
|
|
|
|
2014-04-06 10:53:23 -07:00
|
|
|
|
var appName = null;
|
2014-04-26 21:46:12 -07:00
|
|
|
|
if (castPlayer.session && castPlayer.session.receiver && castPlayer.session.receiver.friendlyName) {
|
|
|
|
|
appName = castPlayer.session.receiver.friendlyName;
|
2014-04-06 10:53:23 -07:00
|
|
|
|
}
|
2014-02-23 10:53:25 -07:00
|
|
|
|
|
2014-04-06 10:53:23 -07:00
|
|
|
|
return {
|
|
|
|
|
name: PlayerName,
|
|
|
|
|
id: PlayerName,
|
2014-04-26 21:46:12 -07:00
|
|
|
|
playerName: self.name, // TODO: PlayerName == self.name, so do we need to use either/or?
|
2014-04-06 10:53:23 -07:00
|
|
|
|
playableMediaTypes: ["Audio", "Video"],
|
|
|
|
|
isLocalPlayer: false,
|
2014-04-13 10:27:13 -07:00
|
|
|
|
appName: appName,
|
2014-04-15 20:49:49 -07:00
|
|
|
|
supportedCommands: ["VolumeUp",
|
|
|
|
|
"VolumeDown",
|
|
|
|
|
"Mute",
|
|
|
|
|
"Unmute",
|
|
|
|
|
"ToggleMute",
|
|
|
|
|
"SetVolume",
|
|
|
|
|
"DisplayContent"]
|
2014-04-06 10:53:23 -07:00
|
|
|
|
};
|
|
|
|
|
};
|
2014-04-09 16:58:09 -07:00
|
|
|
|
|
|
|
|
|
self.setCurrentTime = function (ticks, item, updateSlider) {
|
|
|
|
|
// Convert to ticks
|
|
|
|
|
ticks = Math.floor(ticks);
|
|
|
|
|
var timeText = Dashboard.getDisplayTime(ticks);
|
|
|
|
|
if (self.currentDurationTicks) {
|
|
|
|
|
timeText += " / " + Dashboard.getDisplayTime(self.currentDurationTicks);
|
|
|
|
|
if (updateSlider) {
|
|
|
|
|
var percent = ticks / self.currentDurationTicks;
|
|
|
|
|
percent *= 100;
|
|
|
|
|
self.positionSlider.val(percent).slider('enable').slider('refresh');
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
self.positionSlider.slider('disable').slider('refresh');
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
self.currentTimeElement.html(timeText);
|
|
|
|
|
};
|
|
|
|
|
|
2014-04-23 04:12:27 -07:00
|
|
|
|
self.changeStream = function (position) {
|
|
|
|
|
castPlayer.seekMedia(position);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
self.seek = function (position) {
|
2014-04-18 12:59:06 -07:00
|
|
|
|
castPlayer.seekMedia(position);
|
2014-04-10 04:12:48 -07:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
self.removeFromPlaylist = function (i) {
|
|
|
|
|
self.playlist.remove(i);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
self.currentPlaylistIndex = function (i) {
|
|
|
|
|
if (i == null) {
|
|
|
|
|
return currentPlaylistIndex;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var newItem = self.playlist[i];
|
|
|
|
|
|
|
|
|
|
Dashboard.getCurrentUser().done(function (user) {
|
|
|
|
|
self.playInternal(newItem, 0, user);
|
|
|
|
|
currentPlaylistIndex = i;
|
|
|
|
|
});
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
self.nextTrack = function () {
|
|
|
|
|
var newIndex = currentPlaylistIndex + 1;
|
|
|
|
|
var newItem = self.playlist[newIndex];
|
|
|
|
|
|
|
|
|
|
if (newItem) {
|
|
|
|
|
Dashboard.getCurrentUser().done(function (user) {
|
|
|
|
|
self.playInternal(newItem, 0, user);
|
|
|
|
|
currentPlaylistIndex = newIndex;
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
self.previousTrack = function () {
|
|
|
|
|
var newIndex = currentPlaylistIndex - 1;
|
|
|
|
|
if (newIndex >= 0) {
|
|
|
|
|
var newItem = self.playlist[newIndex];
|
|
|
|
|
if (newItem) {
|
|
|
|
|
Dashboard.getCurrentUser().done(function (user) {
|
|
|
|
|
self.playInternal(newItem, 0, user);
|
|
|
|
|
currentPlaylistIndex = newIndex;
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2014-04-12 10:27:53 -07:00
|
|
|
|
self.beginPlayerUpdates = function () {
|
|
|
|
|
// Setup polling here
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
self.endPlayerUpdates = function () {
|
|
|
|
|
// Stop polling here
|
|
|
|
|
};
|
|
|
|
|
|
2014-04-10 04:12:48 -07:00
|
|
|
|
self.volumeDown = function () {
|
2014-04-15 20:49:49 -07:00
|
|
|
|
var vol = castPlayer.volumeLevel - 0.02;
|
|
|
|
|
castPlayer.setReceiverVolume(false, vol / 100);
|
2014-04-10 04:12:48 -07:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
self.volumeUp = function () {
|
2014-04-15 20:49:49 -07:00
|
|
|
|
var vol = castPlayer.volumeLevel + 0.02;
|
|
|
|
|
castPlayer.setReceiverVolume(false, vol / 100);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
self.setVolume = function (vol) {
|
|
|
|
|
castPlayer.setReceiverVolume(false, vol / 100);
|
2014-04-09 16:58:09 -07:00
|
|
|
|
};
|
2014-04-12 10:27:53 -07:00
|
|
|
|
|
2014-04-27 18:57:29 -07:00
|
|
|
|
self.sendCommand = function (cmd) {
|
|
|
|
|
|
|
|
|
|
// Full list
|
|
|
|
|
// https://github.com/MediaBrowser/MediaBrowser/blob/master/MediaBrowser.Model/Session/GeneralCommand.cs#L23
|
|
|
|
|
|
|
|
|
|
switch (cmd.Name) {
|
|
|
|
|
|
|
|
|
|
case 'VolumeUp':
|
|
|
|
|
self.volumeUp();
|
|
|
|
|
break;
|
|
|
|
|
case 'VolumeDown':
|
|
|
|
|
self.volumeDown();
|
|
|
|
|
break;
|
|
|
|
|
case 'Mute':
|
|
|
|
|
self.mute();
|
|
|
|
|
break;
|
|
|
|
|
case 'Unmute':
|
|
|
|
|
self.unMute();
|
|
|
|
|
break;
|
|
|
|
|
case 'ToggleMute':
|
|
|
|
|
self.toggleMute();
|
|
|
|
|
break;
|
|
|
|
|
case 'SetVolume':
|
|
|
|
|
self.setVolume(cmd.Arguments.Volume);
|
|
|
|
|
break;
|
|
|
|
|
case 'SetAudioStreamIndex':
|
|
|
|
|
break;
|
|
|
|
|
case 'SetSubtitleStreamIndex':
|
|
|
|
|
break;
|
|
|
|
|
case 'ToggleFullscreen':
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
// Not player-related
|
|
|
|
|
Dashboard.processGeneralCommand(cmd);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
2014-04-12 10:27:53 -07:00
|
|
|
|
self.getPlayerState = function () {
|
|
|
|
|
|
|
|
|
|
var deferred = $.Deferred();
|
|
|
|
|
|
|
|
|
|
var result = self.getPlayerStateInternal();
|
|
|
|
|
|
|
|
|
|
deferred.resolveWith(null, [result]);
|
|
|
|
|
|
|
|
|
|
return deferred.promise();
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
self.getPlayerStateInternal = function () {
|
|
|
|
|
|
2014-04-22 10:25:54 -07:00
|
|
|
|
var state = {
|
|
|
|
|
PlayState: {
|
|
|
|
|
|
|
|
|
|
CanSeek: self.positionTicks < self.runtimeTicks,
|
|
|
|
|
PositionTicks: self.positionTicks,
|
|
|
|
|
VolumeLevel: castPlayer.currentVolume * 100,
|
|
|
|
|
IsPaused: self.isPaused,
|
|
|
|
|
IsMuted: self.isMuted
|
|
|
|
|
|
|
|
|
|
// TODO: Implement
|
|
|
|
|
// AudioStreamIndex: null,
|
|
|
|
|
// SubtitleStreamIndex: null,
|
|
|
|
|
// PlayMethod: 'DirectStream' or 'Transcode'
|
|
|
|
|
}
|
2014-04-15 20:49:49 -07:00
|
|
|
|
};
|
2014-04-22 10:25:54 -07:00
|
|
|
|
|
|
|
|
|
// TODO: Implement
|
|
|
|
|
var isPlaying = false;
|
|
|
|
|
|
|
|
|
|
if (isPlaying) {
|
|
|
|
|
|
|
|
|
|
//state.PlayState.MediaSourceId = 'xxx';
|
|
|
|
|
|
|
|
|
|
state.NowPlayingItem = {
|
|
|
|
|
|
2014-04-22 10:31:06 -07:00
|
|
|
|
RunTimeTicks: self.runtimeTicks,
|
2014-04-22 10:25:54 -07:00
|
|
|
|
Name: 'Chromecast'
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
var nowPlayingItem = state.NowPlayingItem;
|
|
|
|
|
|
|
|
|
|
// TODO: Fill in these properties using chromecast mediainfo and/or custom data
|
|
|
|
|
//nowPlayingItem.Id = item.Id;
|
|
|
|
|
//nowPlayingItem.MediaType = item.MediaType;
|
|
|
|
|
//nowPlayingItem.Type = item.Type;
|
|
|
|
|
//nowPlayingItem.Name = item.Name;
|
|
|
|
|
|
|
|
|
|
//nowPlayingItem.IndexNumber = item.IndexNumber;
|
|
|
|
|
//nowPlayingItem.IndexNumberEnd = item.IndexNumberEnd;
|
|
|
|
|
//nowPlayingItem.ParentIndexNumber = item.ParentIndexNumber;
|
|
|
|
|
//nowPlayingItem.ProductionYear = item.ProductionYear;
|
|
|
|
|
//nowPlayingItem.PremiereDate = item.PremiereDate;
|
|
|
|
|
//nowPlayingItem.SeriesName = item.SeriesName;
|
|
|
|
|
//nowPlayingItem.Album = item.Album;
|
|
|
|
|
//nowPlayingItem.Artists = item.Artists;
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return state;
|
2014-04-12 10:27:53 -07:00
|
|
|
|
};
|
2014-04-06 10:53:23 -07:00
|
|
|
|
}
|
2014-02-23 13:35:58 -07:00
|
|
|
|
|
2014-04-11 08:36:25 -07:00
|
|
|
|
MediaController.registerPlayer(new chromecastPlayer());
|
2014-02-22 22:52:30 -07:00
|
|
|
|
|
2014-04-05 10:01:04 -07:00
|
|
|
|
$(MediaController).on('playerchange', function () {
|
|
|
|
|
|
2014-04-06 10:53:23 -07:00
|
|
|
|
if (MediaController.getPlayerInfo().name == PlayerName) {
|
|
|
|
|
|
|
|
|
|
if (castPlayer.deviceState != DEVICE_STATE.ACTIVE && castPlayer.isInitialized) {
|
|
|
|
|
castPlayer.launchApp();
|
|
|
|
|
}
|
2014-04-05 10:01:04 -07:00
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
2014-02-22 22:52:30 -07:00
|
|
|
|
})(window, window.chrome, console);
|