Plex External Player

Play plex videos in an external player

当前为 2016-04-23 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name Plex External Player
  3. // @namespace https://github.com/Kayomani/PlexExternalPlayer
  4. // @version 1.7
  5. // @description Play plex videos in an external player
  6. // @author Kayomani
  7. // @include /^https?://.*:32400/web.*
  8. // @include http://*:32400/web/index.html*
  9. // @require http://code.jquery.com/jquery-1.11.3.min.js
  10. // @connect *
  11. // @require https://cdnjs.cloudflare.com/ajax/libs/toastr.js/latest/js/toastr.min.js
  12. // @grant GM_xmlhttpRequest
  13. // ==/UserScript==
  14.  
  15. $("head").append (
  16. '<link href="//cdnjs.cloudflare.com/ajax/libs/toastr.js/latest/css/toastr.min.css" rel="stylesheet" type="text/css">'
  17. );
  18.  
  19.  
  20. toastr.options = {
  21. "closeButton": true,
  22. "debug": false,
  23. "newestOnTop": true,
  24. "progressBar": true,
  25. "positionClass": "toast-bottom-right",
  26. "preventDuplicates": false,
  27. "onclick": null,
  28. "showDuration": "300",
  29. "hideDuration": "1000",
  30. "timeOut": "5000",
  31. "extendedTimeOut": "1000",
  32. "showEasing": "swing",
  33. "hideEasing": "linear",
  34. "showMethod": "fadeIn",
  35. "hideMethod": "fadeOut"
  36. };
  37.  
  38. var showToast = function(msg, error){
  39. var title = 'Plex External Player';
  40. if(error){
  41. toastr.error(msg, title, {timeOut: 10000});
  42. logMessage(msg);
  43. }
  44. else
  45. {
  46. toastr.success(msg, title);
  47. }
  48. };
  49.  
  50. var logMessage = function(msg){
  51. console.log('Plex External: ' + msg);
  52. };
  53.  
  54. var makeRequest = function(url){
  55. return new Promise( function (resolve, reject) {
  56. var origAccessToken = localStorage.myPlexAccessToken;
  57. var serverNode = JSON.parse(localStorage.users).users[1].servers[0];
  58. var newAccessToken = typeof serverNode != 'undefined' ? serverNode.accessToken : origAccessToken;
  59. GM_xmlhttpRequest({
  60. method: "GET",
  61. headers: {
  62. "X-Plex-Token": newAccessToken
  63. },
  64. url: url,
  65. onload: resolve,
  66. onreadystatechange: function(state) {
  67. if (state.readyState === 4) {
  68. if (state.status !== 200) {
  69. showToast('Error calling: ' + url + '. Response: ' + state.responseText + ' Code:' + state.status + ' Message: ' + state.statusText, 1);
  70. }
  71. }
  72. },
  73. onerror: reject
  74. });
  75. });
  76. };
  77.  
  78.  
  79.  
  80. var markAsPlayedInPlex = function(id) {
  81. logMessage('Marking ' + id + ' as played');
  82. return makeRequest(window.location.origin + '/:/scrobble?key='+ id +'&identifier=com.plexapp.plugins.library').catch(function(){
  83. showToast('Failed to mark item ' + id + ' as played');
  84. });
  85. };
  86.  
  87. var openItemOnAgent = function(path, id, openFolder) {
  88. if(openFolder){
  89. var fwd = path.lastIndexOf('/');
  90. var bck = path.lastIndexOf('\\');
  91. var best = fwd>bck?fwd:bck;
  92. if(best>-1){
  93. path = path.substr(0, best);
  94. }
  95. }
  96. showToast('Playing ' + path, 0);
  97. logMessage('Playing ' + path);
  98. // umicrosharp doesn't handle plus properly
  99. path = path.replace(/\+/g, '[PLEXEXTPLUS]');
  100. var url = 'http://127.0.0.1:7251/?protocol=2&item=' + encodeURIComponent(path);
  101. return new Promise(function (resolve, reject) {
  102. makeRequest(url).then(function(){
  103. markAsPlayedInPlex(id).then(resolve, reject);
  104. },reject);
  105. });
  106. };
  107.  
  108. var clickListener = function(e) {
  109. e.preventDefault();
  110. e.stopPropagation();
  111. var a = jQuery(e.target).closest('a');
  112. var link = a.attr('href');
  113. var openFolder = a.attr('data-type') === 'folder';
  114. var url = link;
  115. if (link === '#' || link === undefined || link === 'javascript:void(0)') {
  116. url = window.location.hash;
  117. }
  118.  
  119. if (url.indexOf('%2Fmetadata%2F') > -1) {
  120. var idx = url.indexOf('%2Fmetadata%2F');
  121. var id = url.substr(idx + 14);
  122.  
  123. // Get metadata
  124. var metaDataPath = window.location.origin + '/library/metadata/' + id + '?checkFiles=1&includeExtras=1';
  125. makeRequest(metaDataPath)
  126. .then(function(response){
  127. // Play the first availible part
  128. var parts = response.responseXML.getElementsByTagName('Part');
  129. for (var i = 0; i < parts.length; i++) {
  130. if (parts[i].attributes['file'] !== undefined) {
  131. openItemOnAgent(parts[i].attributes['file'].value, id, openFolder).catch(function(){
  132. showToast('Failed to connect to agent, is it running or firewalled?',1);
  133. });
  134. return;
  135. }
  136. }
  137.  
  138. if (parts.length === 0) {
  139. // If we got a directory/Season back then get the files in it
  140. var dirs = response.responseXML.getElementsByTagName('Directory');
  141. if (dirs.length > 0) {
  142. makeRequest(window.location.origin + dirs[0].attributes['key'].value)
  143. .then(function(response){
  144. var videos = response.responseXML.getElementsByTagName('Video');
  145. var file = null;
  146. var id = null;
  147. if(videos.length === 0)
  148. {
  149. showToast('Could not determine which video to play as there are multiple seasons.',true);
  150. return;
  151. }
  152. for (var i = 0; i < videos.length; i++) {
  153. var vparts = videos[i].getElementsByTagName('Part');
  154. if (vparts.length > 0) {
  155. file = vparts[0].attributes['file'].value;
  156. id = vparts[0].attributes['id'].value;
  157. if (videos[i].attributes['lastViewedAt'] === null || videos[i].attributes['lastViewedAt'] === undefined) {
  158. break;
  159. }
  160. }
  161. }
  162.  
  163. if (file !== null)
  164. {
  165. openItemOnAgent(file, id, openFolder).catch(function(){
  166. showToast('Failed to connect to agent, is it running or firewalled?',1);
  167. });
  168. }
  169. }).catch(function(){
  170. showToast('Failed to get information for directory',1);
  171. });
  172. }
  173. }
  174. }, function(error){
  175. showToast('Error getting metadata from' + metaDataPath, 1);
  176. });
  177. }
  178. };
  179.  
  180. var bindClicks = function() {
  181. jQuery(".glyphicon.play").each(function(i, e) {
  182. e = jQuery(e);
  183. if (!e.hasClass('plexextplayer')) {
  184. if (!e.parent().hasClass('hidden')) {
  185. e.addClass('plexextplayer');
  186. var parent = e.parent().parent();
  187. if (parent.is('li')) {
  188. var template = jQuery('<li><a class="btn-gray" href="javascript:void(0)" title="Play Externally" data-toggle="Play Externally" data-original-title="Play Externally"><i class="glyphicon play plexextplayer plexextplayerico"></i></a></li><li><a class="btn-gray" href="#" title="Open containing folder" data-type="folder" data-toggle="Play Externally" data-original-title="Open containing folder"><i class="glyphicon play plexextplayer plexfolderextplayerico"></i></a></li>');
  189. parent.after(template);
  190. template.click(clickListener);
  191. } else if (parent.is('div') && parent.hasClass('media-poster-actions')) {
  192. var template = jQuery('<button class="play-btn media-poster-btn btn-link" tabindex="-1"><i class="glyphicon play plexextplayer plexextplayerico"></i></button>');
  193. parent.prepend(template);
  194. template.click(clickListener);
  195. }
  196. }
  197. }
  198. });
  199. // React cover page
  200. jQuery(".plex-icon-more-560").each(function(i, e) {
  201. e = jQuery(e);
  202. var poster = e.parent().parent();
  203. if(poster.length === 1 && poster[0].className.startsWith('MetadataPosterCardOverlay'))
  204. {
  205. var existingButton = poster.find('.plexextplayerico');
  206. if(existingButton.length === 0)
  207. {
  208. var url = poster.find('a').attr('href');
  209. var template = jQuery('<a href="'+ url +'" aria-haspopup="false" aria-role="button" class="" type="button"><i class="glyphicon play plexextplayer plexextplayerico plexextplayericocover"></i></button>');
  210. var newButton = template.appendTo(poster);
  211. newButton.click(clickListener);
  212. poster.mouseenter(function(){
  213. newButton.find('i').css('display','block');
  214. });
  215. poster.mouseleave(function(){
  216. newButton.find('i').css('display','none');
  217. });
  218. }
  219. }
  220. });
  221. };
  222.  
  223. // Make buttons smaller
  224. jQuery('body').append('<style>.plexextplayericocover {right: 10px; top: 10px; position:absolute; display:none;font-size:15px;} .glyphicon.plexfolderextplayerico:before { content: "\\e343"; } .glyphicon.plexextplayerico:before { content: "\\e161"; }</style>');
  225.  
  226. // Bind buttons and check for new ones every 100ms
  227. setInterval(bindClicks, 100);
  228. bindClicks();