Add Spotify to YouTube

Adds a link to Spotify on music videos on YouTube

当前为 2014-07-04 提交的版本,查看 最新版本

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