// ==UserScript== // @id youtube-me-again // @name YouTube Me Again! // @namespace hateradio))) // @author hateradio // @version 4.1 // @description YouTube Me Again! automatically converts YouTube and Vimeo links into real embedded videos. Allowing you to hide, rescale, and change any video's aspect ratio. // @homepage https://userscripts.org/scripts/show/60843 // @icon https://dl.dropboxusercontent.com/u/14626536/userscripts/i/ytma/ytma32.png // @icon64 https://dl.dropboxusercontent.com/u/14626536/userscripts/i/ytma/ytma64.png // @screenshot http://i.min.us/ic9lvy.png http://i.min.us/kc9lvy.jpg // @include http://*.neogaf.com/forum/showthread.php* // @include http://*.neogaf.com/forum/showpost.php?p* // @include http://*.neogaf.com/forum/newreply.php* // @include http://*.neogaf.com/forum/editpost.php* // @include http://*.neogaf.com/forum/private.php* // @include http://*.neogaf.net/forum/showthread.php* // @include http://*.neogaf.net/forum/showpost.php?p* // @include http://*.neogaf.net/forum/newreply.php* // @include http://*.neogaf.net/forum/editpost.php* // @include http://*.neogaf.net/forum/private.php* // @match http://*.neogaf.com/forum/showthread.php* // @match http://*.neogaf.com/forum/showpost.php?p* // @match http://*.neogaf.com/forum/newreply.php* // @match http://*.neogaf.com/forum/editpost.php* // @match http://*.neogaf.com/forum/private.php* // @match http://*.neogaf.net/forum/showthread.php* // @match http://*.neogaf.net/forum/showpost.php?p* // @match http://*.neogaf.net/forum/newreply.php* // @match http://*.neogaf.net/forum/editpost.php* // @match http://*.neogaf.net/forum/private.php* // @include http*://*what.cd/forums.php?*viewthread* // @include http*://*what.cd/torrents.php?* // @include http*://*what.cd/user.php?* // @match *://*.what.cd/forums.php?*viewthread* // @match *://*.what.cd/torrents.php?* // @match *://*.what.cd/user.php?* // @updated 24 Aug 2013 | 17,400 // -updated 04 June 2013 | 13,586 // @grant GM_xmlhttpRequest // @downloadURL none // ==/UserScript== // Do not modify and re-release this script. If you would like to add support for another site, tell me and I'll put it in the includes. // Whitelist these sites on NoScript/NotScript/etc. // neogaf.com or neogaf.net, youtube.com, youtube-nocookie.com, vimeo.com, vimeocdn.com, dropboxusercontent.com /*jslint indent: 4, maxerr: 50, browser: true, devel: true, nomen: true, plusplus: true, regexp: true, newcap: true */ (function () { 'use strict'; var $$, strg, update; if (!Function.prototype.bind) { Function.prototype.bind = function (self) { var args = [].slice.call(arguments, 1), fn = this; return function () { return fn.apply(self, args.concat([].slice.call(arguments))); }; }; } // DOM Handle $$ = { s : function (selector, cb) { var s = document.querySelectorAll(selector), i = -1; while (++i < s.length) { if (cb(s[i], i, s) === false) { break; } } }, o : function (object, cb) { var i; for (i in object) { if (object.hasOwnProperty(i)) { if (cb(i, object[i], object) === false) { break; } } } }, a : function (e) { var i = 1, j = arguments.length, f = document.createDocumentFragment(); for (i; i < j; i++) { f.appendChild(arguments[i]); } e.appendChild(f); return e; }, e : function (t, o, e, p) { var a, b, c = document.createElement(t); if (typeof (o) === 'object') { for (a in o) { if (o.hasOwnProperty(a)) { b = a.charAt(0); switch (b) { case '_': c.style[a.substring(1)] = o[a]; break; case '$': c.setAttribute(a.substring(1), o[a]); break; default: c[a] = o[a]; break; } } } } if (e && p) { c.appendChild(e); } else if (e) { e.appendChild(c); } return c; }, p : function (el, search) { // search is a regexp if (search.test(el.tagName)) { return el; } if (el.parentElement) { return this.p(el.parentElement, search); } return false; } }; // S T O R A G E HANDLE strg = { on: (function () { try { var a, b = localStorage, c = Math.random().toString(16).substr(2, 8); b.setItem(c, c); a = b.getItem(c); return a === c ? !b.removeItem(c) : false; } catch (e) { return false; } }()), read: function (key) { return this.on ? JSON.parse(localStorage.getItem(key)) : false; }, save: function (key, dat) { return this.on ? !localStorage.setItem(key, JSON.stringify(dat)) : false; }, wipe: function (key) { return this.on ? !localStorage.removeItem(key) : false; }, zero: function (o) { var k; for (k in o) { if (o.hasOwnProperty(k)) { return false; } } return true; }, grab: function (key, def) { var s = strg.read(key); return strg.zero(s) ? def : s; } }; // U P D A T E HANDLE update = { name : 'YouTube Me Again!', version : 4100, key : 'ujs_YTMA_UPDT_HR', callback : 'ytmaupdater', page : 'https://userscripts.org/scripts/show/60843', urij : 'https://dl.dropboxusercontent.com/u/14626536/userscripts/updt/ytma/ytma.json', uric : 'https://dl.dropboxusercontent.com/u/14626536/userscripts/updt/ytma/ytma.js', // If you get "Failed to load source for:" in Firebug, allow dropboxusercontent.com to run scripts. checkchrome: false, interval: 5, day: (new Date()).getTime(), time: function () { return new Date(this.day + (1000 * 60 * 60 * 24 * this.interval)).getTime(); }, top: document.head || document.body, css: function (t) { if (!this.style) { this.style = document.createElement('style'); this.style.type = 'text/css'; this.top.appendChild(this.style); } this.style.appendChild(document.createTextNode(t + '\n')); }, js: function (t) { var j = document.createElement('script'); j.type = 'text/javascript'; j[/^https?\:\/\//i.test(t) ? 'src' : 'textContent'] = t; this.top.appendChild(j); }, notification: function (j) { if (j) {if (this.version < j.version) { window.localStorage.setItem(this.key, JSON.stringify({date: this.time(), version: j.version, page: j.page })); } else { return true; } } var a = document.createElement('a'), b = JSON.parse(window.localStorage.getItem(this.key)); a.href = b.page || '#'; a.id = 'userscriptupdater'; a.title = 'Update now.'; a.textContent = 'An update for ' + this.name + ' is available.'; document.body.appendChild(a); return true; }, check: function (opt) { if (typeof (GM_updatingEnabled) === 'boolean' || !strg.on) { return; } var stored = strg.read(this.key), J, page; this.csstxt(); if (opt || !stored || stored.date < this.day) { page = stored && stored.page ? stored.page : '#'; strg.save(this.key, {date: this.time(), version: this.version, page: page}); if (!opt && typeof (GM_xmlhttpRequest) === 'function' && !this.chrome()) { GM_xmlhttpRequest({method: 'GET', url: update.urij, onload: function (r) { update.notification(JSON.parse(r.responseText)); }, onerror: function () { update.check(1); } }); } else { J = this.notification.toString().replace('function', 'function ' + this.callback).replace('this.version', this.version).replace(/(?:this\.key)/g, "'" + this.key + "'").replace('this.time()', this.time()).replace('this.name', 'j.name'); this.js(J); this.js(this.uric); } } else if (this.version < stored.version) { this.notification(); } }, chrome: function () { if (this.checkchrome === true && typeof (chrome) === 'object') { return true; } }, csstxt: function () { if (!this.pop) { this.pop = true; this.css('#userscriptupdater,#userscriptupdater:visited{-moz-box-shadow:0 0 6px #787878;-webkit-box-shadow:0 0 6px #787878;box-shadow:0 0 6px #787878;border:1px solid #777;-moz-border-radius:4px;border-radius:4px;cursor:pointer;color:#555;font-family:Arial, Verdana, sans-serif;font-size:11px;font-weight:700;text-align:justify;min-height:45px;position:fixed;z-index:999999;right:10px;top:10px;width:170px;background:#ebebeb url() no-repeat 13px 15px;padding:12px 20px 10px 65px}#userscriptupdater:hover,#userscriptupdater:visited:hover{color:#55698c!important;background-position:13px -85px;border-color:#8f8d96}'); } } }; update.check(); // Y T M A CLASS /** @param id Unique ID @param site Website eg: youtube, vimeo @param a The link */ function YTMA(id, site, a) { if (!YTMA.sites[site]) { return; } this.data = { id : id, uid : id + (YTMA.num += 1), site : site, uri : a.href, thumb : 1 }; this.spn = $$.e('span', {title: 'YouTube Me!', className : 'ylinks _' + this.data.id}); this.spn.addEventListener('click', this.show.bind(this), false); this.wrp = $$.e('div', {className : 'ytmwrap'}); this.wrs = $$.e('div', {id : 'w' + this.data.uid, className : 'yclear y_' + this.data.site}); this.ul = $$.e('ul', {className : 'ytmbar arialsans'}); this.setup(a); } YTMA.num = 0; YTMA.sites = { youtube : 'https://www.youtube-nocookie.com/', vimeo : 'http://vimeo.com/' // soundcloud : 'http://soundcloud.com', // vine : 'https://vine.co/' }; YTMA.siteByTest = { youtu : 'youtube', vimeo : 'vimeo' // soundcloud : 'soundcloud', // vine : 'vine' }; YTMA.reg = { not : /(?:\b(?:smallfont|user_title|spoiler)\b)/, // Class attributes to ignore id : /(?:(?:youtu)(?:\.be\/|.*?(?:v\=|#p\/u\/\d*?\/)|.*?(?:v\=|#p\/c\/[a-zA-Z0-9]+\/\d*?\/))([A-Za-z0-9-_]{11}))|(?:vimeo\.com\/(\d+))/i, //|(?:vine\.co\/v\/([A-Za-z0-9-_]{11}))|(?:soundcloud\.com\/(.+?\/.+)) site : /(youtu)|(vimeo)|(vine)/, time : /(?:t\=(?:(\d+)m)?(\d+)?s?)/, ios : /(?:\b(?:ipod|iphone|ipad))\b/i }; YTMA.img = { play : { youtube : '', vimeo : '' }, fav : { youtube : '', vimeo : '' }, css : { bar : '', load : '' } }; YTMA.selector = 'a[href*="youtube."], a[href*="youtu.be"], a[href*="vimeo.com"]'; //, a[href*="vine.co"] YTMA.pod = YTMA.reg.ios.test(navigator.userAgent); YTMA.ie = document.documentMode; // IE, basically | window.navigator.cpuClass /** 0 "Frame", 1 Flash // 3 Hidden, 4 Small (240p), 5 Medium (360p), 6 Large (480p), 7 (720p) // 1 4:3, 2 16:9, 3 10:16 // Description // URL Anchor // 4 240, 5 360, 6 480, 7 720, 8 1080 */ YTMA.user = { init : function () { this.load(); if (strg.on) { this.form(); this.mark(); } }, defaults : function () { return { focus : 0, desc : 1, // Autoload descriptions. 1 = Yes, 0 = No flash : YTMA.pod || YTMA.ie ? 0 : 1, // if iOS or IE9+, use "html5" player ratio : 2, size : 360, quality : 360 }; }, load : function () { var s = strg.grab('ytmasetts', {}), d = this.defaults();// if(s){console.log(s);} YTMA.user.opt = { flash : isNaN(s.flash) ? d.flash : +s.flash, size : isNaN(s.size) || s.size < 240 ? d.size : +s.size, ratio : isNaN(s.ratio) ? d.ratio : +s.ratio, desc : isNaN(s.desc) ? d.desc : +s.desc, focus : isNaN(s.focus) ? d.focus : +s.focus, quality : isNaN(s.quality) ? d.quality : +s.quality }; // console.log(YTMA.user.opt.toSource()); // console.log(YTMA.user.opt); }, save : function (evt) { var e = evt.target; if (e.tagName.toLowerCase() === 'input') { switch (e.name) { case 'ytmaframe': YTMA.user.opt.flash = +e.checked; break; case 'ytmasize': YTMA.user.opt.size = e.value; break; case 'ytmaratio': YTMA.user.opt.ratio = +e.value; break; case 'ytmainfo': YTMA.user.opt.desc = +e.checked; break; case 'ytmafocus': YTMA.user.opt.focus = +e.checked; break; case 'ytmaiq': YTMA.user.opt.quality = +e.value; break; default: return; } YTMA.user.error.className = strg.save('ytmasetts', YTMA.user.opt) ? 'ytnone' : ''; YTMA.user.load(); // console.log('n'+YTMA.user.opt.toSource()); console.log('n'+YTMA.user.opt); } }, mark : function () { var a = {}; a.ytmaframe = !!YTMA.user.opt.flash; a.ytmainfo = !!YTMA.user.opt.desc; a.ytmafocus = !!YTMA.user.opt.focus; a['ytmaratio' + YTMA.user.opt.ratio] = true; a['ytmasize' + YTMA.user.opt.size] = true; a['ytmaiq' + YTMA.user.opt.quality] = !!YTMA.user.opt.quality; $$.o(a, function (idx, bool) { try { document.getElementById(idx).checked = bool; } catch (e) {} }); }, reset : function () { YTMA.user.opt = YTMA.user.defaults(); YTMA.user.mark(); strg.wipe('ytmasetts'); }, form : function () { var f = [ '
YouTube Me Again! Persistent Site Settings

Change video defaults.

Information

Flash Support

', '
Player Size

', '
Quality

', '
Window Focus

', '
Aspect Ratio

', '

Error! Your settings could not be saved.

Close Reset *Disable this option for iOS and other devices.

' ].join(''); YTMA.form = $$.e('div', {id : 'ytmasettings', className : 'ytnone', innerHTML : f}, document.body); YTMA.user.error = document.getElementById('ytmasettingserror'); YTMA.form.addEventListener('click', YTMA.user.save, false); document.getElementById('ytmaclose').addEventListener('click', YTMA.user.toggle, false); document.getElementById('ytmareset').addEventListener('click', YTMA.user.reset, false); }, toggle : function (force) { YTMA.form.className = (force === true || /(?:ytnone)/.test(YTMA.form.className)) ? '' : 'ytnone'; } }; YTMA.css = function () { update.css(['\n/* youtube me again! */\n.yclear,.ytmwrap,object{clear:both;text-align:left}.yclear{overflow:auto;margin:0 0 6px}#ytmasettings,.arialsans{font-family:Arial,Helvetica,sans-serif!important}.ylinks{display:block;width:118px;height:68px;overflow:hidden;font-style:normal;background-color:#111!important;background-repeat:no-repeat!important;background-position:center -11px!important;cursor:pointer;-moz-border-radius:2px;border-radius:1px;position:relative;float:left;-moz-box-shadow:0 0 2px #222;box-shadow:0 0 2px #222;margin:4px}.ylinks img{border:0;opacity:1;top:28px;left:42px;z-index:9;position:absolute}.ylinks:hover img{opacity:.9}.ytmainit{display:block;background:#111;color:#fff;text-shadow:#333 0 0 2px;opacity:1.0;text-align:left;font-size:11px;margin:0;padding:3px;-moz-border-radius:2px 2px 0 0;border-radius:2px 2px 0 0;height:12px}.ytmainit:after{content:" "; display:block; width:104%; height:52px; background: url(', YTMA.img.play.youtube, ') no-repeat center center;}.ytmainit:hover:after{opacity:.9}.y_vimeo .ytmainit:after{background-image: url(', YTMA.img.play.vimeo, ');}.ytmwrap{overflow:hidden;font-style:normal;margin:3px 0 0;padding:1px 2px} ul.ytmbar{overflow:hidden;margin:0;padding:3px 0 4px;list-style-position:outside !important}.ytmbar li{-webkit-user-select:none;-moz-user-select:none;-o-user-select:none;user-select:none;list-style-type:none;cursor:pointer;float:left;color:#858585;border:1px solid #222;border-bottom:1px solid #000;box-shadow:0 0 1px #333;height:14px;font-size:12px!important;line-height:12px!important;background:#161616 url(', YTMA.img.css.bar, ');margin:0 !important;padding:5px 9px 3px !important}.yliop{-moz-border-radius:2px 0 0 2px;border-radius:2px 0 0 2px}.ylimi{border-left:0!important}li.ylicl{border-left:0!important;-moz-border-radius:0 2px 2px 0;border-radius:0 2px 2px 0;margin:0 2px 0 0 !important}.ylicr{-moz-border-radius:2px;border-radius:2px}.ytmbar li:hover{color:#ccc;text-shadow:1px 1px 0 #333;background:#222 url(', YTMA.img.css.bar, ') 0 -23px} .ytmbar li[id]{color:#ddd;text-shadow:0 0 2px #444} span.yloading{font-style:italic;background:url(', YTMA.img.css.load, ') 0 3px no-repeat;text-indent:17px}.ypg,.ypgv{background:url(', YTMA.img.fav.youtube, ') 2px -1px no-repeat !important;-moz-border-radius:1px;border-radius:1px;padding:0 3px 0 21px !important}.ypgv{background:url(', YTMA.img.fav.vimeo, ') 3px 2px no-repeat !important;padding:0 3px 0 16px !important}.ypg b,.ypg strong{font-weight:400!important}.ypg u{text-decoration:none}.ypg i,.ypg em{font-style:normal}.ypg:hover{background-color:#fff}.ytitled{float:left;max-width:500px;font-style:normal;font-weight:700;margin:5px 0 0 3px}.yterror{color:#CC2F24;font-style:italic} q[ondblclick]{cursor:pointer}.ydescr{display:block;word-wrap:break-word;font-weight:400;max-height:48px;overflow:auto;padding-right:20px}.ydescropen{resize:both;white-space:pre-line;}.ydescropen[style]{max-height:none}#ytmasettings{font-size:12px;width:410px;border-radius:3px;background:#fbfbfb;border:1px solid #bbb;color:#444;box-shadow:0 0 5px rgba(0,0,0,.2), 0 0 3px rgba(239,239,239,.1) inset;margin:1px 0 5px;padding:4px 8px 0}#ytmasettings *{font-weight:400}#ytmasettings p{font-size:12px;clear:both;margin:5px 0;padding:0}#ytmaclears{overflow:hidden}#ytmasettings fieldset{border-radius:3px;border:1px solid #ccc;display:inline-block;width:180px;margin:0 2px;padding:4px 7px 9px}#ytmasettings p>input{margin:3px 3px 0 4px !important}#ytmasettings p>span>input[type=radio]{margin:3px 5px!important}#ytmasettingst{font-size:14px;border-bottom:1px dashed #D00;margin-bottom:6px;padding:0 3px 3px}#ytmasettings .ytmahalf{width:80px;display:inline-block}#ytmasettings .ytmahalf label{width:50px;display:inline-block;cursor:pointer}#ytmasettingserror{font-weight:700;color:#d00}#ytmaopts small{cursor:pointer;display:inline-block;margin-right:20px}#ytmaopts small:hover{color:#222}.ytnone,.ypg br{display:none}'].join('')); }; YTMA.jid = { sites: {}, init: function () { $$.o(YTMA.sites, function (site) { YTMA.jid.sites[site] = { set: {}, current: 0, length: 0, throttle: null }; }); }, add: function (id, site) { if (!this.sites[site].set[id]) { this.sites[site].set[id] = true; this.sites[site].length += 1; } } }; YTMA.json = { maxBatch : 50, intervals : [250, 10000], start: function () { var time = 0; if (YTMA.jid.sites.youtube.length > 100) { time = 2500; YTMA.json.intervals[1] = 15000; } $$.o(YTMA.sites, function (site) { return new YTMA.json.Batch(site, time); }); }, contentScope : function () { var $Y_T_M_A = { version: 'ytma.4.1.dat', cb: function (json) { var o, site; if (json.provider_name) { site = 'soundcloud'; o = $Y_T_M_A.fn.soundcloud(json); } else if (json.entry) { site = 'youtube'; o = $Y_T_M_A.fn.youtube(json.entry); } else { // Object.prototype.toString.call(json) === '[object Array]' site = 'vimeo'; o = $Y_T_M_A.fn.vimeo(json[0]); } if (o.desc && o.desc.length > 140) { o.full = o.desc; o.desc = o.desc.substr(0, 130) + ' . . .'; } $Y_T_M_A.set(site, o); $Y_T_M_A.proc(o); }, fn: { soundcloud: function (j) { return { id: unescape(j.html).match(/tracks\/(\d+)/)[1], title: j.title, desc: j.description, th: j.thumbnail_url }; }, vimeo: function (j) { return { id: j.id, title: j.title + ' ' + $Y_T_M_A.secs(j.duration), desc: j.description.replace(//g, ''), th: unescape(j.thumbnail_medium) }; }, youtube: function (j) { var o = { id: j.media$group.yt$videoid.$t }; if (!j.media$group.yt$duration) { o.error = true; } else { o.title = j.title.$t + ' ' + $Y_T_M_A.secs(j.media$group.yt$duration.seconds); o.desc = j.media$group.media$description.$t; } return o; } }, data: {}, set: function (site, obj) { if (!this.data[site]) { this.data[site] = {}; } this.data[site][obj.id] = obj; this.save(); }, read: function (d) { try { d = JSON.parse(localStorage.getItem(this.version)); if (d instanceof Object) { this.data = d; } } catch (e) { console.error(e); } // alert(this.data); }, save: function () { try { // count this.data props < 1300 localStorage.setItem(this.version, JSON.stringify(this.data)); } catch (e) {} }, secs: function (secs) { var p, H, M, S; try { H = Math.floor(secs / 3600) % 60; M = H ? ('0' + Math.floor(secs / 60) % 60).slice(-2) : Math.floor(secs / 60) % 60; S = ('0' + secs % 60).slice(-2); p = [' (', H ? H + ':' : '', M, ':', S, ')'].join(''); } catch (e) {} return p; }, proc: function (data) { if (data.error) { return $Y_T_M_A.select('.ytitled._' + data.id, function (el) { el.textContent = 'Error, unable to load data.'; el.className = 'yterror ytitled'; }); } if (data.th) { // for Vimeo thumbs $Y_T_M_A.select('.ylinks', function (el) { el.setAttribute('style', 'background: url(' + data.th + ')'); }); } $Y_T_M_A.select('.ytitled._' + data.id, function (el) { var q; el.textContent = data.title; el.className = 'ytitled _' + data.id; if (data.desc) { q = document.createElement('q'); q.className = 'ydescr'; q.textContent = data.desc; if (data.full) { q.setAttribute('data-full', data.full); q.title = 'Click to toggle the length of the description.'; q.ondblclick = $Y_T_M_A.titleToggle; } el.appendChild(q); } }); }, select: function (selector, cb) { var s = document.querySelectorAll(selector), i = -1; while (++i < s.length) { cb(s[i]); } }, titleToggle: function () { this.className = this.className === 'ydescr' ? 'ydescr ydescropen' : 'ydescr'; this.textContent = this.textContent.length < 140 ? this.getAttribute('data-full') : this.getAttribute('data-full').substr(0, 130) + ' . . .'; } }; return $Y_T_M_A; }, runner : function () { // content-scrope runner update.js('var $Y_T_M_A = (' + this.contentScope + ')(); $Y_T_M_A.read();'); }, script : function (id, site) { var s = $$.e('script', { id : id, src : YTMA.DB.json[site].replace('%id', id) }); s.addEventListener('error', YTMA.json.errors, false); s.addEventListener('load', YTMA.json.loads, false); document.body.appendChild(s); }, loads : function () { // console.log('loads'); document.body.removeChild(this); }, errors : function () { $$.s('.ytitled._' + this.id, function (el) { el.textContent = 'Error, unable to load video data.'; el.className = 'yterror ytitled'; }); document.body.removeChild(this); } }; YTMA.$ = YTMA.json.contentScope(); YTMA.$.data = strg.grab(YTMA.$.version, {}); YTMA.json.Batch = function (site, time) { this.site = site; this.ref = YTMA.jid.sites[site]; this.fromStorage(); this.timeouts(time); }; YTMA.json.Batch.prototype = { constructor: YTMA.json.Batch, fromStorage: function () { $$.o(this.ref.set, function (id) { // console.log('current:', id, this.ref.current); if (strg.on && YTMA.$.data[this.site] && YTMA.$.data[this.site][id]) { console.log('from strg', id); delete this.ref.set[id]; this.ref.length -= 1; YTMA.$.proc(YTMA.$.data[this.site][id]); return; } }.bind(this)); }, timeouts : function (ms) { if (this.ref.throttle) { // console.log('clear YTMA.json.throttle'); window.clearInterval(this.ref.throttle); } if (this.ref.length > 0) { console.log(YTMA.json.intervals[1]); window.setTimeout(this.setInterval.bind(this), ms); } }, setInterval: function () { this.ref.throttle = window.setInterval(this.loader.bind(this), YTMA.json.intervals[0]); }, loader : function () { console.log(this.ref.length); if (this.ref.length > 0) { $$.o(this.ref.set, function (id) { // console.log('current:', id, this.ref.current); delete this.ref.set[id]; this.ref.length -= 1; this.ref.current += 1; YTMA.json.script(id, this.site); if (this.ref.current > 0 && this.ref.current % YTMA.json.maxBatch === 0) { this.timeouts(YTMA.json.intervals[1]); // console.log('000'); return false; } }.bind(this)); } else { window.clearInterval(this.ref.throttle); // console.log('cleared: loader'); } } }; YTMA.getIdAndSite = function (uri) { var id, site; try { site = YTMA.siteByTest[YTMA.reg.site.test(uri) ? RegExp.lastMatch : '']; id = uri.href.match(YTMA.reg.id).filter(function (i) { return i; })[1]; if (id && site) { // console.log(id, site); return [id, site]; } throw TypeError('invalid id/site: ' + uri); } catch (e) { console.log(e); return false; } }; YTMA.factory = function () { $$.s(this.selector, function (el, i) { var p = el.parentNode, idSite = YTMA.getIdAndSite(el); if (p && !YTMA.reg.not.test(p.className) && idSite) { YTMA.set[idSite[0] + '_' + i] = new YTMA(idSite[0], idSite[1], el); } }); // console.log(YTMA.set); if (YTMA.user.opt.desc) { YTMA.json.start(); } }; YTMA.set = []; YTMA.init = function () { if (!this.initializer) { this.initializer = true; YTMA.css(); YTMA.jid.init(); YTMA.user.init(); YTMA.json.runner(); YTMA.factory(); } }; YTMA.DB = { ratios : { 1 : 3 / 4, 2 : 9 / 16, 3 : 16 / 10 }, sizes : { 0 : 0, 240 : 360, 360 : 640, 480 : 853, 720 : 1280 }, qualities : { 240 : 'small', 360 : 'medium', 480 : 'large', 720 : 'hd720', 1080 : 'hd1080' }, json : { youtube : 'https://gdata.youtube.com/feeds/api/videos/%id?v=2&alt=json-in-script&callback=$Y_T_M_A.cb', vimeo : 'http://vimeo.com/api/v2/video/%id.json?callback=$Y_T_M_A.cb', soundcloud : 'http://soundcloud.com/oembed?format=json&callback=$Y_T_M_A.cb&iframe=' + true + '&url=' // (!!YTMA.user.opt.flash) http://soundcloud.com/matas/hobnotropic }, sources : { youtube : function () { return [['text/html', YTMA.sites[this.parent.data.site] + 'embed/' + this.parent.data.id + '?theme=dark&color=white&showinfo=1&vq=' + this.$quality + '#at=' + this.$start], ['application/x-shockwave-flash', YTMA.sites[this.parent.data.site] + 'v/' + this.parent.data.id + '?version=3&theme=dark&color=white&showinfo=1&vq=' + this.$quality + '&start=' + this.$start]]; }, vimeo : function () { return [['text/html', 'http://player.vimeo.com/video/' + this.parent.data.id + '?badge=0'], false]; }, vine : function () { return [['text/html', 'https://vine.co/v/' + this.parent.data.id + '/embed#/simple', 'vine-embed'], false]; }, soundcloud : function () { return [['text/html', 'https://w.soundcloud.com/player/?url=http://api.soundcloud.com/tracks/' + this.parent.data.id], false]; } } }; /** @param parent YTMA object */ YTMA.Player = function (parent) { this.parent = parent; this.$obj = this.$quality = this.$size = this.$ratio = this.$source = null; this.init_start(); this.$quality = YTMA.DB.qualities[YTMA.user.opt.quality] || YTMA.DB.qualities[360]; this.$source = YTMA.DB.sources[this.parent.data.site].call(this)[YTMA.user.opt.flash] || YTMA.DB.sources[this.parent.data.site].call(this)[(YTMA.user.opt.flash + 1) % 2]; this.init_frame(); this.size(); }; YTMA.Player.prototype = { constructor : YTMA.Player, dimmensions : function (n) { // var bar = (this.parent.data.site === 'youtube' && this.$size > 0) ? 30 : 0; // +30 pixels for the youtube bar this.$ratio = n === null || isNaN(n) ? YTMA.user.opt.ratio : n; console.log(n, this.$ratio, YTMA.user.opt.ratio); this.$obj.height = Math.round(this.$obj.width * YTMA.DB.ratios[this.$ratio]); // + bar; }, size : function (n) { this.$size = isNaN(n) ? YTMA.user.opt.size : n; this.$obj.width = YTMA.DB.sizes[this.$size] > -1 ? YTMA.DB.sizes[this.$size] : YTMA.DB.sizes[360]; this.dimmensions(this.$ratio); }, reset : function () { this.size(); this.parent.selected.item.call(this.parent, this.parent.ul.querySelector('li[n="' + this.$size + '"]'), 'size'); }, init_start : function () { try { var m = this.parent.data.uri.match(YTMA.reg.time); m = [m[1] * 60, +m[2]]; this.$start = m[1] ? m[0] + m[1] : m[0]; } catch (e) { this.$start = 0; } }, init_frame : function () { // console.log(this.$source); if (YTMA.user.opt.flash && this.$source[0] === 'application/x-shockwave-flash') { this.$obj = $$.e('object', {type : 'application/x-shockwave-flash', data : this.$source[1]}); $$.a(this.$obj, $$.e('param', { name : 'movie', value : this.$obj.data }), $$.e('param', {name : 'AllowScriptAccess', value : 'always'}), $$.e('param', {name : 'allowFullScreen', value : 'true'})); } else { this.$obj = $$.e('iframe', {$frameborder : 0, $allowfullscreen : 'always', type : 'text/html', src : this.$source[1], className: this.$source[2] || ''}); } } }; YTMA.prototype = { // constructor : YTMA, settings : function (evt) { var e = evt.target, n = +e.getAttribute('n'), t = e.getAttribute('data-type'), b = false; if (e.tagName.toLowerCase() === 'li') {// switch or this.settings.fire[t].call(this, e, n) if (t === 'settings') { if (YTMA.form.parentNode != this.wrp) { b = !!this.wrp.insertBefore(YTMA.form, this.ul.nextSibling); } YTMA.user.toggle(b); } else if (t === 'close') { this.toggle(); } else if (t === 'ratio') { this.player.dimmensions(n); this.selected.item.call(this, e, 'ratio'); } else if (t === 'size') { this.player.size(n); this.selected.item.call(this, e, 'size'); } } }, selected : { size : null, ratio: null, item : function (e, type) { e.id = type + this.data.uid; try { this.selected[type].removeAttribute('id'); } catch (er) {} this.selected[type] = e; } }, manual : function (e) { e.preventDefault(); var p = e.target.parentNode; p.className += ' yloading'; p.textContent = 'Loading data . . .'; YTMA.json.script(this.data.id, this.data.site); }, setup : function (link) { var className = this.dom.mod[this.data.site].call(this, link); this.dom.link.call(this, link, className); this.dom.span.call(this); }, dom : { li : function (type, txt, n, t, c) { var l = $$.e('li', {'$data-type' : type, textContent : txt, $n : n, title : t, className : c}); if ((type === 'size' && this.player.$size === n) || (type === 'ratio' && this.player.$ratio === n)) { this.selected.item.call(this, l, type); } return l; }, link : function (link, css) { if (link.getElementsByTagName('img').length === 0 && css) { link.className += ' ' + css; } link.title = 'Visit the video page.'; link.parentNode.insertBefore(this.wrs, link.nextSibling); }, mod : { // modifies the link or other site-specific items; return css class for the link youtube : function (a) { this.spn.addEventListener('mouseover', this.thumb.start.bind(this), false); this.spn.addEventListener('mouseout', this.thumb.stop.bind(this), false); this.spn.style.backgroundImage = ['url(https://i3.ytimg.com/vi/', this.data.id, '/', this.data.thumb, '.jpg)'].join(''); a.href = a.href.replace('http:', 'https:').replace('youtu.be/', 'youtube.com/watch?v='); return 'ypg'; }, vimeo : function () { this.spn.title = 'Vimeo Me Too!'; return 'ypgv'; }, vine : function () { this.spn.title = 'Vine Me!'; return 'ypgvine'; }, soundcloud: function () { this.spn.title = 'Sound Off!'; return 'ypgsc'; } }, ui : function () { var close = 'ylicr', f = document.createDocumentFragment(); // todo, YTMA.db.SIZE.PORTRAIT, etc $$.a(f, this.dom.li.call(this, 'ratio', '4:3', 1, 'SD', 'yliop'), this.dom.li.call(this, 'ratio', '16:9', 2, 'Widescreen', 'ylimi'), this.dom.li.call(this, 'ratio', '10:16', 3, 'Portrait', 'ylicl'), this.dom.li.call(this, 'size', '\u00D8', 0, 'Hide the video.', 'yliop'), this.dom.li.call(this, 'size', 'S', 240, '240p', 'ylimi'), this.dom.li.call(this, 'size', 'M', 360, '360p', 'ylimi'), this.dom.li.call(this, 'size', 'L', 480, '480p', 'ylimi'), this.dom.li.call(this, 'size', 'X', 720, '720p', 'ylicl')); if (strg.on) { f.appendChild(this.dom.li.call(this, 'settings', '!', null, 'Edit YTMA\'s settings.', 'yliop')); close = 'ylicl'; } f.appendChild(this.dom.li.call(this, 'close', '\u00D7', null, 'Close the video.', close)); this.ul.appendChild(f); this.ul.addEventListener('click', this.settings.bind(this), false); this.wrp.appendChild(this.ul); this.spn.parentNode.insertBefore(this.wrp, this.spn.nextSibling); }, span : function () { var a, f = document.createDocumentFragment(); $$.e('span', {className : 'ytmainit arialsans', textContent : this.spn.title}, this.spn); f.appendChild(this.spn); if (YTMA.user.opt.desc) { YTMA.jid.add(this.data.id, this.data.site); f.appendChild($$.e('span', {className : 'ytitled yloading _' + this.data.id, textContent : 'Loading data . . .' })); } else { a = $$.e('a', { textContent: 'Load description.', href: '#', title: 'Load this video\'s description.' }); a.addEventListener('click', this.manual.bind(this), false); f.appendChild($$.a($$.e('span', {className : 'ytitled ymanual _' + this.data.id}), a)); } this.wrs.appendChild(f); } }, createPlayer : function () { this.player = new YTMA.Player(this); this.dom.ui.call(this); }, show : function () { if (!this.player) { this.createPlayer(); } this.toggle(true); if (YTMA.user.opt.focus) { document.location.set = '#' + this.wrs.id; } }, toggle : function (show) {// the object is removed from or reattached to the DOM if (show) { if (this.player.$size === 0) { this.player.reset(); } this.spn.className += ' ytnone'; this.wrp.className = this.wrp.className.replace(' ytnone', ''); this.wrp.appendChild(this.player.$obj); } else { this.spn.className = this.spn.className.replace(' ytnone', ''); this.wrp.className += ' ytnone'; this.wrp.removeChild(this.player.$obj); } }, thumb : { start : function () { this.data.thumb = this.data.thumb < 3 ? this.data.thumb + 1 : 1; this.spn.style.backgroundImage = ['url(https://i3.ytimg.com/vi/', this.data.id, '/', this.data.thumb, '.jpg)'].join(''); this.data.timeout = window.setTimeout(this.thumb.start.bind(this), 1000); }, stop : function () { window.clearTimeout(this.data.timeout); } } }; YTMA.init(); }());