Add Spotify to YouTube

Adds a link to Spotify on music videos on YouTube

当前为 2014-08-21 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name Add Spotify to YouTube
  3. // @description Adds a link to Spotify on music videos on YouTube
  4. // @include *://youtube.*/*
  5. // @include *://*.youtube.*/*
  6.  
  7.  
  8. // @version 0.0.1.20140821222704
  9. // @namespace https://greasyfork.org/users/3216
  10. // ==/UserScript==
  11.  
  12. var userscript = function()
  13. {
  14.  
  15. // http://stackoverflow.com/questions/2246901/how-can-i-use-jquery-in-greasemonkey-scripts-in-google-chrome
  16. function addJQuery(callback) {
  17. var script = document.createElement("script");
  18. script.setAttribute("src", "//ajax.googleapis.com/ajax/libs/jquery/1/jquery.min.js");
  19. script.addEventListener('load', function() {
  20. var script = document.createElement("script");
  21. script.textContent = "window.jQ=jQuery.noConflict(true);(" + callback.toString() + ")();";
  22. document.body.appendChild(script);
  23. }, false);
  24. document.body.appendChild(script);
  25. }
  26.  
  27. function main()
  28. {
  29.  
  30. var videoinfo = {url: top.location.href, complete: 0, text: "", timer: null, running: 0}
  31. var DEBUG = 0;
  32.  
  33. if(DEBUG) var DEBUGLOG = console;
  34. else { var TDEBUGLOG = function() { return {log: function() {}}; }; var DEBUGLOG = TDEBUGLOG(); }
  35.  
  36. function IsMusic()
  37. {
  38. if(jQ('#eow-title #watch-headline-show-title').length != 0) return 1;
  39. if(jQ('#eow-category').text().trim() == "Music") return 1;
  40. var metadata = jQ('.metadata-info-title');
  41. for(var i = 0; i < metadata.length; i++)
  42. {
  43. if(metadata[i].innerText.match(/Buy ".*?" on|Artist/) != null) return 1;
  44. }
  45.  
  46. if(jQ('#eow-title').text().split('-', 2).length == 2) return 1; // Might be dangerous!
  47. return 0;
  48. }
  49.  
  50. function GetVideoTitle()
  51. {
  52. var artistname;
  53. var trackname;
  54. var fullname;
  55. var artist = jQ('#eow-title #watch-headline-show-title');
  56. if(artist.length != 0 && !artistname)
  57. {
  58. artistname = artist.text();
  59. }
  60. var artist = jQ('.metadata-info');
  61. if(artist.length != 0 && !artistname)
  62. {
  63. for(var i = 0; i < artist.length; i++)
  64. {
  65. var tname = artist[i].innerText.match(/(Artist)[ \n\r]*(.*)/);
  66. if(tname)
  67. {
  68. if(tname[2].trim() != "Various Artists")
  69. {
  70. artistname = tname[2];
  71. break;
  72. }
  73. }
  74. }
  75. }
  76. if(!artistname)
  77. {
  78. var tname = jQ('#eow-title').text().split('-', 2);
  79. if(tname.length == 2)
  80. {
  81. artistname = tname[0].trim();
  82. }
  83. }
  84. var track = jQ('.metadata-info-title').text().match(/Buy "(.*?)" on/);
  85. if(track != null && !trackname)
  86. {
  87. trackname = track[1];
  88. }
  89. if(!trackname)
  90. {
  91. //trackname = jQ('#eow-title').text().replace(artistname, "").replace(/-/, '').trim();
  92. var tname = jQ('#eow-title').text().split(/[-]+/, 2);
  93. if(tname.length == 2)
  94. {
  95. trackname = tname[1].trim();
  96. }
  97. else
  98. {
  99. trackname = trackname = jQ('#eow-title').text().replace(/-/, '').trim();
  100. if(artistname) trackname = trackname.replace(artistname, "");
  101. }
  102. }
  103. if(!artistname && trackname)
  104. {
  105. artistname = " ";
  106. }
  107. if(artistname && trackname)
  108. {
  109. trackname = trackname.replace(/\((.*?)\)|\[(.*?)\]| (Re)?mastered| f(ea)?t(uring)?(.)? .*|\"|\'/gi, "").trim();
  110. artistname = artistname.replace(/f(ea)?t(uring)?(.)? .*/gi, "").trim(); // & .*
  111. return {artist: artistname, track: trackname};
  112. }
  113. else
  114. {
  115. if(!artistname) DEBUGLOG.log("failed to get artistname");
  116. if(!trackname) DEBUGLOG.log("failed to get trackname");
  117. }
  118. return 0;
  119. }
  120.  
  121. // http://stackoverflow.com/questions/3446170/escape-string-for-use-in-javascript-regex
  122. function escapeRegExp(str) {
  123. return str.replace(/[\-\[\]\/\{\}\(\)\*\+\?\.\\\^\$\|]/g, "\\$&");
  124. }
  125.  
  126. function FindOriginal(title, data)
  127. {
  128. extra = [];
  129. if(!/remix/i.test(title.track))
  130. {
  131. extra.push("remix");
  132. }
  133.  
  134. for(var i = 0; i < Math.min(data.info.num_results, 5); i++)
  135. {
  136. track = data.tracks[i];
  137. if(!new RegExp(escapeRegExp(title.track) + ' .*(radio|clean|live|remaster(ed)?|edit|' + extra.join('|') + ')', 'gi').test(track.name))
  138. {
  139. return i;
  140. }
  141. }
  142. return 0;
  143. }
  144.  
  145. function CheckAddress()
  146. {
  147. if(window.location.href.match(/^(https?\:\/\/(www\.)?youtube.com\/watch)/))
  148. {
  149. return 1;
  150. }
  151. return 0;
  152. }
  153.  
  154. function AddSpotify()
  155. {
  156. if(!CheckAddress()) return 1;
  157.  
  158. videoinfo.complete = 0;
  159. if(IsMusic())
  160. {
  161. var title = GetVideoTitle();
  162. DEBUGLOG.log("title", title);
  163. if(title != 0)
  164. {
  165. jQ.getJSON('//ws.spotify.com/search/1/track.json?q=' + encodeURIComponent(title.artist + " " + title.track), function(data)
  166. {
  167. DEBUGLOG.log("data", data);
  168. DEBUGLOG.log("fullname:", title.artist, title.track);
  169. if(data.info.num_results > 0)
  170. {
  171. var artists = [];
  172. var index = FindOriginal(title, data);
  173. DEBUGLOG.log("Index", index, "selected");
  174. var track = data.tracks[index];
  175. jQ.each(track.artists, function(key, val) { artists.push('<a href="' + val.href + '">' + val.name + '</a>'); });
  176. var artistsname = artists.join(", ");
  177. var trackname = track.name;
  178. var htmloutput = '<div class="youtubespotify" style="font-size: 70%"><a href="' + track.href + '">';
  179. htmloutput += '<img src="data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABUAAAAVCAYAAACpF6WWAAAAAXNSR0IArs4c6QAAAARnQU1BAACxjwv8YQUAAAAJcEhZcwAADsQAAA7EAZUrDhsAAAQhSURBVDhPhZVtSJ11GMZ/zznHoz56XHbsbFmtpgj5khuZWCO3hDnG+tLqy+hDRRBsxPoSFX4IIigZsQjWgogFoygI5iAsSpdrMqHhXEKzCUujhWe+TI969OiZL6frfp4jKS12w9/n5fz/131d133fj05GwW1iLn2D38eOE5/tYCZ9lZXVFMGAy6a8SkqL9lIVO0Jh7r3Z3RvjP6BLKwucvlJPfGaASC7khiAnAAEHVrVzeRXSy5C8hcCrebaml5xgfva0HxtAryfO8nV/s9hAfo7Agj5gSNdgFnRJoMsrlhwWBD6zCAe3d7K1eE8WRQSyV/5KdPJFXzPFAswTu5BAbBlwWCuknXYNWxItS5anZ9v/5eVm7/xaeExN8tEul6irQ2KYFxZTk677sMnX4VUxzJh0yV5Mw60sW1t2P5mCN5tSnhUe6KneGqZSA0TvgoICGP0TRq7B9LgAtNmkF0SgZAtsrZCX5ahw8nVOBZ0XqGxIKdndbjUv1F/BSS7GM8fOlXJ/DL7/HNpOZDXcIcoqoek52HNQKvQ8nYAJJXm9KU5w30vhd0aTF4iIyTcfQkLs8mTDw49C7ePwSIPud8ADYle4CeaTYib5iZvwazec/kRqxHbHLv99KBDGOXWxLjM+1+cVJ1KkrEsC2OZLtoIEtRzdo8pbn1gHTAmwux3OnIThAY8477WpDlGI5tfhHPvJ1TYzWC9VyckR6NGBod/k7XV5NiNwFS66GR6Un1WPwZNPw5aH/Nbq74HBfmh8xveYjIvT+iMZrx+1jOnLdX5mC0csTfKyirAgiesjdh+82AIN++TnlGzR7zYYVjTng05XqlKeVFeDMdirKqpotU8okSyxSTL5ASWdHIW+8/DzGbh0zgff3givfQSz0/5gOIjpyZ66zFiyj1wdMv+K7/F9HZb88az8gL1XovIa2FblK5iehI/f0rtaST8gA9V6xnJzkTztGmzJdP/RiivfCtUBX7XChW99Fv8XO/fD828okXw2hgsCtL61Xm2saMGZXYhn3v2ulJJCydUEJSRx4m+oVEGMdUjvzAIbgpFh+OUHaFfVLQ4dVeFkU1rzb6CT8vXt/XF/oj7trmF0dsBrKwO2tarKWhvZMk9tBe13G195365BqdkpdcX+qKZlWUxfrUO7NFEGarPf0uZSrBG1/jRmawXyelRhF8uxFiGBW7NbWxnLaSl5/4A/+7Jc2XVz+KkOJjQta5usPby17tm76tmYpSR1Sezs+abOHd7d4eFYeEy9O8XQxFlOdDUT0RDY18ljqvdrbC1st+eK/li15+Tnq02dlMX+/Z5uALUwKz47X8+1sQEKJNH7QEuPJ187V7SM3bwqXRGr5pXdd/jyr4/k4g0uDh9naLxD8q6SXkmpl11KIpWUx/bSUHZEim73Pwr+AWfotNJ4G681AAAAAElFTkSuQmCC">';
  180. htmloutput += ' ' + trackname + '</a> - ' + artistsname + '</div>';
  181. jQ('.youtubespotify').remove();
  182. jQ('#watch-headline-title').append(htmloutput);
  183. }
  184. else
  185. {
  186. jQ('.youtubespotify').remove();
  187. DEBUGLOG.log("No song found");
  188. }
  189. });
  190. } else
  191. {
  192. jQ('.youtubespotify').remove();
  193. DEBUGLOG.log("failed to get title");
  194. }
  195. }
  196. else
  197. {
  198. DEBUGLOG.log("not music!!");
  199. }
  200. videoinfo.complete = 1;
  201. videoinfo.url = top.location.href;
  202. videoinfo.text = jQ('#eow-title').text().trim();
  203. }
  204.  
  205. AddSpotify();
  206.  
  207. function timerfunction()
  208. {
  209. videoinfo.running = 1;
  210. if(videoinfo.complete && videoinfo.url != top.location.href)
  211. {
  212. if(videoinfo.text != jQ('#eow-title').text().trim())
  213. {
  214. AddSpotify();
  215. }
  216. }
  217. }
  218.  
  219. window.addEventListener('focus', function()
  220. {
  221. clearInterval(videoinfo.timer);
  222. if(!videoinfo.running) videoinfo.timer = setInterval(timerfunction, CheckAddress() ? 1000 : 2000);
  223. });
  224.  
  225. window.addEventListener('blur', function()
  226. {
  227. clearInterval(videoinfo.timer);
  228. videoinfo.running = 0;
  229. });
  230.  
  231. }
  232.  
  233. addJQuery(main);
  234.  
  235. };
  236.  
  237. // http://stackoverflow.com/questions/7971930/how-to-call-youtube-flash-api-of-existing-videos-using-greasemonkey
  238. function addJS_Node (text, s_URL, funcToRun, runOnLoad) {
  239. var D = document;
  240. var scriptNode = D.createElement ('script');
  241. if (runOnLoad) {
  242. scriptNode.addEventListener ("load", runOnLoad, false);
  243. }
  244. scriptNode.type = "text/javascript";
  245. if (text) scriptNode.textContent = text;
  246. if (s_URL) scriptNode.src = s_URL;
  247. if (funcToRun) scriptNode.textContent = '(' + funcToRun.toString() + ')()';
  248.  
  249. var targ = D.getElementsByTagName ('head')[0] || D.body || D.documentElement;
  250. targ.appendChild (scriptNode);
  251. }
  252.  
  253. addJS_Node(null, null, userscript);