Guest User

derp

a guest
Aug 8th, 2019
47
0
Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
text 39.53 KB | None | 0 0
  1. // ==UserScript==
  2. // @name YouTube Link Title
  3. // @description Adds video titles, shows previews and embeds on click. Also supported: Vimeo, LiveLeak, Dailymotion, vidme, WorldStarHipHop, Vine, Coub, Streamable
  4. // @namespace http://w9p.co/userscripts/
  5. // @version 2017.2.28
  6. // @author kuehlschrank
  7. // @homepage http://w9p.co/userscripts/ytlt
  8. // @icon https://w9p.co/userscripts/ytlt/icon.png
  9. // @include http*
  10. // @exclude http*//*.google.*/*
  11. // @exclude *//*.googleapis.com/*
  12. // @exclude *//vimeo.com/*
  13. // @exclude *//*.vimeo.com/*
  14. // @exclude *//*.worldstarhiphop.com/*
  15. // @exclude *//vine.co/*
  16. // @exclude *//*.dailymotion.com/*
  17. // @exclude *//coub.com/*
  18. // @exclude *//vid.me/*
  19. // @exclude *//*.vid.me/*
  20. // @exclude *://disqus.com/embed/*worldstarhiphop*
  21. // @exclude *//streamable.com/*
  22. // @exclude *//*.streamable.com/*
  23. // @grant GM_getValue
  24. // @grant GM_setValue
  25. // @grant GM_xmlhttpRequest
  26. // @grant GM_addStyle
  27. // @grant GM_registerMenuCommand
  28. // @connect-src googleapis.com
  29. // @connect-src vimeo.com
  30. // @connect-src vine.co
  31. // @connect-src liveleak.com
  32. // @connect-src worldstarhiphop.com
  33. // @connect-src dailymotion.com
  34. // @connect-src coub.com
  35. // @connect-src vid.me
  36. // @connect-src streamable.com
  37. // ==/UserScript==
  38.  
  39. 'use strict';
  40.  
  41. function onLoad() {
  42. if(location.hostname.indexOf('liveleak.com') > -1) {
  43. if(location.pathname == '/ll_embed' && location.hash == '#play') document.body.querySelector('video').play();
  44. return;
  45. } else if(location.hostname.indexOf('youtube.com') > -1 || location.hostname.indexOf('youtube-nocookie.com') > -1) {
  46. if(location.pathname.indexOf('/embed/') == 0 && window.top != window.self) {
  47. document.body.querySelector('.ytp-chrome-controls a[href*="www.youtube.com"]').addEventListener('click', function(e) {
  48. window.parent.postMessage({YTLT:'close'}, '*');
  49. });
  50. }
  51. return;
  52. }
  53. cfg.load();
  54. var q = 'a[href*="//t.co/"]';
  55. for(var sid in sites) {
  56. q += sites[sid].patterns.reduce(function(prev, cur) { return prev + ',a[href*="' + cur +'"]'; }, '');
  57. }
  58. links.query = q;
  59. links.process(d.body);
  60. var M = window.MutationObserver;
  61. if(M) {
  62. var opts = location.hostname == 'www.facebook.com' ? {childList:true, subtree:true, attributes:true, attributeFilter:['href']} : {childList:true, subtree:true};
  63. new M(onMutations).observe(d.body, opts);
  64. } else {
  65. d.body.addEventListener('DOMNodeInserted', onNodeInserted);
  66. }
  67. window.addEventListener('message', onMessage);
  68. GM_registerMenuCommand('Set up YouTube Link Title', setup.show);
  69. }
  70.  
  71. function onNodeInserted(e) {
  72. var t = e.target;
  73. if(t.nodeType == 1) window.setTimeout(links.process, 100, t);
  74. }
  75.  
  76. function onMutations(muts) {
  77. for(var i = muts.length, m; i-- && (m = muts[i]);) {
  78. if(m.type == 'attributes') {
  79. window.setTimeout(links.process, 100, m.target);
  80. } else {
  81. for(var j = m.addedNodes.length, node; j-- && (node = m.addedNodes[j]);) {
  82. if(node.nodeType == 1) window.setTimeout(links.process, 100, node);
  83. }
  84. }
  85. }
  86. }
  87.  
  88. function onMessage(e) {
  89. if(typeof e.data != 'object' || !e.data.YTLT) return;
  90. var li = e.data.YTLT;
  91. if(li == 'close') return popup.close();
  92. if(window.frames.length > 1) {
  93. li.self = true;
  94. e.source.postMessage({YTLT:li}, '*');
  95. } else {
  96. popup.show(li);
  97. }
  98. }
  99.  
  100. function ce(n, props) {
  101. var n = d.createElement(n);
  102. if(props) {
  103. forEach(props, function(p) {
  104. if(p == 'click' || p == 'mousedown' || p == 'mousedown') {
  105. n.addEventListener(p, props[p]);
  106. } else {
  107. n[p] = props[p];
  108. }
  109. });
  110. }
  111. return n;
  112. }
  113.  
  114. function forEach(o, f) {
  115. var props = Object.keys(o);
  116. for(var i = props.length, p; i-- && (p = props[i]);) {
  117. if(o.hasOwnProperty(p)) f(p);
  118. }
  119. }
  120.  
  121. function rm(id) {
  122. var node = $(id);
  123. if(node) node.parentNode.removeChild(node);
  124. }
  125.  
  126. var d = document, $ = function(id) { return d.getElementById(id); }, log = function(s) { console.log(s); };
  127.  
  128. var setup = {
  129. id: 'YTLT-setup',
  130. html: function() {
  131. var h = '<div>YouTube Link Title</div><ul>';
  132. this.forEach(function(n) {
  133. var s = cfg.settings[n];
  134. if(typeof s.default == 'boolean') {
  135. h += '<li><input type="checkbox" name="'+ n +'"> ' + s.title + '</li>'
  136. } else if(s.options) {
  137. h += '<li>' + s.title + ': <select name="' + n + '">';
  138. forEach(s.options, function(on) {
  139. h += '<option value="' + on + '">' + s.options[on] +'</option>';
  140. });
  141. h += '</select></li>';
  142. } else if(typeof s.default == 'string') {
  143. h += '<li>' + s.title + ': <input type="text" name="'+ n +'" maxlength="' + s.maxlength + '" style="width:' + s.maxlength*12 +'px"></li>';
  144. }
  145. });
  146. return h += '</ul><div><button name="save">Save settings</button></div></div>';
  147. },
  148. q: function(n) {
  149. return d.querySelector('#' + this.id + ' *[name="' + n + '"]');
  150. },
  151. get: function(n) {
  152. var s = cfg.settings[n];
  153. if(typeof s.default == 'boolean') return this.q(n).checked;
  154. if(s.options) { var sel = this.q(n); return sel.options[sel.selectedIndex].value; }
  155. if(typeof s.default == 'string') return this.q(n).value;
  156. },
  157. set: function(n, val) {
  158. var s = cfg.settings[n];
  159. if(typeof s.default == 'boolean') {
  160. this.q(n).checked = val;
  161. } else if(s.options) {
  162. var sel = this.q(n);
  163. for(var i = sel.options.length; i--;) { if(sel.options[i].value == val) sel.selectedIndex = i; }
  164. } else if(typeof s.default == 'string') {
  165. this.q(n).value = val.trim();
  166. }
  167. },
  168. forEach: function(f) {
  169. forEach(cfg.settings, function(n) { if(cfg.settings[n].title) f(n); });
  170. },
  171. show: function() {
  172. rm(setup.id);
  173. GM_addStyle('\
  174. #'+setup.id+' { position:fixed;z-index:2147483647;top:40px;right:40px;padding:20px 30px;background-color:white;width:auto;border:1px solid black }\
  175. #'+setup.id+' * { color:black;text-align:left;line-height:normal;font-size:12px }\
  176. #'+setup.id+' div { text-align:center;font-weight:bold;font-size:14px }\
  177. #'+setup.id+' ul { margin:15px 0 15px 0;padding:0;list-style:none }\
  178. #'+setup.id+' li { margin:0;padding:3px 0 3px 0;vertical-align:middle }'
  179. );
  180. d.body.appendChild(ce('div', {id:setup.id,innerHTML:setup.html()}));
  181. setup.q('save').addEventListener('click', function() {
  182. if(setup.get('country') != cfg.country) cache.clear('yt');
  183. setup.forEach(function(n) { cfg[n] = setup.get(n); });
  184. cfg.x = cfg.y = false;
  185. cfg.save();
  186. this.disabled = true;
  187. this.innerHTML = 'Reloading...';
  188. location.reload();
  189. });
  190. setup.forEach(function(n) {
  191. setup.set(n, cfg[n]);
  192. setup.q(n).addEventListener('change', setup.update);
  193. });
  194. setup.update();
  195. },
  196. update: function() {
  197. setup.forEach(function(n) {
  198. var s = cfg.settings[n];
  199. if(!s.depends) return;
  200. setup.q(n).parentNode.style.display = (Object.keys(s.depends).every(function(dn) { return s.depends[dn].test(setup.get(dn)); })) ? '' : 'none';
  201. });
  202. }
  203. }
  204.  
  205. var links = {
  206. process: function(node) {
  207. var i = 0, list = node.tagName.toUpperCase() == 'A' ? [node] : node.querySelectorAll(links.query), isFB = location.hostname == 'www.facebook.com';
  208. if(list.length < 1) return;
  209. function processChunk() {
  210. var a, li, vi, num = 0, ae = d.activeElement;
  211. while(a = list[i++]) {
  212. if(!(li = links.parseInfo(a)) || ae && ae != d.body && ae.contains(a) || isFB && (a.parentNode.outerHTML.indexOf('shareLink') > -1 || a.parentNode.outerHTML.indexOf('ScaledImageContainer') > -1)) continue;
  213. if((vi = cache.get(li.sid, li.vid)) || cfg.urls_only && !li.url) {
  214. links.decorate(a, li, vi);
  215. } else {
  216. (function(a, li) {
  217. net.info(li, function(vi) { links.decorate(a, li, vi); });
  218. })(a, li);
  219. }
  220. if(++num == 15) return window.setTimeout(processChunk, 100);
  221. }
  222. list = null;
  223. }
  224. node = null;
  225. processChunk();
  226. },
  227. parseInfo: function(a) {
  228. var url = a.getAttribute('data-expanded-url') || a.getAttribute('data-full-url') || a.href;
  229. try { url = decodeURIComponent(url.replace(/^https?:\/\/(?:www\.facebook\.com\/.+|sys\.4chan\.org\/derefer\?url=|out\.reddit\.com.+)(http.+)/, '$1')).replace(/&token=[^&]+/, ''); } catch(ex) {};
  230. var test = function(pattern) { return this.indexOf(pattern.toLowerCase()) != -1; };
  231. for(var sid in sites) {
  232. var s = sites[sid], info;
  233. if(!sites[sid].patterns.some(test, url.toLowerCase())) continue;
  234. if(info = sites[sid].parse(url)) return {sid:sid,vid:info.vid,t:info.t,url:sites[sid].patterns.some(test, a.textContent.toLowerCase()),oUrl:url};
  235. }
  236. return null;
  237. },
  238. decorate: function(a, li, vi) {
  239. if(['wh'].indexOf(li.sid) > -1 && !vi.status && location.protocol == 'https:') vi.status = 5;
  240. if(a.className.indexOf('YTLT-') > -1) {
  241. a.removeEventListener('click', this.onClick, false);
  242. a.removeEventListener('mouseover', this.onMouseOver, false);
  243. } else {
  244. if(!this.styleAdded) {
  245. var css = '\
  246. #YTLT-preview { position:fixed;z-index:2147483645;width:320px;padding:0;border:1px solid black;background-color:black; }\
  247. #YTLT-preview img { max-height:240px;width:inherit;vertical-align:bottom; }\
  248. #YTLT-preview div { padding:4px 2px;width:316px;text-align:center;font-family:sans-serif;font-size:13px;color:white;font-weight:bold; }\
  249. a.YTLT-na { text-decoration: line-through!important; }\
  250. a.YTLT-text { font-weight:bold!important;font-style:italic!important; }\
  251. a.YTLT-icon { padding-left:18px!important; }\
  252. a.YTLT-icon.YTLT-na:hover, a.YTLT-icon.YTLT-ne:hover { background:transparent url() center left no-repeat!important; }';
  253. for(var sid in sites) {
  254. css += 'a.YTLT-icon-' + sid + ' { background:transparent url(' + sites[sid].icon + ') center left no-repeat!important; }';
  255. }
  256. GM_addStyle(css);
  257. this.styleAdded = true;
  258. }
  259. var c = ' YTLT-link';
  260. if(li.url || location.hostname == 'twitter.com') {
  261. c += ' YTLT-text YTLT-icon YTLT-icon-' + li.sid;
  262. if(vi) a.innerHTML = vi.title;
  263. } else {
  264. if(vi && !cfg.previews && vi.title) a.title = vi.title;
  265. if(a.hasChildNodes() && a.innerHTML.indexOf('<img') == -1 && a.textContent.trim()) {
  266. c += ' YTLT-icon YTLT-icon-' + li.sid;
  267. }
  268. }
  269. if(vi.status == 3 || vi.status == 4) {
  270. c += ' YTLT-na';
  271. } else if(vi.status == 5) {
  272. c += ' YTLT-ne';
  273. }
  274. a.className += c;
  275. if(cfg.rewrite && sites[li.sid].url) {
  276. a.href = sites[li.sid].url(li.vid, li.t, li.oUrl);
  277. }
  278. }
  279. if(cfg.embed_mode != 'off') {
  280. a.target = '_blank';
  281. if(!vi.status) {
  282. a.onclick = null;
  283. a.addEventListener('click', this.onClick);
  284. }
  285. }
  286. if(cfg.previews && (vi && vi.preview || sites[li.sid].preview) && !a.querySelector('iframe')) {
  287. var url = embedding.preview(vi, li);
  288. if(['twitter.com', 'github.com'].indexOf(location.hostname) > -1) {
  289. a.title = '';
  290. GM_xmlhttpRequest({
  291. method: 'GET',
  292. url: url,
  293. overrideMimeType: 'text/plain; charset=x-user-defined',
  294. headers: {'Accept':'image/png,image/*;q=0.8,*/*;q=0.5'},
  295. onload: function(res) {
  296. var txt = res.responseText, ui8 = new Uint8Array(txt.length), fr = new FileReader();
  297. for(var i = txt.length; i--;) ui8[i] = txt.charCodeAt(i);
  298. fr.onload = function() { a.setAttribute('data-ytlt-preview', fr.result); };
  299. fr.readAsDataURL(new Blob([ui8.buffer], {type:'image/jpeg'}));
  300. }
  301. });
  302. } else {
  303. a.setAttribute('data-ytlt-preview', url);
  304. new Image().src = url;
  305. }
  306. a.setAttribute('data-ytlt-title', vi.title);
  307. a.addEventListener('mouseover', this.onMouseOver);
  308. }
  309. },
  310. onClick: function(e) {
  311. if(e.ctrlKey || e.altKey || e.shiftKey || e.metaKey || e.button != 0) return;
  312. e.preventDefault();
  313. e.stopImmediatePropagation();
  314. rm('YTLT-preview');
  315. embedding.play(this);
  316. },
  317. onMouseOver: function(e) {
  318. rm('YTLT-preview');
  319. var url = this.getAttribute('data-ytlt-preview'), title = this.getAttribute('data-ytlt-title');
  320. var qm = d.compatMode == 'BackCompat';
  321. var w = qm ? d.body.clientWidth : d.documentElement.clientWidth, h = qm ? d.body.clientHeight : d.documentElement.clientHeight;
  322. var div = ce('div', {id:'YTLT-preview'});
  323. if(url) {
  324. var img = ce('img', {src:url});
  325. div.appendChild(img);
  326. }
  327. if(title && this.textContent.toLowerCase().indexOf(title.toLowerCase()) == -1) div.appendChild(ce('div', {innerHTML:title}));
  328. d.body.appendChild(div);
  329. var r = (this.firstElementChild || this).getBoundingClientRect();
  330. var dw = Math.max(div.offsetWidth, 320), dh = Math.max(div.offsetHeight, 240);
  331. if(h - r.height > 2*dh) {
  332. if(r.top + r.bottom > h) {
  333. div.style.bottom = h - r.top + 15 + 'px';
  334. } else {
  335. div.style.top = r.bottom + 15 + 'px';
  336. }
  337. div.style.left = Math.min(w - dw - 10, Math.max(10, e.pageX - dw/2)) + 'px';
  338. } else {
  339. if(r.right + r.left > w) {
  340. div.style.right = w - r.right + r.width + 15 + 'px';
  341. } else {
  342. div.style.left = r.left + r.width + 15 + 'px';
  343. }
  344. if(r.top+r.bottom > h) {
  345. div.style.bottom = Math.max(20, h - r.top - r.height/2 - dh/2) + 'px';
  346. } else {
  347. div.style.top = Math.max(10, r.top + r.height/2 - dh/2) + 'px';
  348. }
  349. }
  350. d.addEventListener('mouseout', links.onMouseOut);
  351. },
  352. onMouseOut: function(e) {
  353. d.removeEventListener('mouseout', links.onMouseOut, false);
  354. rm('YTLT-preview');
  355. }
  356. };
  357.  
  358. var net = {
  359. info: function(li, f) {
  360. var id = li.sid + li.vid;
  361. if(typeof net.pending[id] != 'object') {
  362. net.pending[id] = [f];
  363. sites[li.sid].request(li.vid, function(vi) {
  364. if(vi) {
  365. cache.set(li.sid, li.vid, vi);
  366. for(var i = net.pending[id].length; i--;) {
  367. window.setTimeout(net.pending[id][i], 0, vi);
  368. }
  369. }
  370. delete net.pending[id];
  371. });
  372. } else {
  373. net.pending[id].push(f);
  374. }
  375. },
  376. pending: {},
  377. json: function(url, f) {
  378. GM_xmlhttpRequest({
  379. method: 'GET',
  380. url: url,
  381. onload: function(req) {
  382. var obj;
  383. try {
  384. obj = JSON.parse(req.responseText);
  385. } catch(ex) {
  386. log('JSON data from ' + url + ' could not be parsed: ' + ex + '\nHTTP: ' + req.status + '\nResponse: ' + req.responseText);
  387. }
  388. window.setTimeout(f, 0, req.status, obj, req.responseText);
  389. },
  390. onerror: function(req) {
  391. log('Request to ' + url + ' failed.');
  392. window.setTimeout(f, 0, req.status, null, req.responseText);
  393. }
  394. });
  395. },
  396. text: function(url, re, f) {
  397. GM_xmlhttpRequest({
  398. method: 'GET',
  399. url: url,
  400. onload: function(req) {
  401. var m = [], txt = req.responseText;
  402. for(var i = 0, len = re.length; i < len; i++) {
  403. m.push(re[i].exec(txt));
  404. }
  405. window.setTimeout(f, 0, req.status, m);
  406. },
  407. onerror: function(req) {
  408. log('Request to ' + url + ' failed.');
  409. window.setTimeout(f, 0, req.status, []);
  410. }
  411. });
  412. }
  413. };
  414.  
  415. var embedding = {
  416. play: function(a) {
  417. var li = links.parseInfo(a);
  418. if(cfg.embed_mode == 'inline') {
  419. var embed = embedding.embed(li, cfg.big);
  420. a.parentNode.replaceChild(embed, a);
  421. window.setTimeout(window.scrollTo, 0, 0, window.scrollY + embed.getBoundingClientRect().top - window.innerHeight/2 + embed.offsetHeight/2);
  422. } else if(cfg.embed_mode == 'window' || location.hostname == 'api.solidopinion.com' && location.search.indexOf('nextbigfuture.com')) {
  423. var embed = embedding.embed(li, cfg.big);
  424. var w = parseInt(embed.style.width), h = parseInt(embed.style.height);
  425. embed.style.width = '100%';
  426. embed.style.height = '100%';
  427. var div = ce('div');
  428. div.appendChild(embed);
  429. var features = 'left='+(window.screen.width/2-w/2)+',top='+(window.screen.height/2-h/2)+',width='+w+',height='+h+',status=no,scrollbars=no,location=no,menubar=no,toolbar=no,personalbar=no,dependent=no';
  430. window.open('data:text/html,<html><title>Video Player</title><body style="padding:0;margin:0;overflow:hidden;background:black">' + encodeURIComponent(div.innerHTML) + '</body></html>', 'YouTube Link Title', features);
  431. div.removeChild(embed);
  432. } else if(cfg.embed_mode == 'player') {
  433. popup.show(li);
  434. }
  435. },
  436. embed: function(li, big) {
  437. var site = sites[li.sid], embed = ce('iframe', {className:'YTLT-embed'});
  438. embed.setAttribute('webkitAllowFullScreen', '');
  439. embed.setAttribute('allowfullscreen', '');
  440. embed.setAttribute('YTLT-sid', li.sid);
  441. embedding.resize(embed, big);
  442. embed.src = site.embed(li.vid, li.t);
  443. embed.style.display = 'block';
  444. return embed;
  445. },
  446. resize: function(embed, big) {
  447. var site = sites[embed.getAttribute('YTLT-sid')];
  448. var w = site.sizes[big?1:0][0];
  449. var h = site.sizes[big?1:0][1];
  450. embed.style.width = w + 'px';
  451. embed.style.height = h + 'px';
  452. },
  453. preview: function(vi, li) {
  454. if(vi && vi.preview) {
  455. return vi.preview;
  456. } else if(sites[li.sid].preview) {
  457. return sites[li.sid].preview(li.vid);
  458. }
  459. }
  460. };
  461.  
  462. var popup = {
  463. show: function(li) {
  464. if(window.top != window.self && !li.self) return window.parent.postMessage({YTLT:li}, '*');
  465. this.close();
  466. if(!this.styleAdded) {
  467. GM_addStyle('\
  468. #YTLT-bg { position:fixed;z-index:2147483647;top:0;right:0;bottom:0;left:0;background-color:black;opacity:0.9; }\
  469. #YTLT-player { display:block;position:fixed;z-index:2147483647;line-height:normal;background-color:black;border:2px solid black;font-size:13px;line-height:16px;padding:0;margin:0;left:auto;top:auto;right:auto;bottom:auto; }\
  470. #YTLT-player-titlebar { display:block;line-height:17px;padding:0;background:#1b1b1b;border-bottom:2px solid black;text-align:'+(cfg.reverse_buttons?'left':'right')+'; }\
  471. #YTLT-player-left { display:block;position:absolute;left:-6px;top:6px;bottom:6px;width:8px;padding:0;margin:0;background:none;cursor:ew-resize;s }\
  472. #YTLT-player-bottom { display:block;position:absolute;left:6px;right:6px;bottom:-6px;height:8px;padding:0;margin:0;background:none;cursor:ns-resize;s }\
  473. #YTLT-player-darken { padding:0px 11px 3px 11px; }\
  474. #YTLT-player-resize { padding:1px 11px 2px 11px; }\
  475. #YTLT-player-close { padding:0px 28px 3px 28px; }\
  476. #YTLT-player .YTLT-embed { border:0;line-height:normal; }\
  477. #YTLT-player .YTLT-player-titlebar-button { vertical-align:top;display:inline-block;margin:0;font-weight:bold;font-size:14px;text-decoration:none;border:0;border-'+(cfg.reverse_buttons?'right':'left')+':2px solid black;color:#757575;text-decoration:none;cursor:pointer;font-family:"segoe ui",verdana,sans-serif; }\
  478. #YTLT-player .YTLT-player-titlebar-button:hover { color:#959595!important; }\
  479. .YTLT-embed { display:block;background-color:black;border:2px solid black;margin:0;padding:0; }\
  480. #YTLT-player.YTLT-player-moving { opacity:0.8;cursor:move;border:2px solid white; }\
  481. #YTLT-player.YTLT-player-resizing { opacity:0.8; }\
  482. #YTLT-player.YTLT-player-moving .YTLT-embed, #YTLT-player.YTLT-player-resizing .YTLT-embed { visibility:hidden; }\
  483. .YTLT-noselect { -moz-user-select:none;-webkit-user-select:none;-o-user-select:none; }'
  484. );
  485. this.styleAdded = true;
  486. }
  487. this.li = li;
  488. var titlebar = ce('div', {id:'YTLT-player-titlebar',mousedown:this.onTitlebarMouseDown});
  489. var buttons = [
  490. ce('a', {id:'YTLT-player-darken',className:'YTLT-player-titlebar-button YTLT-noselect',innerHTML:'&#9788;',click:this.onDarkenClick}),
  491. ce('a', {id:'YTLT-player-resize',className:'YTLT-player-titlebar-button YTLT-noselect',click:this.onResizeClick}),
  492. ce('a', {id:'YTLT-player-close',className:'YTLT-player-titlebar-button YTLT-noselect',innerHTML:'X',click:this.onCloseClick})
  493. ];
  494. if(cfg.reverse_buttons) buttons.reverse();
  495. for(var i = 0; i < buttons.length; i++) {
  496. titlebar.appendChild(buttons[i]);
  497. }
  498. var player = ce('div', {id:'YTLT-player'});
  499. player.appendChild(titlebar);
  500. player.appendChild(ce('div', {id:'YTLT-player-left',mousedown:this.onBorderMouseDown}));
  501. player.appendChild(ce('div', {id:'YTLT-player-bottom',mousedown:this.onBorderMouseDown}));
  502. player.appendChild(embedding.embed(li, cfg.big));
  503. if(cfg.darken) d.body.appendChild(ce('div', {id:'YTLT-bg'}));
  504. d.body.appendChild(player);
  505. this.update();
  506. var w = player.offsetWidth, h = player.offsetHeight;
  507. var x = parseFloat(cfg.x), y = parseFloat(cfg.y);
  508. var qm = d.compatMode == 'BackCompat';
  509. var cw = qm ? d.body.clientWidth : d.documentElement.clientWidth;
  510. var ch = qm ? d.body.clientHeight : d.documentElement.clientHeight;
  511. var mx = cw - x - w/2;
  512. var isPosValid = mx > 0 && mx < cw && y > -5 && y < ch - h/2;
  513. player.style.right = (isPosValid ? x : 80) + 'px';
  514. player.style.top = (isPosValid ? y : ch/2 - h/2) + 'px';
  515. d.addEventListener('keydown', this.onKeyDown);
  516. },
  517. close: function() {
  518. rm('YTLT-player');
  519. rm('YTLT-bg');
  520. d.removeEventListener('keydown', popup.onKeyDown, true);
  521. },
  522. update: function(e) {
  523. $('YTLT-player-resize').innerHTML = cfg.big ? '&#65293;' : '&#65291;';
  524. },
  525. onCloseClick: function(e) {
  526. e.preventDefault();
  527. popup.close();
  528. },
  529. onResizeClick: function(e) {
  530. e.preventDefault();
  531. cfg.big = !cfg.big;
  532. cfg.save();
  533. var embed = d.body.querySelector('#YTLT-player .YTLT-embed');
  534. embedding.resize(embed, cfg.big);
  535. popup.update();
  536. },
  537. onDarkenClick: function(e) {
  538. if($('YTLT-bg')) {
  539. rm('YTLT-bg');
  540. cfg.darken = false;
  541. cfg.save();
  542. } else {
  543. var player = $('YTLT-player');
  544. player.parentNode.insertBefore(ce('div', {id:'YTLT-bg'}), player);
  545. cfg.darken = true;
  546. cfg.save();
  547. }
  548. },
  549. onKeyDown: function(e) {
  550. if(e.keyCode == 27) popup.close();
  551. },
  552. onTitlebarMouseDown: function(e) {
  553. var t = e.target;
  554. if(t.id != 'YTLT-player-titlebar') return;
  555. t.parentNode.className = 'YTLT-player-moving';
  556. var r = t.getBoundingClientRect();
  557. t.dx = e.clientX - r.left;
  558. t.dy = e.clientY - r.top;
  559. d.body.className += ' YTLT-noselect';
  560. d.addEventListener('mousemove', popup.onTitlebarMouseMove);
  561. d.addEventListener('mouseup', popup.onTitlebarMouseUp);
  562. },
  563. onTitlebarMouseMove: function(e) {
  564. var titlebar = $('YTLT-player-titlebar');
  565. titlebar.parentNode.style.right = (d.compatMode == 'BackCompat' ? d.body.clientWidth : d.documentElement.clientWidth) - e.clientX - titlebar.offsetWidth + titlebar.dx - 2 + 'px';
  566. titlebar.parentNode.style.top = e.clientY - titlebar.dy - 2 + 'px';
  567. },
  568. onTitlebarMouseUp: function(e) {
  569. var titlebar = $('YTLT-player-titlebar');
  570. titlebar.parentNode.className = '';
  571. titlebar.dx = null;
  572. titlebar.dy = null;
  573. d.removeEventListener('mousemove', popup.onTitlebarMouseMove, false);
  574. d.removeEventListener('mouseup', popup.onTitlebarMouseUp, false);
  575. d.body.className = d.body.className.replace('YTLT-noselect', '');
  576. cfg.x = parseInt(titlebar.parentNode.style.right);
  577. cfg.y = titlebar.parentNode.style.top;
  578. cfg.save();
  579. },
  580. onBorderMouseDown: function(e) {
  581. var embed = d.body.querySelector('#YTLT-player .YTLT-embed');
  582. embed.parentNode.className = 'YTLT-player-resizing';
  583. var r = embed.getBoundingClientRect();
  584. embed.ratio = r.width/r.height;
  585. embed.anchor = [r.right, r.top];
  586. embed.control = e.target.id;
  587. d.body.className += ' YTLT-noselect';
  588. d.addEventListener('mousemove', popup.onBorderMouseMove);
  589. d.addEventListener('mouseup', popup.onBorderMouseUp);
  590. },
  591. onBorderMouseMove: function(e) {
  592. var embed = d.body.querySelector('#YTLT-player .YTLT-embed'), w, h;
  593. if(embed.control == 'YTLT-player-left') {
  594. var dx = embed.anchor[0] - e.clientX;
  595. w = dx;
  596. h = dx / embed.ratio;
  597. } else {
  598. var dy = e.clientY - embed.anchor[1];
  599. w = embed.ratio * dy;
  600. h = dy;
  601. }
  602. embed.style.width = w + 'px';
  603. embed.style.height = h + 'px';
  604. },
  605. onBorderMouseUp: function(e) {
  606. var embed = d.body.querySelector('#YTLT-player .YTLT-embed');
  607. embed.parentNode.className = '';
  608. embed.x = null;
  609. embed.y = null;
  610. embed.w = null;
  611. embed.h = null;
  612. d.removeEventListener('mousemove', popup.onBorderMouseMove, false);
  613. d.removeEventListener('mouseup', popup.onBorderMouseUp, false);
  614. d.body.className = d.body.className.replace('YTLT-noselect', '');
  615. },
  616. };
  617.  
  618. var sites = {
  619. 'yt':{
  620. patterns:["youtube.com", "youtu.be", "youtube.googleapis.com"],
  621. parse:function(url) { return /^https?:\/\/((www\.|m\.)?youtu(\.be|be\.(googleapis\.)?com).+v[=\/]|#p\/[a-z]\/.+\/|youtu\.be\/)([a-z0-9_-]{8,})(.*[#&\?]t=([0-9hms]+))?/i.exec(url) ? {vid:RegExp.$5, t:RegExp.$7} : null; },
  622. sizes:[[640,360],[853,480]],
  623. embed:function(vid, t) { var m = /((\d+)h)?((\d+)m)?(\d+)?/.exec(t); t = parseInt(m[2] || 0)*3600 + parseInt(m[4] || 0)*60 + parseInt(m[5] || 0); return 'https://www.youtube.com/embed/' + vid + '?autoplay=1&rel=1' + (t ? '&start=' + t : ''); },
  624. url:function(vid, t, oUrl) { return 'https://www.youtube.com/watch?v=' + vid + (/\b(list=[a-z0-9_-]+)/i.exec(oUrl) ? '&' + RegExp.$1 : '') + (t ? '#t=' + t : ''); },
  625. preview:function(vid) { return 'https://img.youtube.com/vi/' + vid + '/0.jpg'; },
  626. icon:'',
  627. request:function(vid, f) {
  628. var part = ['snippet', 'status'], fields = ['id', 'snippet/title', 'status/privacyStatus', 'status/embeddable'], country;
  629. if(cfg.country) {
  630. country = cfg.country.toUpperCase();
  631. part.push('contentDetails');
  632. fields.push('contentDetails/regionRestriction');
  633. }
  634. net.json('https://www.googleapis.com/youtube/v3/videos?id=' + vid + '&part=' + part.join(',') + '&fields=items(' + fields.join(',') + ')&' + String.fromCharCode.apply(String, [107,101,121,61,65,73,122,97,83,121,68,87,120,79,114,52,76,105,52,65,54,72,116,89,120,57,107,55,98,86,98,67,81,54,56,118,112,119,74,74,117,111,99]), function(code, obj, txt) {
  635. if(code != 200) return window.setTimeout(f, 0);
  636. var item = obj.items[0];
  637. if(!item) {
  638. window.setTimeout(f, 0, {title:'Video not found', status:4});
  639. } else if(item.status.privacyStatus == 'private') {
  640. window.setTimeout(f, 0, {title:'Private video', status:3});
  641. } else {
  642. var status;
  643. if(item.contentDetails && item.contentDetails.regionRestriction) {
  644. var rr = item.contentDetails.regionRestriction;
  645. if(rr.blocked && rr.blocked.indexOf(country) > -1 || rr.allowed && rr.allowed.indexOf(country) < 0) status = 3;
  646. } else if(!item.status.embeddable) {
  647. status = 5;
  648. }
  649. window.setTimeout(f, 0, {title:item.snippet.title, status:status});
  650. }
  651. });
  652. }
  653. },
  654. 'vm':{
  655. patterns:["vimeo.com"],
  656. parse:function(url) { return /^https?:\/\/vimeo\.com\/(m\/)?([0-9]+)/i.exec(url) ? {vid:RegExp.$2} : null; },
  657. sizes:[[640,360],[853, 480]],
  658. embed:function(vid, t) { return 'https://player.vimeo.com/video/' + vid + '?title=1&portrait=1&byline=1&color=aa0000&autoplay=1'; },
  659. url:function(vid, t) { return 'https://vimeo.com/' + vid; },
  660. icon:'',
  661. request:function(vid, f) {
  662. net.json('https://vimeo.com/api/oembed.json?url=http://vimeo.com/' + vid, function(code, obj) {
  663. if(code == 404) {
  664. window.setTimeout(f, 0, {title:'Video not found', status:4});
  665. } else if(code == 403) {
  666. window.setTimeout(f, 0, {title:'Unknown video', status:5});
  667. } else if(code == 200 && obj) {
  668. window.setTimeout(f, 0, {title:obj.title ? obj.title : 'Unknown video', preview:obj.thumbnail_url});
  669. } else {
  670. window.setTimeout(f, 0);
  671. }
  672. });
  673. }
  674. },
  675. 'vn':{
  676. patterns:["vine.co"],
  677. parse:function(url) { return /^https?:\/\/(www\.)?vine\.co\/v\/([a-z0-9]+)/i.exec(url) ? {vid:RegExp.$2} : null; },
  678. sizes:[[500,500],[660,660]],
  679. embed:function(vid) { return 'https://vine.co/v/' + vid + '/card?audio=1'; },
  680. url:function(vid) { return 'https://vine.co/v/' + vid; },
  681. icon:'',
  682. request:function(vid, f) {
  683. net.text('https://vine.co/v/' + vid, [/"og:title" content="(.+?)">/, /"og:image" content="(.+?)"/], function(code, m) {
  684. var title = m[0], preview = m[1];
  685. if(code == 404) {
  686. window.setTimeout(f, 0, {title:'Video not found', status:4});
  687. } else if(code == 200 && title) {
  688. window.setTimeout(f, 0, {title:title ? title[1] : 'Unknown video', preview:preview ? preview[1] : null});
  689. } else {
  690. window.setTimeout(f, 0);
  691. }
  692. });
  693. }
  694. },
  695. 'll':{
  696. patterns:["liveleak.com/view"],
  697. parse: function(url) { return /^https?:\/\/(m|www)\.liveleak\.com\/view.+i=([0-9a-z_]+)/i.exec(url) ? {vid:RegExp.$2} : null; },
  698. sizes:[[625,352],[852,480]],
  699. embed:function(vid) { return 'https://www.liveleak.com/ll_embed?i=' + vid + '#play'; },
  700. url:function(vid) { return 'https://www.liveleak.com/view?i=' + vid; },
  701. icon:'',
  702. request:function(vid, f) {
  703. net.text('https://www.liveleak.com/view?i=' + vid, [/Item not found!/i, /<title>LiveLeak\.com - (.+?)<\/title>/i, /"og:image" content="(.+?)"/], function(code, m) {
  704. var notfound = m[0], title = m[1], preview = m[2];
  705. if(notfound) {
  706. window.setTimeout(f, 0, {title:'Video not found', status:4});
  707. } else if(code == 200 && title) {
  708. window.setTimeout(f, 0, {title:title ? title[1] : 'Unknown video', preview:preview ? preview[1].replace('_thumb_', '_sf_') : null});
  709. } else {
  710. window.setTimeout(f, 0);
  711. }
  712. });
  713. }
  714. },
  715. 'wh':{
  716. patterns:["worldstarhiphop.com"],
  717. parse: function(url) { return /^https?:\/\/(www\.|m\.)?worldstarhiphop\.com\/.+v=([a-z0-9]+)/i.exec(url) ? {vid:RegExp.$2} : null; },
  718. sizes:[[448,374],[640,534]],
  719. embed:function(vid) { return 'http://www.worldstarhiphop.com/videos/ea/16711680/' + vid; },
  720. url:function(vid) { return 'http://www.worldstarhiphop.com/videos/video.php?v=' + vid; },
  721. icon:'',
  722. request:function(vid, f) {
  723. net.text('http://www.worldstarhiphop.com/videos/video.php?v=' + vid, [/<title>(Video: )?(.+?)\s*(\(\*Warning|<\/title>)/i, /"image_src" href="(.+?)"/], function(code, m) {
  724. var title = m[0], preview = m[1];
  725. if(title) {
  726. window.setTimeout(f, 0, {title:title[2], preview:preview ? preview[1] : null});
  727. } else {
  728. window.setTimeout(f, 0);
  729. }
  730. });
  731. }
  732. },
  733. 'dm':{
  734. patterns:["dailymotion.com/video"],
  735. parse:function(url) { return /^https?:\/\/www\.dailymotion\.com\/video\/([a-z0-9]+)(.+start=([0-9]+))?/i.exec(url) ? {vid:RegExp.$1,t:RegExp.$3} : null; },
  736. sizes:[[620,352],[853,480]],
  737. embed:function(vid, t) { return 'https://www.dailymotion.com/embed/video/' + vid + '?logo=0&autoplay=1' + (t ? '&start=' + t : ''); },
  738. url:function(vid, t) { return 'https://www.dailymotion.com/video/' + vid + (t ? '?start=' + t : ''); },
  739. icon:'',
  740. request:function(vid, f) {
  741. net.json('https://www.dailymotion.com/services/oembed?format=json&url=http://www.dailymotion.com/video/' + vid, function(code, obj) {
  742. if(code == 404) {
  743. window.setTimeout(f, 0, {title:'Video not found', status:4});
  744. } else if(code == 200 && obj) {
  745. window.setTimeout(f, 0, {title:obj.title ? obj.title : 'Unknown video', preview:obj.thumbnail_url});
  746. } else {
  747. window.setTimeout(f, 0);
  748. }
  749. });
  750. }
  751. },
  752. 'cb':{
  753. patterns:["coub.com/view/"],
  754. parse:function(url) { return /^https?:\/\/coub\.com\/view\/([a-z0-9]+)/i.exec(url) ? {vid:RegExp.$1} : null; },
  755. sizes:[[640,360],[853,480]],
  756. embed:function(vid, t) { return 'https://coub.com/embed/' + vid + '?muted=false&autostart=true&noSiteButtons=true'; },
  757. url:function(vid, t) { return 'https://coub.com/view/' + vid; },
  758. icon:'',
  759. request:function(vid, f) {
  760. net.json('https://coub.com/api/oembed.json?url=http://coub.com/view/' + vid, function(code, obj) {
  761. if(code == 404) {
  762. window.setTimeout(f, 0, {title:'Video not found', status:4});
  763. } else if(code == 403) {
  764. window.setTimeout(f, 0, {title:'Unknown video', status:5});
  765. } else if(code == 200 && obj) {
  766. window.setTimeout(f, 0, {title:obj.title ? obj.title : 'Unknown video', preview:obj.thumbnail_url});
  767. } else {
  768. window.setTimeout(f, 0);
  769. }
  770. });
  771. }
  772. },
  773. 'vd':{
  774. patterns:["vid.me/"],
  775. parse:function(url) { return /^https?:\/\/vid\.me\/(e\/)?([a-z0-9]+)$/i.exec(url) ? {vid:RegExp.$2} : null; },
  776. sizes:[[640,360],[853,480]],
  777. embed:function(vid, t) { return 'https://vid.me/e/' + vid + '?autoplay=1'; },
  778. url:function(vid, t) { return 'https://vid.me/' + vid; },
  779. icon:'',
  780. request:function(vid, f) {
  781. net.text('https://vid.me/' + vid, [/"og:title" content="(.+?)">/, /"og:image" content="(.+?)"/], function(code, m) {
  782. var title = m[0], preview = m[1];
  783. if(code == 404) {
  784. window.setTimeout(f, 0, {title:'Video not found', status:4});
  785. } else {
  786. window.setTimeout(f, 0, {title:title ? title[1] : 'Unknown video', preview:preview ? preview[1] : null});
  787. }
  788. });
  789. }
  790. },
  791. 'st':{
  792. patterns:["streamable.com/"],
  793. parse:function(url) { return /^https?:\/\/streamable\.com\/([a-z0-9]+)$/i.exec(url) ? {vid:RegExp.$1} : null; },
  794. sizes:[[650,360],[870,480]],
  795. embed:function(vid, t) { return 'https://streamable.com/e/' + vid + '?autoplay=1'; },
  796. url:function(vid, t) { return 'https://streamable.com/' + vid; },
  797. icon:'',
  798. request:function(vid, f) {
  799. net.text('https://streamable.com/' + vid, [/<h1[^>]*>(.+?)</, /"og:image" content="(.+?)"/], function(code, m) {
  800. var title = m[0], preview = m[1];
  801. if(code == 404) {
  802. window.setTimeout(f, 0, {title:'Video not found', status:4});
  803. } else {
  804. window.setTimeout(f, 0, {title:title ? title[1] : 'Unknown video', preview:preview ? preview[1].replace('http://', 'https://') : null});
  805. }
  806. });
  807. }
  808. },
  809. };
  810.  
  811. var cache = {
  812. get: function(sid, vid) {
  813. if(!cache.obj) cache.load();
  814. var data = cache.obj[sid + vid];
  815. if(!data) return false;
  816. data = data.split('\t');
  817. return data[0] ? {title:data[0],status:data[1],preview:data[2]} : false;
  818. },
  819. set: function(sid, vid, info) {
  820. if(!info) return;
  821. cache.load();
  822. cache.obj[sid + vid] = [info.title,info.status,info.preview].join('\t').trim();
  823. cache.save();
  824. },
  825. save: function() {
  826. if(!cache.obj) return;
  827. try {
  828. cache.clean();
  829. GM_setValue('cache', JSON.stringify(cache.obj));
  830. } catch(ex) { log('Error while saving cache: ' + ex); }
  831. cache.obj = null;
  832. },
  833. load: function() {
  834. try {
  835. cache.obj = JSON.parse(GM_getValue('cache'));
  836. } catch(ex) { }
  837. if(cache.obj == null || typeof(cache.obj) != 'object') cache.obj = {};
  838. },
  839. clean: function() {
  840. var overflow = Object.keys(cache.obj).length - 300;
  841. if(overflow < 1) return;
  842. var i = 0, f = Object.prototype.hasOwnProperty;
  843. for(var p in cache.obj) {
  844. if(f.call(cache.obj, p)) {
  845. delete cache.obj[p];
  846. if(++i == overflow + 20) return;
  847. }
  848. }
  849. },
  850. clear: function(sid) {
  851. cache.load();
  852. var f = Object.prototype.hasOwnProperty;
  853. for(var p in cache.obj) {
  854. if(p.indexOf(sid) == 0 && f.call(cache.obj, p)) {
  855. delete cache.obj[p];
  856. }
  857. }
  858. cache.save();
  859. }
  860. };
  861.  
  862. var cfg = {
  863. settings: {
  864. country: { title:'Check for restrictions? Enter your <a href="http://en.wikipedia.org/wiki/ISO_3166-1_alpha-2#Officially_assigned_code_elements" target="_blank" style="text-decoration:underline!important;" title="e.g. US, GB or DE">country code</a>', default:'', maxlength:2},
  865. reverse_buttons: { title:'Move buttons to the left', default:false, depends:{embed_mode:/player/} },
  866. big: { title:'480p instead of 360p', default:false, depends:{embed_mode:/window|inline/} },
  867. embed_mode: { title:'Embed on click', default:'player', options:{off:'off', player:'in-page popup', inline:'inline', window:'popup window'} },
  868. rewrite: { title:'Rewrite addresses to non-mobile HTTPS URLs', default:true },
  869. urls_only: { title:'Look up info only when link text is URL', default:false },
  870. previews: { title:'Show preview image on hover', default:true },
  871. darken: {},
  872. x: {},
  873. y: {}
  874. },
  875. load: function() {
  876. forEach(cfg.settings, function(n) { cfg[n] = GM_getValue(n, cfg.settings[n].default); });
  877. },
  878. save: function() {
  879. forEach(cfg.settings, function(n) { if(typeof cfg[n] != 'undefined') GM_setValue(n, cfg[n]); });
  880. }
  881. };
  882.  
  883. window.setTimeout(onLoad, 100);
Add Comment
Please, Sign In to add comment