YouTube Thumbnails - Full Video Thumbnails for YouTube

Shows complete video thumbnails for YouTube videos. You can click a thumbnail image to jump to that point in the video.

当前为 2018-08-09 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name YouTube Thumbnails - Full Video Thumbnails for YouTube
  3. // @namespace driver8.net
  4. // @description Shows complete video thumbnails for YouTube videos. You can click a thumbnail image to jump to that point in the video.
  5. // @match *://*.youtube.com/*
  6. // @version 0.4.5
  7. // @require https://ajax.googleapis.com/ajax/libs/jquery/2.2.4/jquery.min.js
  8. // @grant GM_addStyle
  9. // @grant unsafeWindow
  10. // @grant GM_registerMenuCommand
  11. // @grant GM_getValue
  12. // @grant GM_setValue
  13. // @grant GM_xmlhttpRequest
  14. // @run-at document-idle
  15. // ==/UserScript==
  16.  
  17. var AND_BUTTS = GM_getValue('ytAndButts', false);
  18. var BEFORE_TITLE = true;
  19. var TIMEOUT_DELAY = 50;
  20. var MAX_TRIES = 1;
  21. var DIV_PADDING = 10;
  22. var TD_PADDING = 2;
  23. var DISABLE_SPF = GM_getValue('ytDisableSPF', false);
  24. var MIN_WIDTH = 4;
  25. var MAX_IMAGES = 400;
  26. var LOGGING = true;
  27.  
  28. var $thumbDiv, $thumbHeader, $titleDiv, storyboard_spec, storyboard, best_size_idx, best_size, len_seconds = 0, video_id, tries, yp,
  29. scriptStarted = false, scriptFinished = false;
  30.  
  31. function log(...msg) {
  32. LOGGING && console.log(...msg);
  33. }
  34.  
  35. console.log("Hi, yt thumbs");
  36.  
  37. function findStuff(event) {
  38. event && console.log(event);
  39. if (!window.location.href.match(/watch\?/) || (scriptStarted && !scriptFinished)) {
  40. console.log('scriptStarted', scriptStarted, 'scriptFinished', scriptFinished);
  41. console.log("don't run");
  42. return;
  43. }
  44. console.log("started...");
  45. scriptStarted = true;
  46. scriptFinished = false;
  47.  
  48. yp = (window && window.ytplayer) || (unsafeWindow && unsafeWindow.ytplayer);
  49. storyboard_spec = yp && yp.config && yp.config.args.storyboard_spec;
  50. if (storyboard_spec) {
  51. len_seconds = parseInt(yp.config.args.length_seconds);
  52. video_id = yp.config.args.video_id;
  53. }
  54. console.log('yp', yp, 'sb_spec', storyboard_spec);
  55.  
  56. var m = document.location.href.match(/(?:watch\?.*v=|\/v\/)([\w\-=]+)/);
  57. var current_id = m[1];
  58.  
  59. if ($('#thumbDiv.yes-thumbs').size() > 0) {
  60. console.log('$', $('#thumbDiv.yes-thumbs'), 'current_id', current_id, 'existing id', $('#thumbDiv.yes-thumbs').data('video_id'));
  61. if ($('#thumbDiv.yes-thumbs').data('video_id') === current_id) {
  62. scriptFinished = true;
  63. return;
  64. } else {
  65. $('#thumbDiv.yes-thumbs').remove();
  66. }
  67. } else if ($('#thumbDiv').size() > 0) { //no-thumbs
  68. $('#thumbDiv').remove();
  69. }
  70.  
  71. if (video_id === current_id) {
  72. tryTitle();
  73. } else {
  74. storyboard_spec = null;
  75. var y = window.ytcfg || unsafeWindow.ytcfg;
  76. console.log('ytcfg', y);
  77. var reqHeaders = {
  78. //"Referer": document.location.href,
  79. "X-SPF-Previous": document.location.href,
  80. "X-SPF-Referer": document.location.href,
  81. "X-YouTube-Client-Name": y.data_.INNERTUBE_CONTEXT_CLIENT_NAME,
  82. "X-YouTube-Client-Version": y.data_.INNERTUBE_CONTEXT_CLIENT_VERSION,
  83. //"X-Youtube-Identity-Token": y.data_.XSRF_TOKEN,
  84. "X-Youtube-Identity-Token": y.data_.ID_TOKEN,
  85. "X-YouTube-Page-CL": y.data_.PAGE_CL,
  86. "X-YouTube-Page-Label": y.data_.PAGE_BUILD_LABEL,
  87. "X-YouTube-STS": y.data_.FILLER_DATA.player.sts,
  88. "X-YouTube-Utc-Offset": 0,
  89. "X-YouTube-Variants-Checksum": y.data_.VARIANTS_CHECKSUM
  90. };
  91. console.log('req headers', reqHeaders);
  92. var oReq = new XMLHttpRequest();
  93. oReq.addEventListener('load', function(evt) {
  94. console.log('evt', evt, 'resp', this.response);
  95. try {
  96. var resp = this.response;
  97. storyboard_spec = resp[2].player.args.storyboard_spec;
  98. len_seconds = parseInt(resp[2].player.args.length_seconds);
  99. video_id = resp[2].player.args.video_id;
  100. console.log('sb_spec', storyboard_spec);
  101. } catch (e) {
  102. console.log('problem loading storyboard_spec from json', e);
  103. }
  104. tries = 0;
  105. tryTitle();
  106. });
  107. oReq.open('GET', document.location.href + '&pbj=1');
  108. oReq.responseType = 'json';
  109. for (var h in reqHeaders) {
  110. oReq.setRequestHeader(h, reqHeaders[h]);
  111. }
  112. oReq.send();
  113. // GM_xmlhttpRequest({
  114. // method: 'GET',
  115. // url: document.location.href + '&pbj=1',
  116. // headers: reqHeaders,
  117. // onload: function (resp) {
  118. // console.log(resp);
  119. // try {
  120. // var foo = JSON.parse(resp.responseText);
  121. // console.log(foo);
  122. // storyboard_spec = foo[2].player.args.storyboard_spec;
  123. // len_seconds = parseInt(foo[2].player.args.length_seconds);
  124. // video_id = foo[2].player.args.video_id;
  125. // console.log(storyboard_spec);
  126. // } catch (e) {
  127. // console.log('problem loading storyboard_spec from json', e);
  128. // }
  129. // tries = 0;
  130. // tryTitle();
  131. // }
  132. // });
  133. }
  134. }
  135.  
  136. function tryTitle() {
  137. console.log('tryTitle');
  138. //$titleDiv = $('div#container.ytd-video-primary-info-renderer, div#info.ytd-watch, #watch-header');
  139. //$titleDiv = $('ytd-video-primary-info-renderer.ytd-watch-flexy, div#info.ytd-watch, #watch-header');
  140. $titleDiv = $('div#info.ytd-watch-flexy, div#info.ytd-watch, #watch-header');
  141. if (!$titleDiv || $titleDiv.prop('nodeName') !== 'DIV') {
  142. console.log("no title div");
  143. if (tries++ < MAX_TRIES) {
  144. console.log("trying again for title div");
  145. window.setTimeout(tryTitle, TIMEOUT_DELAY);
  146. } else {
  147. console.log("done trying for title div");
  148. scriptFinished = true;
  149. }
  150. } else {
  151. console.log('found title div', $titleDiv);
  152. setUp();
  153. }
  154. }
  155.  
  156. function setUp() {
  157. if (storyboard_spec) {
  158. console.log('sb_spec', storyboard_spec);
  159. storyboard = parseStoryboardSpec(storyboard_spec);
  160. console.log('sb', storyboard);
  161. best_size_idx = chooseBestStoryboardSize(storyboard);
  162. best_size = storyboard.sizes[best_size_idx];
  163. console.log(best_size);
  164. if (! len_seconds > 0) {
  165. var len_str = $('div.ytp-progress-bar').attr('aria-valuemax') || yp.config.args.length_seconds;
  166. len_seconds = parseInt(len_str);
  167. }
  168. console.log("len: ", len_seconds);
  169.  
  170. $thumbDiv = $('<div id="thumbDiv" class="style-scope yt-card yes-thumbs thumb-inactive"></div>');
  171. $thumbDiv.data('video_id', video_id);
  172. $thumbHeader = $('<h1 width="100%" class="title ytd-video-primary-info-renderer">Thumbnails</h1>');
  173. $thumbDiv.append($thumbHeader);
  174. $thumbDiv.click(showThumbs);
  175. } else if (unsafeWindow.ytplayer) {
  176. $thumbDiv = $('<div id="thumbDiv" class="style-scope yt-card no-thumbs"></div>');
  177. $thumbHeader = $('<h1 width="100%" class="title ytd-video-primary-info-renderer">No Thumbnails Available</h1>');
  178. $thumbDiv.append($thumbHeader);
  179. } else {
  180. console.log("Reload for thumbs");
  181. $thumbDiv = $('<div id="thumbDiv" class="style-scope yt-card no-thumbs"></div>');
  182. $thumbHeader = $('<h1 width="100%" class="title ytd-video-primary-info-renderer">Reload for Thumbnails</h1>');
  183. $thumbDiv.append($thumbHeader);
  184. }
  185.  
  186. if (BEFORE_TITLE) {
  187. $titleDiv.before($thumbDiv);
  188. } else {
  189. $titleDiv.after($thumbDiv);
  190. }
  191. console.log($titleDiv);
  192.  
  193. console.log("Script done");
  194.  
  195. styleIt();
  196. console.log("finished.");
  197. scriptFinished = true;
  198. }
  199.  
  200. function showThumbs(event) {
  201. var duration = best_size.duration / 1000;
  202. var num_thumbs = 1;
  203. if (duration > 0) {
  204. num_thumbs = Math.ceil(len_seconds / duration / best_size.cols / best_size.rows);
  205. } else {
  206. duration = len_seconds / best_size.cols / best_size.rows;
  207. }
  208. console.log("Thumb header clicked. Loading " + num_thumbs + " images.");
  209. var total_width = best_size.width * best_size.cols + DIV_PADDING * 2;
  210. var parent_diff = $thumbDiv.parent().innerWidth() - total_width;
  211. parent_diff < 0 && $thumbDiv.css({'left': + parent_diff + 'px'});
  212. // $thumbDiv.css({'position': 'relative', 'width': total_width + 'px'});
  213. // $thumbDiv.css({'position': 'relative'});
  214. $thumbDiv.removeClass('thumb-inactive');
  215. $thumbDiv.addClass('thumb-active');
  216.  
  217. // Grab the youtube sessionlink ID for time links
  218. var sessionlink = $('#logo-container').data("sessionlink");
  219.  
  220. var badImage = false;
  221. (function loadImage(imgNum) {
  222. if (badImage === false && imgNum < num_thumbs && imgNum < MAX_IMAGES) {
  223. // EX: https:\/\/i.ytimg.com\/sb\/2XY3AvVgDns\/storyboard3_L$L\/$N.jpg
  224. // EX: https://i.ytimg.com/sb/k4YRWT_Aldo/storyboard3_L2/M0.jpg?sigh=RVdv4fMsE-eDcsCUzIy-iCQNteI
  225. var link = storyboard.baseUrl.replace('\\', '');
  226. link = link.replace('$L', best_size_idx);
  227. link = link.replace('$N', best_size.img_name);
  228. link = link.replace('$M', imgNum);
  229. link += '?sigh=' + best_size.sigh;
  230.  
  231. console.log(link);
  232.  
  233. // Create a table for the timestamps over the image
  234. var $newTable = $('<table>');
  235. $newTable.addClass('imgTable');
  236. var x = imgNum * duration * best_size.rows * best_size.cols; // the starting time for this table
  237. var innerStr = '';
  238. var doclocation = document.location.href.replace(/\#.*/, '');
  239. for (var i = 0; i < best_size.rows; i++) {
  240. if (x <= len_seconds + 1) { // if we haven't reached the end of the video
  241. //console.log( 'x', x, 'len_seconds', len_seconds);
  242. innerStr += '<tr>';
  243. for (var j = 0; j < best_size.cols; j++) {
  244. innerStr += '<td><a href="' + doclocation + '#t=' + x + '" data-seek-to="' + x + '">';
  245. if (x <= len_seconds + 1) {
  246. var hrs = Math.floor(x / 3600);
  247. var mins = Math.floor((x % 3600) / 60);
  248. var secs = Math.round(x % 60);
  249. innerStr += hrs > 0 ? hrs + ':' : ''; // hrs
  250. innerStr += ( hrs > 0 && mins < 10 ? "0" + mins : mins ) + ':'; // mins
  251. innerStr += secs < 10 ? "0" + secs : secs; // secs
  252. }
  253. innerStr += '</a></td>';
  254. x += duration;
  255. }
  256. innerStr += '<tr>';
  257. }
  258. }
  259. $newTable.html(innerStr);
  260. $newTable.error(function() {
  261. badImage = true;
  262. $(this).remove();
  263. console.log("Hid bad image");
  264. });
  265. //$newTable.load(function() {
  266. // loadImage(imgNum + 1);
  267. //});
  268. $newTable.css({'background-image': 'url(' + link + ')', 'width': best_size.width * best_size.cols});
  269.  
  270. $thumbDiv.append($newTable);
  271.  
  272. //setTimeout(loadImage, Math.min(LOAD_DELAY_START + imgNum * LOAD_DELAY_FACTOR, LOAD_DELAY_MAX), imgNum + 1);
  273. setTimeout(loadImage, 1, imgNum + 1);
  274. }
  275.  
  276. $('.imgTable a').click(function(event) {
  277. event.preventDefault();
  278. var seekTime = parseInt($(this).data('seekTo'));
  279. console.log('seek time', seekTime);
  280. $('video.html5-main-video')[0].currentTime = seekTime;
  281. $('html, body').scrollTop(0);
  282. event.stopImmediatePropagation();
  283. });
  284. })(0);
  285.  
  286. console.log("Done making thumb div");
  287. $thumbDiv.off('click');
  288. $thumbDiv.click(hideThumbs);
  289. }
  290.  
  291. function hideThumbs(event) {
  292. if ($(event.target).is('#thumbDiv, #thumbDiv h1')) {
  293. $thumbDiv.children('table').hide();
  294. $thumbDiv.off('click');
  295. $thumbDiv.click(showThumbsAgain);
  296. $thumbDiv.removeClass('thumb-active');
  297. $thumbDiv.addClass('thumb-inactive');
  298. } else {
  299. $('html, body').scrollTop(0);
  300. }
  301. }
  302.  
  303. function showThumbsAgain(event) {
  304. if ($(event.target).is('#thumbDiv, #thumbDiv h1')) {
  305. $thumbDiv.children('table').show();
  306. $thumbDiv.off('click');
  307. $thumbDiv.click(hideThumbs);
  308. $thumbDiv.removeClass('thumb-inactive');
  309. $thumbDiv.addClass('thumb-active');
  310. }
  311. }
  312.  
  313. function parseStoryboardSpec(spec) {
  314. // EX: https:\/\/i.ytimg.com\/sb\/Pk2oW4SDDxY\/storyboard3_L$L\/$N.jpg|48#27#100#10#10#0#default#vpw4l5h3xmm2AkCT6nMZbvFIyJw|80#45#90#10#10#2000#M$M#hCWDvBSbgeV52mPYmOHjgdLFI1o|160#90#90#5#5#2000#M$M#ys1MKEnwYXA1QAcFiugAA_cZ81Q
  315. var sections = spec.split('|');
  316. console.log(sections);
  317. var sb = {
  318. sizes: [],
  319. baseUrl: sections.shift()
  320. };
  321.  
  322. // EX: 80#45#90#10#10#2000#M$M#hCWDvBSbgeV52mPYmOHjgdLFI1o
  323. // EX: 160# 90# 90# 5# 5# 2000# M$M# ys1MKEnwYXA1QAcFiugAA_cZ81Q
  324. sections.forEach(function(value, idx) {
  325. var data = value.split('#');
  326. console.log(data);
  327. var new_size = {
  328. width : +data[0],
  329. height : +data[1],
  330. qual : +data[2], // image quality???
  331. cols : +data[3],
  332. rows : +data[4],
  333. duration : +data[5], // duration of each snapshot in milliseconds
  334. img_name : data[6],
  335. sigh : data[7]
  336. };
  337. sb.sizes[idx] = new_size;
  338. });
  339. console.log(sb);
  340. return sb;
  341. }
  342.  
  343. function chooseBestStoryboardSize(sb) {
  344. var sizes = sb.sizes;
  345. var widest = 0;
  346. var widest_idx = -1;
  347. for (var i = 0; i < sizes.length; i++) {
  348. if (widest < sizes[i].width || (widest == sizes[i].width && sizes[widest_idx].cols < sizes[i].cols)) {
  349. if (sizes[i].cols >= MIN_WIDTH) {
  350. widest = sizes[i].width;
  351. widest_idx = i;
  352. }
  353. }
  354. }
  355. return widest_idx;
  356. }
  357.  
  358.  
  359. function styleIt() {
  360. console.log("styling");
  361.  
  362. var td_width = best_size ? best_size.width - 2 * TD_PADDING : 10;
  363. var td_height = best_size ? best_size.height - 2 * TD_PADDING : 10;
  364. var userStyles0 =
  365. "table.imgTable { border-collapse: collapse; background-repeat: no-repeat; cursor: auto; background-color: rgba(0, 0, 0, 0) !important; } " +
  366. ".imgTable tbody { background-color: rgba(0, 0, 0, 0) !important; }" +
  367. ".imgTable td { width: " + td_width + "px;" +
  368. " height: " + td_height + "px;" +
  369. " padding: " + TD_PADDING + "px;" +
  370. " border-width: 0px;" +
  371. " background-color: rgba(0, 0, 0, 0) !important;" +
  372. " vertical-align: top;" +
  373. " color: white;" +
  374. " text-shadow:" +
  375. " -1px -1px 0 #000, " +
  376. " 1px -1px 0 #000, " +
  377. " -1px 1px 0 #000, " +
  378. " 1px 1px 0 #000; " +
  379. " !important}" +
  380. ".imgTable a { text-decoration: none; color: white; font-size: small; display: block; width: 100%; height: 100%; } " +
  381. "#thumbDiv { cursor: pointer; padding-right: var(--watch-sidebar-width); } " +
  382. //".thumb-inactive { background-image: url(data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAMAAAAoLQ9TAAAAD1BMVEXAwMABAQEBAQEAAAAAAADcECccAAAABHRSTlMAPEjK3fOUzQAAADFJREFUGNNjYGRmQQLMjAzMTAxIgImZgYUBBbDgFGBhIU0AZiluATIMxe9SDM+hex8AtbABVeH7XA4AAAAASUVORK5CYII=); }" +
  383. //".thumb-active { background-image: url(data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAMAAAAoLQ9TAAAAD1BMVEXAwMABAQEBAQEAAAAAAADcECccAAAABHRSTlMAPEjJRPrFdwAAACtJREFUGNNjYGRmQQLMjAzMTAxIgImZgYUBBbBQTQBmKW4BqluL4Tl07wMAnDABJd8eqrEAAAAASUVORK5CYII=); }" +
  384. ".thumb-active h1 { margin-bottom: 2px; }" +
  385. ".no-thumbs { cursor: default !important; }" +
  386. ".no-thumbs h1 { color: grey !important; }";
  387.  
  388. //.yt-simple-endpoint{display:inline-block;cursor:pointer;text-decoration:none;color:var(--yt-endpoint-color, hsl(0, 0%, 6.7%));}.yt-simple-endpoint:hover{color:var(--yt-endpoint-hover-color, var(--yt-endpoint-color));text-decoration: none;}
  389. var userStyles1 =
  390. "#video-title.old-endpoint.ytd-grid-video-renderer {" +
  391. "color:var(--yt-primary-color);" +
  392. "display:block;" +
  393. "max-height:3.2rem;" +
  394. "overflow:hidden;" +
  395. "font-size:1.4rem;" +
  396. "font-weight:500;" +
  397. "line-height:1.6rem;" +
  398. "}" +
  399. ".old-endpoint{display:inline-block;cursor:pointer;text-decoration:none;color:var(--yt-endpoint-color, hsl(0, 0%, 6.7%));}.old-endpoint:hover{color:var(--yt-endpoint-hover-color, var(--yt-endpoint-color));text-decoration: none;}";
  400. GM_addStyle(userStyles0);
  401. GM_addStyle(userStyles1);
  402. window.setTimeout(killSPF, 500);
  403.  
  404. console.log("DONE");
  405. }
  406.  
  407. function killSPF() {
  408. return;
  409. if (DISABLE_SPF) {
  410. console.log('trying to kill SPF');
  411. $('a.spf-link, a.yt-simple-endpoint').each(function () {
  412. var $link = $(this);
  413. //console.log('$link', $link);
  414. $link.removeClass('spf-link');
  415. $link.removeClass('yt-simple-endpoint');
  416. $link.addClass('old-endpoint');
  417. $link.off();
  418. });
  419. }
  420. //AND_BUTTS && $('#info-contents h1.title').text($('#info-contents h1.title').text() + " and Butts");
  421. }
  422.  
  423. //GM_registerMenuCommand("Toggle SPF", function toggleSpf() {
  424. // DISABLE_SPF = !DISABLE_SPF;
  425. // console.log("Diable SPF: " + DISABLE_SPF);
  426. // GM_setValue('ytDisableSPF', DISABLE_SPF);
  427. //}, 's');
  428. //GM_registerMenuCommand("and Butts", function toggleButts() {
  429. // AND_BUTTS = !AND_BUTTS;
  430. // console.log("Butts: " + AND_BUTTS);
  431. // GM_setValue('ytAndButts', AND_BUTTS);
  432. //}, 'b');
  433.  
  434. //<div>Icons made by <a href="http://www.flaticon.com/authors/dave-gandy" title="Dave Gandy">Dave Gandy</a> from <a href="http://www.flaticon.com" title="Flaticon">www.flaticon.com</a> is licensed by <a href="http://creativecommons.org/licenses/by/3.0/" title="Creative Commons BY 3.0" target="_blank">CC 3.0 BY</a></div>
  435. // http://www.flaticon.com/free-icon/add-square-button_25300
  436. //data:image/svg+xml;utf8;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iaXNvLTg4NTktMSI/Pgo8IS0tIEdlbmVyYXRvcjogQWRvYmUgSWxsdXN0cmF0b3IgMTYuMC4wLCBTVkcgRXhwb3J0IFBsdWctSW4gLiBTVkcgVmVyc2lvbjogNi4wMCBCdWlsZCAwKSAgLS0+CjwhRE9DVFlQRSBzdmcgUFVCTElDICItLy9XM0MvL0RURCBTVkcgMS4xLy9FTiIgImh0dHA6Ly93d3cudzMub3JnL0dyYXBoaWNzL1NWRy8xLjEvRFREL3N2ZzExLmR0ZCI+CjxzdmcgeG1sbnM9Imh0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnIiB4bWxuczp4bGluaz0iaHR0cDovL3d3dy53My5vcmcvMTk5OS94bGluayIgdmVyc2lvbj0iMS4xIiBpZD0iQ2FwYV8xIiB4PSIwcHgiIHk9IjBweCIgd2lkdGg9IjE2cHgiIGhlaWdodD0iMTZweCIgdmlld0JveD0iMCAwIDMyOC45MTEgMzI4LjkxMSIgc3R5bGU9ImVuYWJsZS1iYWNrZ3JvdW5kOm5ldyAwIDAgMzI4LjkxMSAzMjguOTExOyIgeG1sOnNwYWNlPSJwcmVzZXJ2ZSI+CjxnPgoJPGc+CgkJPHBhdGggZD0iTTMxMC4xOTksMTguNzFDMjk3LjczNSw2LjI0MiwyODIuNjUsMC4wMDcsMjY0Ljk1MSwwLjAwN0g2My45NTRjLTE3LjcwMywwLTMyLjc5LDYuMjM1LTQ1LjI1MywxOC43MDQgICAgQzYuMjM1LDMxLjE3NywwLDQ2LjI2MSwwLDYzLjk2djIwMC45OTFjMCwxNy41MTUsNi4yMzIsMzIuNTUyLDE4LjcwMSw0NS4xMWMxMi40NjcsMTIuNTY2LDI3LjU1MywxOC44NDMsNDUuMjUzLDE4Ljg0M2gyMDEuMDA0ICAgIGMxNy42OTksMCwzMi43NzctNi4yNzYsNDUuMjQ4LTE4Ljg0M2MxMi40Ny0xMi41NTksMTguNzA1LTI3LjU5NiwxOC43MDUtNDUuMTFWNjMuOTYgICAgQzMyOC45MTEsNDYuMjYxLDMyMi42NjYsMzEuMTc3LDMxMC4xOTksMTguNzF6IE0yOTIuMzYyLDI2NC45NmMwLDcuNjE0LTIuNjczLDE0LjA4OS04LjAwMSwxOS40MTQgICAgYy01LjMyNCw1LjMzMi0xMS43OTksNy45OTQtMTkuNDEsNy45OTRINjMuOTU0Yy03LjYxNCwwLTE0LjA4Mi0yLjY2Mi0xOS40MTQtNy45OTRjLTUuMzMtNS4zMjUtNy45OTItMTEuOC03Ljk5Mi0xOS40MTRWNjMuOTY1ICAgIGMwLTcuNjEzLDIuNjYyLTE0LjA4Niw3Ljk5Mi0xOS40MTRjNS4zMjctNS4zMjcsMTEuOC03Ljk5NCwxOS40MTQtNy45OTRoMjAxLjAwNGM3LjYxLDAsMTQuMDg2LDIuNjYzLDE5LjQxLDcuOTk0ICAgIGM1LjMyNSw1LjMyOCw3Ljk5NCwxMS44MDEsNy45OTQsMTkuNDE0VjI2NC45NnoiIGZpbGw9IiMwMDAwMDAiLz4KCQk8cGF0aCBkPSJNMjQ2LjY4MywxNDYuMTg5SDE4Mi43M1Y4Mi4yMzZjMC0yLjY2Ny0wLjg1NS00Ljg1NC0yLjU3My02LjU2N2MtMS43MDQtMS43MTQtMy44OTUtMi41NjgtNi41NjQtMi41NjhoLTE4LjI3MSAgICBjLTIuNjY3LDAtNC44NTQsMC44NTQtNi41NjcsMi41NjhjLTEuNzE0LDEuNzEzLTIuNTY4LDMuOTAzLTIuNTY4LDYuNTY3djYzLjk1NEg4Mi4yMzNjLTIuNjY0LDAtNC44NTcsMC44NTUtNi41NjcsMi41NjggICAgYy0xLjcxMSwxLjcxMy0yLjU2OCwzLjkwMy0yLjU2OCw2LjU2N3YxOC4yNzFjMCwyLjY2NiwwLjg1NCw0Ljg1NSwyLjU2OCw2LjU2M2MxLjcxMiwxLjcwOCwzLjkwMywyLjU3LDYuNTY3LDIuNTdoNjMuOTU0djYzLjk1MyAgICBjMCwyLjY2NiwwLjg1NCw0Ljg1NSwyLjU2OCw2LjU2M2MxLjcxMywxLjcxMSwzLjkwMywyLjU2Niw2LjU2NywyLjU2NmgxOC4yNzFjMi42NywwLDQuODYtMC44NTUsNi41NjQtMi41NjYgICAgYzEuNzE4LTEuNzA4LDIuNTczLTMuODk3LDIuNTczLTYuNTYzVjE4Mi43M2g2My45NTNjMi42NjIsMCw0Ljg1My0wLjg2Miw2LjU2My0yLjU3YzEuNzEyLTEuNzA4LDIuNTYzLTMuODk3LDIuNTYzLTYuNTYzdi0xOC4yNzEgICAgYzAtMi42NjQtMC44NTItNC44NTctMi41NjMtNi41NjdDMjUxLjUzNiwxNDcuMDQ4LDI0OS4zNDUsMTQ2LjE4OSwyNDYuNjgzLDE0Ni4xODl6IiBmaWxsPSIjMDAwMDAwIi8+Cgk8L2c+CjwvZz4KPGc+CjwvZz4KPGc+CjwvZz4KPGc+CjwvZz4KPGc+CjwvZz4KPGc+CjwvZz4KPGc+CjwvZz4KPGc+CjwvZz4KPGc+CjwvZz4KPGc+CjwvZz4KPGc+CjwvZz4KPGc+CjwvZz4KPGc+CjwvZz4KPGc+CjwvZz4KPGc+CjwvZz4KPGc+CjwvZz4KPC9zdmc+Cg==
  437. // http://www.flaticon.com/free-icon/minus-button_25434
  438. //data:image/svg+xml;utf8;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iaXNvLTg4NTktMSI/Pgo8IS0tIEdlbmVyYXRvcjogQWRvYmUgSWxsdXN0cmF0b3IgMTYuMC4wLCBTVkcgRXhwb3J0IFBsdWctSW4gLiBTVkcgVmVyc2lvbjogNi4wMCBCdWlsZCAwKSAgLS0+CjwhRE9DVFlQRSBzdmcgUFVCTElDICItLy9XM0MvL0RURCBTVkcgMS4xLy9FTiIgImh0dHA6Ly93d3cudzMub3JnL0dyYXBoaWNzL1NWRy8xLjEvRFREL3N2ZzExLmR0ZCI+CjxzdmcgeG1sbnM9Imh0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnIiB4bWxuczp4bGluaz0iaHR0cDovL3d3dy53My5vcmcvMTk5OS94bGluayIgdmVyc2lvbj0iMS4xIiBpZD0iQ2FwYV8xIiB4PSIwcHgiIHk9IjBweCIgd2lkdGg9IjE2cHgiIGhlaWdodD0iMTZweCIgdmlld0JveD0iMCAwIDQwMS45OTggNDAxLjk5OCIgc3R5bGU9ImVuYWJsZS1iYWNrZ3JvdW5kOm5ldyAwIDAgNDAxLjk5OCA0MDEuOTk4OyIgeG1sOnNwYWNlPSJwcmVzZXJ2ZSI+CjxnPgoJPGc+CgkJPHBhdGggZD0iTTM3Ny44NywyNC4xMjZDMzYxLjc4Niw4LjA0MiwzNDIuNDE3LDAsMzE5Ljc2OSwwSDgyLjIyN0M1OS41NzksMCw0MC4yMTEsOC4wNDIsMjQuMTI1LDI0LjEyNiAgICBDOC4wNDQsNDAuMjEyLDAuMDAyLDU5LjU3NiwwLjAwMiw4Mi4yMjh2MjM3LjU0M2MwLDIyLjY0Nyw4LjA0Miw0Mi4wMTQsMjQuMTIzLDU4LjEwMWMxNi4wODYsMTYuMDg1LDM1LjQ1NCwyNC4xMjcsNTguMTAyLDI0LjEyNyAgICBoMjM3LjU0MmMyMi42NDgsMCw0Mi4wMTEtOC4wNDIsNTguMTAyLTI0LjEyN2MxNi4wODUtMTYuMDg3LDI0LjEyNi0zNS40NTMsMjQuMTI2LTU4LjEwMVY4Mi4yMjggICAgQzQwMS45OTMsNTkuNTgsMzkzLjk1MSw0MC4yMTIsMzc3Ljg3LDI0LjEyNnogTTM2NS40NDgsMzE5Ljc3MWMwLDEyLjU2NS00LjQ3LDIzLjMxNC0xMy40MTUsMzIuMjY0ICAgIGMtOC45NDUsOC45NDUtMTkuNjk4LDEzLjQxOC0zMi4yNjUsMTMuNDE4SDgyLjIyN2MtMTIuNTYzLDAtMjMuMzE3LTQuNDczLTMyLjI2NC0xMy40MTggICAgYy04Ljk0NS04Ljk0OS0xMy40MTgtMTkuNjk4LTEzLjQxOC0zMi4yNjRWODIuMjMxYzAtMTIuNTYzLDQuNDczLTIzLjMxNywxMy40MTgtMzIuMjY1QzU4LjkxLDQxLjAyMSw2OS42NjQsMzYuNTUsODIuMjI3LDM2LjU1ICAgIGgyMzcuNTQyYzEyLjU2NiwwLDIzLjMxOSw0LjQ3MSwzMi4yNjUsMTMuNDE3YzguOTQ1LDguOTQ3LDEzLjQxNSwxOS43MDIsMTMuNDE1LDMyLjI2NVYzMTkuNzcxTDM2NS40NDgsMzE5Ljc3MXoiIGZpbGw9IiMwMDAwMDAiLz4KCQk8cGF0aCBkPSJNMzE5Ljc2OSwxODIuNzMxSDgyLjIyN2MtMi42NjMsMC00Ljg1MywwLjg1NS02LjU2NywyLjU2NWMtMS43MDksMS43MTMtMi41NjgsMy45MDMtMi41NjgsNi41Njd2MTguMjcxICAgIGMwLDIuNjY5LDAuODU2LDQuODU5LDIuNTY4LDYuNTdjMS43MTUsMS43MDQsMy45MDUsMi41Niw2LjU2NywyLjU2aDIzNy41NDJjMi42NjMsMCw0Ljg1My0wLjg1NSw2LjU3MS0yLjU2ICAgIGMxLjcxMS0xLjcxMSwyLjU2Ni0zLjkwMSwyLjU2Ni02LjU3di0xOC4yNzFjMC0yLjY2NC0wLjg1NS00Ljg1NC0yLjU2Ni02LjU2N0MzMjQuNjE4LDE4My41ODcsMzIyLjQyOCwxODIuNzMxLDMxOS43NjksMTgyLjczMXoiIGZpbGw9IiMwMDAwMDAiLz4KCTwvZz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8L3N2Zz4K
  439.  
  440. //<div>Icons made by <a href="http://www.flaticon.com/authors/catalin-fertu" title="Catalin Fertu">Catalin Fertu</a> from <a href="http://www.flaticon.com" title="Flaticon">www.flaticon.com</a> is licensed by <a href="http://creativecommons.org/licenses/by/3.0/" title="Creative Commons BY 3.0" target="_blank">CC 3.0 BY</a></div>
  441. // http://www.flaticon.com/free-icon/minus-square-outlined-button_54373
  442. //data:image/svg+xml;utf8;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iaXNvLTg4NTktMSI/Pgo8IS0tIEdlbmVyYXRvcjogQWRvYmUgSWxsdXN0cmF0b3IgMTYuMC4wLCBTVkcgRXhwb3J0IFBsdWctSW4gLiBTVkcgVmVyc2lvbjogNi4wMCBCdWlsZCAwKSAgLS0+CjwhRE9DVFlQRSBzdmcgUFVCTElDICItLy9XM0MvL0RURCBTVkcgMS4xLy9FTiIgImh0dHA6Ly93d3cudzMub3JnL0dyYXBoaWNzL1NWRy8xLjEvRFREL3N2ZzExLmR0ZCI+CjxzdmcgeG1sbnM9Imh0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnIiB4bWxuczp4bGluaz0iaHR0cDovL3d3dy53My5vcmcvMTk5OS94bGluayIgdmVyc2lvbj0iMS4xIiBpZD0iQ2FwYV8xIiB4PSIwcHgiIHk9IjBweCIgd2lkdGg9IjE2cHgiIGhlaWdodD0iMTZweCIgdmlld0JveD0iMCAwIDYxMiA2MTIiIHN0eWxlPSJlbmFibGUtYmFja2dyb3VuZDpuZXcgMCAwIDYxMiA2MTI7IiB4bWw6c3BhY2U9InByZXNlcnZlIj4KPGc+Cgk8ZyBpZD0iX3gzMl9fM18iPgoJCTxnPgoJCQk8cGF0aCBkPSJNNDIwLjc1LDI4Ni44NzVoLTIyOS41Yy0xMC41NTcsMC0xOS4xMjUsOC41NjgtMTkuMTI1LDE5LjEyNWMwLDEwLjU1Nyw4LjU2OCwxOS4xMjUsMTkuMTI1LDE5LjEyNWgyMjkuNSAgICAgYzEwLjU1NywwLDE5LjEyNS04LjU2OCwxOS4xMjUtMTkuMTI1QzQzOS44NzUsMjk1LjQ0Myw0MzEuMzA3LDI4Ni44NzUsNDIwLjc1LDI4Ni44NzV6IE01MzUuNSwwaC00NTlDMzQuMjUzLDAsMCwzNC4yNTMsMCw3Ni41ICAgICB2NDU5QzAsNTc3Ljc0NywzNC4yNTMsNjEyLDc2LjUsNjEyaDQ1OWM0Mi4yNDcsMCw3Ni41LTM0LjI1Myw3Ni41LTc2LjV2LTQ1OUM2MTIsMzQuMjUzLDU3Ny43NDcsMCw1MzUuNSwweiBNNTczLjc1LDUzNS41ICAgICBjMCwyMS4xMzMtMTcuMTE3LDM4LjI1LTM4LjI1LDM4LjI1aC00NTljLTIxLjEzMywwLTM4LjI1LTE3LjExNy0zOC4yNS0zOC4yNXYtNDU5YzAtMjEuMTMzLDE3LjExNy0zOC4yNSwzOC4yNS0zOC4yNWg0NTkgICAgIGMyMS4xMzMsMCwzOC4yNSwxNy4xMzYsMzguMjUsMzguMjVWNTM1LjV6IiBmaWxsPSIjMDAwMDAwIi8+CgkJPC9nPgoJPC9nPgo8L2c+CjxnPgo8L2c+CjxnPgo8L2c+CjxnPgo8L2c+CjxnPgo8L2c+CjxnPgo8L2c+CjxnPgo8L2c+CjxnPgo8L2c+CjxnPgo8L2c+CjxnPgo8L2c+CjxnPgo8L2c+CjxnPgo8L2c+CjxnPgo8L2c+CjxnPgo8L2c+CjxnPgo8L2c+CjxnPgo8L2c+Cjwvc3ZnPgo=
  443. //
  444. // http://www.flaticon.com/free-icon/add-square-outlined-interface-button_54731
  445. //data:image/svg+xml;utf8;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iaXNvLTg4NTktMSI/Pgo8IS0tIEdlbmVyYXRvcjogQWRvYmUgSWxsdXN0cmF0b3IgMTYuMC4wLCBTVkcgRXhwb3J0IFBsdWctSW4gLiBTVkcgVmVyc2lvbjogNi4wMCBCdWlsZCAwKSAgLS0+CjwhRE9DVFlQRSBzdmcgUFVCTElDICItLy9XM0MvL0RURCBTVkcgMS4xLy9FTiIgImh0dHA6Ly93d3cudzMub3JnL0dyYXBoaWNzL1NWRy8xLjEvRFREL3N2ZzExLmR0ZCI+CjxzdmcgeG1sbnM9Imh0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnIiB4bWxuczp4bGluaz0iaHR0cDovL3d3dy53My5vcmcvMTk5OS94bGluayIgdmVyc2lvbj0iMS4xIiBpZD0iQ2FwYV8xIiB4PSIwcHgiIHk9IjBweCIgd2lkdGg9IjE2cHgiIGhlaWdodD0iMTZweCIgdmlld0JveD0iMCAwIDYxMiA2MTIiIHN0eWxlPSJlbmFibGUtYmFja2dyb3VuZDpuZXcgMCAwIDYxMiA2MTI7IiB4bWw6c3BhY2U9InByZXNlcnZlIj4KPGc+Cgk8ZyBpZD0iX3gzMV9fMjZfIj4KCQk8Zz4KCQkJPHBhdGggZD0iTTQyMC43NSwyODYuODc1aC05NS42MjVWMTkxLjI1YzAtMTAuNTU3LTguNTY4LTE5LjEyNS0xOS4xMjUtMTkuMTI1Yy0xMC41NTcsMC0xOS4xMjUsOC41NjgtMTkuMTI1LDE5LjEyNXY5NS42MjUgICAgIEgxOTEuMjVjLTEwLjU1NywwLTE5LjEyNSw4LjU2OC0xOS4xMjUsMTkuMTI1YzAsMTAuNTU3LDguNTY4LDE5LjEyNSwxOS4xMjUsMTkuMTI1aDk1LjYyNXY5NS42MjUgICAgIGMwLDEwLjU1Nyw4LjU2OCwxOS4xMjUsMTkuMTI1LDE5LjEyNWMxMC41NTcsMCwxOS4xMjUtOC41NjgsMTkuMTI1LTE5LjEyNXYtOTUuNjI1aDk1LjYyNWMxMC41NTcsMCwxOS4xMjUtOC41NjgsMTkuMTI1LTE5LjEyNSAgICAgQzQzOS44NzUsMjk1LjQ0Myw0MzEuMzA3LDI4Ni44NzUsNDIwLjc1LDI4Ni44NzV6IE01MzUuNSwwaC00NTlDMzQuMjUzLDAsMCwzNC4yNTMsMCw3Ni41djQ1OUMwLDU3Ny43NDcsMzQuMjUzLDYxMiw3Ni41LDYxMiAgICAgaDQ1OWM0Mi4yNDcsMCw3Ni41LTM0LjI1Myw3Ni41LTc2LjV2LTQ1OUM2MTIsMzQuMjUzLDU3Ny43NDcsMCw1MzUuNSwweiBNNTczLjc1LDUzNS41YzAsMjEuMTMzLTE3LjEzNiwzOC4yNS0zOC4yNSwzOC4yNWgtNDU5ICAgICBjLTIxLjEzMywwLTM4LjI1LTE3LjExNy0zOC4yNS0zOC4yNXYtNDU5YzAtMjEuMTMzLDE3LjExNy0zOC4yNSwzOC4yNS0zOC4yNWg0NTljMjEuMTE0LDAsMzguMjUsMTcuMTM2LDM4LjI1LDM4LjI1VjUzNS41eiIgZmlsbD0iIzAwMDAwMCIvPgoJCTwvZz4KCTwvZz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8Zz4KPC9nPgo8L3N2Zz4K
  446. //data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAMAAAAoLQ9TAAAAGFBMVEX///8AAAAAAAAAAAAAAAAAAAAAAAAAAABcYkG9AAAAB3RSTlMAIDxIgMDJoM584QAAADlJREFUGNOFjzkCACAIw4oU+P+PXREFM2bogeWR8AVXJNQROIhWiBRBjkJIM1J68c941Q5Lr3P1/gYInwHxRyFQjQAAAABJRU5ErkJggg==
  447.  
  448. //plus data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAMAAAAoLQ9TAAAAD1BMVEXAwMABAQEBAQEAAAAAAADcECccAAAABHRSTlMAPEjK3fOUzQAAADFJREFUGNNjYGRmQQLMjAzMTAxIgImZgYUBBbDgFGBhIU0AZiluATIMxe9SDM+hex8AtbABVeH7XA4AAAAASUVORK5CYII=
  449. //minus data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAMAAAAoLQ9TAAAAD1BMVEXAwMABAQEBAQEAAAAAAADcECccAAAABHRSTlMAPEjJRPrFdwAAACtJREFUGNNjYGRmQQLMjAzMTAxIgImZgYUBBbBQTQBmKW4BqluL4Tl07wMAnDABJd8eqrEAAAAASUVORK5CYII=
  450.  
  451.  
  452. //// CODE STOLEN FROM YoutubeCenter AND YouTubePlaylistAutoplayDisable TO DEAL WITH STUPID SPF.
  453. // Runs the function when the page is loading and has finished loading
  454. //window.addEventListener('readystatechange', findStuff);
  455. // Runs the function when the page changes via SPF method: https://github.com/youtube/spfjs/
  456. window.addEventListener('spfdone', findStuff);
  457. //window.addEventListener('yt-navigate', findStuff);
  458. //window.addEventListener('yt-page-type-changed', findStuff);
  459. window.addEventListener('yt-navigate-finish', findStuff);
  460. //window.addEventListener('yt-navigate-finish', killSPF);
  461. //findStuff();
  462. //killSPF();
  463. window.setTimeout(findStuff, 200);
  464.  
  465.  
  466. //VIMEO info
  467. // video at https://vimeo.com/160190376
  468. // source contains URL https://player.vimeo.com/video/160190376/config?byline=0&collections=0&context=Vimeo%5CController%5CClipController.main&default_to_hd=1&outro=nothing&portrait=0&share=1&title=0&watch_trailer=0&s=f32dd867feea75e8b71005f7793d4ba285958b4c_1472286552
  469. // which is JSON with json.request.thumb_preview containing url, frame_height, frame_width, height, width, frames, columns
  470. // json.request.thumb_preview.url = https://i.vimeocdn.com/1472189353-0x9b6eb1264f85669b1a37525cb5b4d18b9edc2b39/sprite/160190376/120?q=48&h=126&ver=1&w=224&n=4