Skip to content

Instantly share code, notes, and snippets.

@VioletRed
Last active August 29, 2015 14:05
Show Gist options
  • Save VioletRed/9577d8c062f3ff056c59 to your computer and use it in GitHub Desktop.
Save VioletRed/9577d8c062f3ff056c59 to your computer and use it in GitHub Desktop.
Play on XBMC
// ==UserScript==
// @name Play on Kodi/XBMC
// @namespace user@violet.local
//
// @description Resolve and play media on Kodi/XBMC
// @description Use with AnyURL plugin from:
// @description https://github.com/VioletRed/script.anyurl.player
//
// @date 2015-01-04
// @version 17
// @include *
// @grant GM_addStyle
// @grant GM_registerMenuCommand
// @grant GM_getValue
// @grant GM_setValue
// @grant GM_xmlhttpRequest
// @grant GM_log
// @updateURL https://github.com/VioletRed/script.anyurl.player/raw/master/json/Play_on_XBMC.user.js
// ==/UserScript==
//
// Simple script to send media to Kodi.
// Adds a "Send to Kodi" button on supported websites
// Supported plugins:
// * Youtube
// * TED
// * AnyURL plugin for other domains (https://github.com/VioletRed/script.anyurl.player).
//
// !!!!!!!!!
// THIS IS THE LAST UPDATE IN GIST.
// CHECK https://github.com/VioletRed/script.anyurl.player FOR FUTURE UPDATES
//
// It uses the old GM_*** API, and needs cleaning.
/* ============================================================================
* Global config
* */
var xbmc_address = GM_getValue('XBMC_ADDRESS');
var xbmc_queued = null;
const xbmc_music_playlist = 0; // Queue for party mode
const xbmc_video_playlist = 1; // Queue for video mode
//Remove known top domain names (i.e 'www', 'm', 'embed')
var top_domain = /^www\.|^m\.|^embed\./
var current_host = window.location.host.toLowerCase().replace(top_domain, '');
/*
* ============================================================================
* Global UI elements
* ============================================================================
*/
var xbmc_ui = null;
var xbmc_title = null;
var xbmc_play_control = null;
var xbmc_msg_timer = null;
/*
* ============================================================================
* Site independent code here!!!!
* ============================================================================
*/
function xbmc_json_error(response) {
consoloe.log("XBMC JSON Error")
}
function xbmc_json_timeout(response) {
consoloe.log("XBMC JSON Timeout")
}
function json_command_answer(command, logmsgok, logmsgerr) {
GM_xmlhttpRequest({
method : 'POST',
url : 'http://' + xbmc_address + '/jsonrpc',
headers : {
"Content-type" : "application/json"
},
data : command,
onload : function(response) {
console.log(logmsgok);
return response;
}
});
console.log(logmsgerr);
return -1;
}
function play_movie_directly(video_url) {
GM_xmlhttpRequest({
method : 'POST',
url : 'http://' + xbmc_address + '/jsonrpc',
headers : {
"Content-type" : "application/json"
},
data : '{"jsonrpc": "2.0", "method": "Player.Open", '
+ '"params":{"item": { "file" : "'
+ encode_video_url(video_url) + '" }}, "id" : 1}',
onload : function(response) {
show_ui_msg("PLAYING", 2000);
console.log('Playing video');
}
});
/* Clear playlist */
setTimeout(
function() {
GM_xmlhttpRequest({
method : 'POST',
url : 'http://' + xbmc_address + '/jsonrpc',
headers : {
"Content-type" : "application/json"
},
timeout: 6000,
data : '{"jsonrpc": "2.0", "method": "Playlist.Clear", '
+ '"params":{"playlistid" : '+xbmc_video_playlist+'}, "id" : 1}',
onerror: xbmc_json_error,
ontimeout: xbmc_json_timeout,
});
}, 5000);
}
function open_video_playlist() {
console.log('New video queue');
GM_xmlhttpRequest({
method : 'POST',
url : 'http://' + xbmc_address + '/jsonrpc',
headers : {
"Content-type" : "application/json"
},
data : '{"jsonrpc": "2.0", "method": "Player.Open", '
+ '"params":{"item": { "playlistid" : ' + xbmc_video_playlist
+ ' }}, "id" : 1}',
onload : function(response) {
console.log('Playing video');
show_ui_msg("PLAYING", 2000);
}
});
}
function dont_open_video_playlist() {
console.log('Queued video at the end ');
show_ui_msg("QUEUED", 5000);
return 0;
}
function play_in_new_playlist(video_url) {
/* Clear playlist */
GM_xmlhttpRequest({
method : 'POST',
url : 'http://' + xbmc_address + '/jsonrpc',
headers : {
"Content-type" : "application/json"
},
timeout: 6000,
data : '{"jsonrpc": "2.0", "method": "Playlist.Clear", '
+ '"params":{"playlistid" : '+xbmc_video_playlist+'}, "id" : 1}',
onload : function(response) {
/* Add movies to Video playlist */
xbmc_queued = "";
queue_movie_last(video_url, open_video_playlist);
},
onerror: xbmc_json_error,
ontimeout: xbmc_json_timeout,
});
}
function queue_movie_at(video_url, xbmc_playlist, xbmc_queue_depth) {
if (xbmc_queued == video_url) {
// Show somehow that this action was already completed
console.log("Already queued "+xbmc_queued);
return;
}
GM_xmlhttpRequest({
method : 'POST',
url : 'http://' + xbmc_address + '/jsonrpc',
headers : {
"Content-type" : "application/json"
},
data : '{"jsonrpc": "2.0", "method": "Playlist.Insert", '
+ '"params":{"item": { "file" : "'
+ encode_video_url_for_queueing(video_url)
+ '" }, "playlistid" :'
+ xbmc_playlist
+ ', "position" : '
+ xbmc_queue_depth
+ ' }, "id" : 1}',
onerror: xbmc_json_error,
ontimeout: xbmc_json_timeout,
onload : function(response) {
xbmc_queued = video_url;
show_ui_msg("QUEUEED", 5000);
console.log('Queueing video');
}
})
}
function queue_movie_last(video_url, last_step) {
if (xbmc_queued == video_url) {
// Show somehow that this action was already completed
console.log("Already queued "+xbmc_queued);
return;
}
GM_xmlhttpRequest({
method : 'POST',
url : 'http://' + xbmc_address + '/jsonrpc',
headers : {
"Content-type" : "application/json"
},
data : '{"jsonrpc": "2.0", "method": "Playlist.Add", '
+ '"params":{"item": { "file" : "'
+ encode_video_url_for_queueing(video_url) + '" }, "playlistid" :'
+ xbmc_video_playlist + ' }, "id" : 1}',
onload : function(response) {
var result = JSON.parse(response.responseText);
if (result.result == "OK") {
xbmc_queued = video_url;
last_step();
return 0;
}
},
onerror: xbmc_json_error,
ontimeout: xbmc_json_timeout
});
return -1;
}
function queue_in_party_mode(video_url) {
GM_xmlhttpRequest({
method : 'POST',
url : 'http://' + xbmc_address + '/jsonrpc',
headers : {
"Content-type" : "application/json"
},
data : '{"jsonrpc": "2.0", "method": "Playlist.GetItems",'
+ '"params":{"playlistid" : '
+ xbmc_music_playlist
+ '}, "id" : 1}',
onload : function(response) {
var xbmc_response = JSON.parse(response.responseText);
if (xbmc_response.result.limits == undefined) {
console.log("Error: Playlist.GetItems bad response");
return;
}
// Queue exist, enqueue media at the end of user
// selection
xbmc_queue_depth = xbmc_response.result.limits.end - 9;
console.log("XBMC queue size is "
+ xbmc_queue_depth);
queue_movie_at(video_url, xbmc_music_playlist, xbmc_queue_depth);
}
})
}
function queue_in_playlist(video_url) {
GM_xmlhttpRequest({
method : 'POST',
url : 'http://' + xbmc_address + '/jsonrpc',
headers : {
"Content-type" : "application/json"
},
data : '{"jsonrpc": "2.0", "method": "Playlist.GetItems",'
+ '"params":{"playlistid" : '
+ xbmc_video_playlist
+ '}, "id" : 1}',
onload : function(response) {
var xbmc_response = JSON.parse(response.responseText);
if (xbmc_response.result.limits == undefined
|| xbmc_response.result.limits.end == 0) {
console.log("Playlist.GetItems bad response");
play_in_new_playlist(video_url)
return;
}
// Queue exist, enqueue media at the end of user
// selection
queue_movie_last(video_url,dont_open_video_playlist);
}
})
}
function play_movie() {
video_url = document.documentURI
console.log('Trying to play/queue movie');
var xbmc_queue_depth = undefined;
show_ui_msg("LOADING", 30000);
/*
* Logic goes like this: First, try to queue the video. If it fails, play
* video directly. Because AJAX is asynchronous, we use a timer for "direct
* play", and cancel it if we succeed to queue the video where we want
*/
if (video_url == undefined || xbmc_queued == video_url) {
return;
}
// Get the current playlist
GM_xmlhttpRequest({
method : 'POST',
url : 'http://' + xbmc_address + '/jsonrpc',
headers : {
"Content-type" : "application/json"
},
data : '{"jsonrpc": "2.0", "method": "Player.GetActivePlayers",'
+ '"params":{}, "id" : 1}',
onload : function(response) {
var xbmc_active = JSON.parse(response.responseText);
if (xbmc_active.result == undefined
|| xbmc_active.result.length == 0) {
console.log("No active players, play directly");
play_movie_directly(video_url)
return; // No active players
}
GM_xmlhttpRequest({
method : 'POST',
url : 'http://' + xbmc_address + '/jsonrpc',
headers : {
"Content-type" : "application/json"
},
data : '{"jsonrpc": "2.0", "method": "Player.GetProperties",'
+ '"params":{"playerid" : '
+ xbmc_active.result[0].playerid
+ ', "properties" : [ "playlistid" , "partymode" ] }, "id" : 1}',
onload : function(response) {
var xbmc_properties = JSON.parse(response.responseText);
if (xbmc_properties.result.partymode != true) {
console.log("Not in party mode, play now");
play_movie_directly(video_url);
return;
}
queue_in_party_mode(video_url)
}
});
}
});
}
function queue_movie() {
video_url = document.documentURI
console.log('Trying queue movie/create new playlist');
var xbmc_queue_depth = undefined;
show_ui_msg("LOADING", 30000);
// Get the current playlist
GM_xmlhttpRequest({
method : 'POST',
url : 'http://' + xbmc_address + '/jsonrpc',
headers : {
"Content-type" : "application/json"
},
data : '{"jsonrpc": "2.0", "method": "Player.GetActivePlayers",'
+ '"params":{}, "id" : 1}',
onerror : xbmc_json_error,
ontimeout : xbmc_json_timeout,
onload : function(response) {
var xbmc_active = JSON.parse(response.responseText);
if (xbmc_active.result == undefined
|| xbmc_active.result.length == 0) {
console.log("No active players, create a new queue");
play_in_new_playlist(video_url);
return;
}
GM_xmlhttpRequest({
method : 'POST',
url : 'http://' + xbmc_address + '/jsonrpc',
headers : {
"Content-type" : "application/json"
},
data : '{"jsonrpc": "2.0", "method": "Player.GetProperties",'
+ '"params":{"playerid" : '
+ xbmc_active.result[0].playerid
+ ', "properties" : [ "playlistid" , "partymode" ] }, "id" : 1}',
onload : function(response) {
var xbmc_properties = JSON.parse(response.responseText);
if (xbmc_properties.result.partymode == true) {
console.log("Party mode, default play");
queue_in_party_mode(video_url);
return;
}
if (xbmc_active.result[0].playerid != 1) {
console.log("Playing music, create a new queue");
play_in_new_playlist(video_url);
} else {
console.log("Queue in playlist");
queue_in_playlist(video_url);
}
},
onerror: function(response) {
/* No active playlist */
console.log("Not playing playlist, queue and play")
play_in_new_playlist(video_url);
}
});
}
});
}
/*
* ============================================================================
* Movie control functions
* ============================================================================
*/
function pause_movie() {
GM_xmlhttpRequest({
method : 'POST',
url : 'http://' + xbmc_address + '/jsonrpc',
headers : {
'Content-Type' : 'application/json'
},
data : '{"jsonrpc":"2.0", "method":"Player.PlayPause", "params":{"playerid":1}, "id" : 1}'
});
}
function stop_movie() {
GM_xmlhttpRequest({
method : 'POST',
url : 'http://' + xbmc_address + '/jsonrpc',
headers : {
'Content-Type' : 'application/json'
},
data : '{"jsonrpc":"2.0", "method": "Player.Stop", "params":{"playerid":1}, "id" : 1}'
});
xbmc_queued = "";
}
function next_movie() {
GM_xmlhttpRequest({
method : 'POST',
url : 'http://' + xbmc_address + '/jsonrpc',
headers : {
'Content-Type' : 'application/json'
},
data : '{"jsonrpc": "2.0", "method": "Player.GoTo", "params":{"playerid" : 1, "to" : "next" }, "id" : 1}'
});
xbmc_queued = "";
}
/*
* ============================================================================
* UI functions
* ============================================================================
*/
function modify_xbmc_address() {
xbmc_address = window
.prompt(
'Enter the address for the XBMC web interface\n(username:password@address:port)',
xbmc_address);
GM_setValue("XBMC_ADDRESS", xbmc_address);
}
//function modify_xbmc_playlist() {
// xbmc_music_playlist = window.prompt('Set the PARTYMODE playlist number (0 or 1)',
// xbmc_music_playlist);
// GM_setValue("XBMC_PLAYLIST", xbmc_music_playlist);
//}
function remove_playing_msg() {
try {
xbmc_ui.removeChild(xbmc_title);
} catch (e) {
// catch and just suppress error
}
try {
clearTimeout(xbmc_msg_timer);
} catch (e) {
// catch and just suppress error
}
}
function show_ui_msg(msg, timeout) {
remove_playing_msg();
xbmc_title.innerHTML = msg;
xbmc_ui.insertBefore(xbmc_title,xbmc_play_control);
xbmc_msg_timer = setTimeout(remove_playing_msg, timeout);
}
function add_play_on_xbmc_buttons() {
console.log('Found clip ' + document.documentURI);
xbmc_ui = document.createElement('div');
xbmc_ui.setAttribute('id', 'xbmc');
xbmc_play_control = document.createElement('div');
xbmc_play_control.setAttribute('id', 'playControl');
var xbmc_other_control = document.createElement('div');
xbmc_other_control.setAttribute('id', 'otherControl');
var xbmc_playback_control = document.createElement('div');
xbmc_playback_control.setAttribute('id', 'playbackControl');
xbmc_title = document.createElement('div');
xbmc_title.setAttribute('id', 'xbmcText');
xbmc_title.innerHTML = 'PLAYING';
var xbmc_play = document.createElement('span');
xbmc_play.addEventListener('click', function() {
play_movie()
}, false);
xbmc_play.setAttribute('id', 'btPlay');
xbmc_play.setAttribute('title', 'Start playback');
var xbmc_pause = document.createElement('span');
xbmc_pause.addEventListener('click', pause_movie, false);
xbmc_pause.setAttribute('id', 'btPause');
xbmc_pause.setAttribute('title', 'Pause playback');
var xbmc_stop = document.createElement('span');
xbmc_stop.addEventListener('click', stop_movie, false);
xbmc_stop.setAttribute('id', 'btStop');
xbmc_stop.setAttribute('title', 'Stop video');
xbmc_play_control.appendChild(xbmc_play); var xbmc_next = document.createElement('span');
xbmc_next.addEventListener('click', next_movie, false);
xbmc_next.setAttribute('id', 'btNext');
xbmc_next.setAttribute('title', 'Play next video');
var xbmc_queue = document.createElement('span');
xbmc_queue.addEventListener('click', queue_movie, false);
xbmc_queue.setAttribute('id', 'btQueue');
xbmc_queue.setAttribute('title', 'Queue video');
xbmc_play_control.appendChild(xbmc_play);
xbmc_playback_control.appendChild(xbmc_queue);
xbmc_playback_control.appendChild(xbmc_next);
xbmc_playback_control.appendChild(xbmc_pause);
xbmc_playback_control.appendChild(xbmc_stop);
xbmc_other_control.appendChild(xbmc_playback_control);
xbmc_ui.appendChild(xbmc_play_control);
xbmc_ui.appendChild(xbmc_other_control);
document.body.parentNode.insertBefore(xbmc_ui, document.body);
GM_addStyle('#xbmc { opacity:0.4; width:90px; position:fixed; z-index:100; bottom:0; right:0; display:block; background:#103040; -moz-border-radius-topleft: 20px; -moz-border-radius-bottomleft:20px; -webkit-border-top-left-radius:20px; -webkit-border-bottom-left-radius:20px; } ')
GM_addStyle('#xbmc:hover { opacity: 0.7; } ')
GM_addStyle('#xbmcText { opacity:0.8; font-family:Terminal; text-align:center; font-size:12px; font-weight:bold; background:#401010; color:#a0a0a0 } ')
// Play control
GM_addStyle('#playControl span, #playbackPlay span:hover { width:40px; height:40px; float:left; display:block; padding-bottom:0px; -moz-background-size:40px; background-size:40px; -webkit-background-size:40px; -o-background-size:40px; -khtml-background-size:40px; cursor:pointer; } ')
GM_addStyle('#btPlay { background: url("") no-repeat; } ')
GM_addStyle('#btPlay:hover { background: url("") no-repeat; } ')
// Other control
GM_addStyle('#playbackControl span, #playbackControl span:hover { width:20px; height:20px; bottom:0; float:left; display:block; margin-left:3px; -moz-background-size:20px; background-size:20px; -webkit-background-size:20px; -o-background-size:20px; -khtml-background-size:20px; cursor:pointer; } ')
GM_addStyle('#btQueue { background: url("") no-repeat; } ')
GM_addStyle('#btQueue:hover { background: url("") no-repeat; } ')
GM_addStyle('#btNext { background: url("") no-repeat; } ')
GM_addStyle('#btNext:hover { background: url("") no-repeat; } ')
GM_addStyle('#btStop { background: url("") no-repeat; } ')
GM_addStyle('#btStop:hover { background: url("") no-repeat; } ')
GM_addStyle('#btPause { background: url("") no-repeat; } ')
GM_addStyle('#btPause:hover { background: url("") no-repeat; } ')
}
/*
* ============================================================================
* Site dependent code here!!!!
* ============================================================================
*/
var supported_hosts = [ '180upload.com', '2gb-hosting.com', 'allmyvideos.net',
'auengine.com', 'bayfiles.com', 'bestreams.net', 'billionuploads.com',
'castamp.com', 'cheesestream.com', 'clicktoview.org', 'cloudy.ch',
'cloudy.com', 'cloudy.ec', 'cloudy.eu', 'cloudy.sx', 'crunchyroll.com',
'cyberlocker.ch', 'daclips.com', 'daclips.in', 'dailymotion.com',
'divxden.com', 'divxstage.eu', 'divxstage.net', 'divxstage.to',
'donevideo.com', 'ecostream.tv', 'entroupload.com', 'filebox.com',
'filedrive.com', 'filenuke.com', 'firedrive.com', 'flashx.tv',
'gorillavid.com', 'gorillavid.in', 'hostingbulk.com', 'hostingcup.com',
'hugefiles.net', 'jumbofiles.com', 'lemuploads.com', 'letwatch.us',
'limevideo.net', 'megarelease.org', 'mega-vids.com',
'mightyupload.com', 'mooshare.biz', 'movdivx.com', 'movieshd.co',
'movpod.in', 'movpod.net', 'movreel.com', 'movshare.net', 'movzap.com',
'mp4star.com', 'mp4stream.com', 'mp4upload.com', 'mrfile.me',
'muchshare.net', 'nolimitvideo.com', 'nosvideo.com', 'novamov.com',
'nowvideo.ch', 'nowvideo.eu', 'nowvideo.sx', 'ovile.com', 'play44.net',
'played.to', 'playwire.com', 'primeshare.tv', 'promptfile.com',
'purevid.com', 'putlocker.com', 'rapidvideo.com', 'realvid.net',
'seeon.tv', 'shared.sx', 'sharefiles4u.com', 'sharerepo.com',
'sharesix.com', 'sharevid.org', 'skyload.net', 'slickvid.com',
'sockshare.com', 'speedvideo.net', 'stagevu.com', 'stream2k.com',
'streamcloud.eu', 'streamin.to', 'ted.com', 'thefile.me',
'thevideo.me', 'trollvid.net', 'tubeplus.me', 'tune.pk', 'ufliq.com',
'uploadc.com', 'uploadcrazy.net', 'veehd.com', 'veoh.com',
'vidbull.com', 'vidbux.com', 'vidcrazy.net', 'video44.net',
'videobb.com', 'videoboxone.com', 'videofun.me', 'videohut.to',
'videomega.tv', 'videoraj.ch', 'videoraj.com', 'videoraj.ec',
'videoraj.eu', 'videoraj.sx', 'videotanker.co', 'videoweed.es',
'videozed.net', 'videozer.com', 'vidhog.com', 'vidpe.com',
'vidplay.net', 'vidspot.net', 'vidstream.in', 'vidto.me', 'vidup.org',
'vidxden.com', 'vidzi.tv', 'vidzur.com', 'vimeo.com', 'vk.com',
'vodlocker.com', 'vureel.com', 'watchfreeinhd.com', 'xvidstage.com',
'yourupload.com', 'youtu.be', 'youtube.com', 'youwatch.org',
'zalaa.com', 'zooupload.com', 'zshare.net', 'zuzvideo.com', ];
function binarySearch(items, value) {
var startIndex = 0, stopIndex = items.length - 1, middle = Math
.floor((stopIndex + startIndex) / 2);
while (items[middle] != value && startIndex < stopIndex) {
// adjust search area
if (value < items[middle]) {
stopIndex = middle - 1;
} else if (value > items[middle]) {
startIndex = middle + 1;
}
// recalculate middle
middle = Math.floor((stopIndex + startIndex) / 2);
}
// make sure it's the right value
return (items[middle] != value) ? -1 : middle;
}
/* Youtube has more features than most streaming sites,it needs special treatment
*/
function parse_yt_params(video_url) {
var regex = /[?&]([^=#]+)=([^&#]*)/g;
var params = {}, match;
while (match = regex.exec(video_url)) {
params[match[1]] = match[2];
}
return params;
}
function encode_video_url_for_queueing(video_url) {
switch (current_host) {
case "youtube.com":
case "youtu.be":
/* Youtube has it's own playlists, but Kodi doesn't support
* queueing a list within another list.
* Thus, we queue only current video. */
var yt_params = parse_yt_params(video_url);
return 'plugin://plugin.video.youtube/play/?video_id='
+ yt_params["v"];
break;
}
/* All other domains use the same URI for queueing and playing */
return encode_video_url(video_url);
}
function encode_video_url(video_url) {
switch (current_host) {
case "youtube.com":
case "youtu.be":
/* Better talk to YouTube plugin directly, it allows for more flexible use */
var yt_params = parse_yt_params(video_url);
if (yt_params["list"]) {
result = 'plugin://plugin.video.youtube/play/?play=1&order=default&playlist_id='
+ yt_params["list"];
if (yt_params["v"]) {
result = result + '&video_id=' + yt_params["v"];
}
return result;
}
return 'plugin://plugin.video.youtube/play/?video_id='
+ yt_params["v"];
break;
case "ted.com":
return 'plugin://plugin.video.ted.talks/?mode=playVideo&url='
+ encodeURIComponent(video_url)
+'&icon=a';
break;
}
return 'plugin://script.video.anyurl/?mode=play_video&url='
+ encodeURIComponent(video_url);
}
/* Add buttons only if necessary */
if (binarySearch(supported_hosts, current_host) >= 0 && top == self) {
GM_registerMenuCommand('Modify the XBMC address', modify_xbmc_address);
// GM_registerMenuCommand('XBMC partymode playlist', modify_xbmc_playlist);
// First run?
if (xbmc_address === undefined)
modify_xbmc_address();
//if (xbmc_music_playlist === undefined)
// modify_xbmc_playlist();
add_play_on_xbmc_buttons()
} else {
console.log("Unsupported host " + document.documentURI)
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment