NOTICE: By continued use of this site you understand and agree to the binding Terms of Service and Privacy Policy.
// ==UserScript== // @name Yays! (Yet Another Youtube Script) // @namespace youtube // @description A lightweight and non-intrusive userscript that control video playback and set the preferred player size and playback quality on YouTube. // @version 1.15.1 // @author Eugene Nouvellieu <eugenox_gmail_com> // @license MIT License // @include http*://*.youtube.com/* // @include http*://youtube.com/* // @run-at document-end // @noframes // @grant unsafeWindow // @grant GM_deleteValue // @grant GM_getValue // @grant GM_setValue // @grant GM_xmlhttpRequest // @homepageURL https://eugenox.appspot.com/script/yays // @updateURL https://eugenox.appspot.com/blob/yays/yays.meta.js // @downloadURL https://eugenox.appspot.com/blob/yays/yays.user.js // @icon https://eugenox.appspot.com/blob/yays/yays.icon.png // ==/UserScript== // Copyright (c) 2012-2015 Eugene Nouvellieu <eugenox_gmail_com> // // Permission is hereby granted, free of charge, to any person obtaining a copy of // this software and associated documentation files (the "Software"), to deal in // the Software without restriction, including without limitation the rights to // use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of // the Software, and to permit persons to whom the Software is furnished to do so, // subject to the following conditions: // // The above copyright notice and this permission notice shall be included in all // copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS // FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR // COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER // IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN // CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. function YAYS(unsafeWindow) { 'use strict'; /* * Meta. */ var Meta = { title: 'Yays! (Yet Another Youtube Script)', version: '1.15.1', releasedate: 'Aug 16, 2015', site: 'https://eugenox.appspot.com/script/yays', ns: 'yays' }; /* * Utility functions. */ function each(iterable, callback, scope) { if ('length' in iterable) { for (var i = 0, len = iterable.length; i < len; ++i) { callback.call(scope, i, iterable[i]); } } else { for (var key in iterable) { if (iterable.hasOwnProperty(key)) { callback.call(scope, key, iterable[key]); } } } } function map() { var args = Array.prototype.constructor.apply([], arguments), callback = args.shift() || bind(Array.prototype.constructor, []), results = [], i, len; if (args.length > 1) { var getter = function(arg) { return arg[i]; }; len = Math.max.apply(Math, map(function(arg) { return arg.length; }, args)); for (i = 0; i < len; ++i) { results.push(callback.apply(null, map(getter, args))); } } else { var arg = args[0]; len = arg.length; for (i = 0; i < len; ++i) { results.push(callback(arg[i])); } } return results; } function unique(values) { values.sort(); for (var i = 0, j; i < values.length; ) { j = i; while (values[j] === values[j - 1]) { j++; } if (j - i) { values.splice(i, j - i); } else { ++i; } } return values; } function combine(keys, values) { var object = {}; map(function(key, value) { object[key] = value; }, keys, values); return object; } function merge(target, source, override) { override = override === undefined || override; for (var key in source) { if (override || ! target.hasOwnProperty(key)) { target[key] = source[key]; } } return target; } function extend(base, proto) { function T() {} T.prototype = base.prototype; return merge(new T(), proto); } function noop() { return; } function bind(func, scope, args) { if (args && args.length > 0) { return func.bind.apply(func, [scope].concat(args)); } return func.bind(scope); } function intercept(original, extension) { original = original || noop; return function() { extension.apply(this, arguments); return original.apply(this, arguments); }; } function asyncCall(func, scope, args) { window.setTimeout(bind(func, scope, args), 0); } function asyncProxy(func) { return function() { asyncCall(func, this, Array.prototype.slice.call(arguments)); }; } function buildURL(path, parameters) { var query = []; each(parameters, function(key, value) { query.push(key.concat('=', encodeURIComponent(value))); }); return path.concat('?', query.join('&')); } function parseJSON(data) { if (typeof JSON != 'undefined') { return JSON.parse(data); } return eval('(' + data + ')'); } /* * Script context. */ var Context = (function() { function BasicContext(context, namespace) { if (context) { this._scope = this._createNamespace(context._scope, namespace); this._ns = context._ns + '.' + namespace; } else { this._scope = unsafeWindow; this._ns = 'window'; } } BasicContext.prototype = { _scope: null, _ns: null, _createNamespace: function(scope, name) { return scope[name] = {}; }, protect: function(entity) { return entity; }, publish: function(name, entity) { this._scope[name] = this.protect(entity); return this._ns + '.' + name; }, revoke: function(name) { delete this._scope[name]; } }; var Context; if (typeof exportFunction == 'function') { Context = function(context, namespace) { BasicContext.call(this, context, namespace); }; Context.prototype = extend(BasicContext, { _createNamespace: function(scope, name) { return createObjectIn(scope, {defineAs: name}); }, protect: function(entity) { switch (typeof entity) { case 'function': return exportFunction(entity, this._scope); case 'object': return cloneInto(entity, this._scope); } } }); } else { Context = BasicContext; } return Context; })(); var pageContext = new Context(), scriptContext = new Context(pageContext, Meta.ns); /* * Console singleton. */ var Console = { debug: function() { unsafeWindow.console.debug('[' + Meta.ns + ']' + Array.prototype.join.call(arguments, ' ')); } }; /* * DOM Helper singleton. */ var DH = { ELEMENT_NODE: 1, build: function(def) { switch (Object.prototype.toString.call(def)) { case '[object Object]': var node = this.createElement(def.tag || 'div'); if ('style' in def) { this.style(node, def.style); } if ('attributes' in def) { this.attributes(node, def.attributes); } if ('listeners' in def) { this.listeners(node, def.listeners); } if ('children' in def) { this.append(node, def.children); } return node; case '[object String]': return this.createTextNode(def); default: return def; } }, id: bind(document.getElementById, document), query: bind(document.querySelectorAll, document), createElement: bind(document.createElement, document), createTextNode: bind(document.createTextNode, document), style: function(node, style) { each(style, node.style.setProperty, node.style); }, append: function(node, children) { each([].concat(children), function(i, child) { node.appendChild(this.build(child)); }, this); node.normalize(); }, insertAfter: function(node, children) { var parent = node.parentNode, sibling = node.nextSibling; if (sibling) { each([].concat(children), function(i, child) { parent.insertBefore(this.build(child), sibling); }, this); parent.normalize(); } else { this.append(parent, children); } }, prepend: function(node, children) { if (node.hasChildNodes()) { each([].concat(children), function(i, child) { node.insertBefore(this.build(child), node.firstChild); }, this); } else { this.append(node, children); } }, remove: function(node) { node.parentNode.removeChild(node); }, attributes: function(node, attributes) { each(attributes, node.setAttribute, node); }, hasClass: function(node, cls) { return node.hasAttribute('class') && new RegExp('\\b' + cls + '\\b').test(node.getAttribute('class')); }, addClass: function(node, clss) { node.setAttribute('class', node.hasAttribute('class') ? unique(node.getAttribute('class').concat(' ', clss).trim().split(/ +/)).join(' ') : clss); }, delClass: function(node, clss) { if (node.hasAttribute('class')) { node.setAttribute('class', node.getAttribute('class').replace(new RegExp('\\s*\\b(?:' + clss.replace(/ +/g, '|') + ')\\b\\s*', 'g'), ' ').trim()); } }, listeners: function(node, listeners) { each(listeners, function(type, listener) { this.on(node, type, listener); }, this); }, on: function(node, type, listener) { node.addEventListener(type, listener, false); }, un: function(node, type, listener) { node.removeEventListener(type, listener, false); }, unwrap: function(element) { try { return XPCNativeWrapper.unwrap(element); } catch (e) { return element; } }, walk: function(node, path) { var steps = path.split('/'), step = null; while (node && (step = steps.shift())) { if (step == '..') { node = node.parentNode; continue; } var selector = /^(\w*)(?:\[(\d+)\])?$/.exec(step), name = selector[1], index = Number(selector[2]) || 0; for (var i = 0, j = 0, nodes = node.childNodes; node = nodes.item(i); ++i) { if (node.nodeType == this.ELEMENT_NODE && (! name || node.tagName.toLowerCase() == name) && j++ == index) { break; } } } return node; }, closest: function(node, predicate) { do { if (predicate(node)) { return node; } } while ((node = node.parentNode) && node.nodeType == this.ELEMENT_NODE); return null; } }; /* * i18n */ var _ = (function() { var vocabulary = ["Playback", "START", "PAUSE", "STOP", "AUTO PAUSE", "AUTO STOP", "Set default playback state", "Quality", "AUTO", "ORIGINAL", "Set default video quality", "Size", "AUTO", "WIDE", "Set default player size", "Player settings", "Help"]; function translation(language) { switch (language) { // Hungarian - eugenox case 'hu': return ["Lej\u00e1tsz\u00e1s", "ELIND\u00cdTVA", "SZ\u00dcNETELTETVE", "MEG\u00c1LL\u00cdTVA", "AUTOMATIKUS SZ\u00dcNETELTET\u00c9S", "AUTOMATIKUS MEG\u00c1LL\u00cdT\u00c1S", "Lej\u00e1tsz\u00e1s alap\u00e9rtelmezett \u00e1llapota", "Min\u0151s\u00e9g", "AUTO", "EREDETI", "Vide\u00f3k alap\u00e9rtelmezett felbont\u00e1sa", "M\u00e9ret", "AUTO", "SZ\u00c9LES", "Lej\u00e1tsz\u00f3 alap\u00e9rtelmezett m\u00e9rete", "Lej\u00e1tsz\u00f3 be\u00e1ll\u00edt\u00e1sai", "S\u00fag\u00f3"]; // Dutch - Mike-RaWare case 'nl': return [null, null, null, null, null, null, null, "Kwaliteit", "AUTOMATISCH", null, "Stel standaard videokwaliteit in", null, null, null, null, null, null]; // Spanish - yonane, Dinepada, jdarlan case 'es': return ["Reproducci\u00f3n autom\u00e1tica", "Iniciar", "Pausar", "Detener", "Auto pausa", "Auto detener", "Fijar reproducci\u00f3n autom\u00e1tica", "Calidad", "Auto", "Original", "Calidad por defecto", "Tama\u00f1o", "Autom\u00e1tico", "Ancho", "Tama\u00f1o del reproductor predeterminado", "Configuraci\u00f3n", "Ayuda"]; // German - xemino, ich01 case 'de': return ["Wiedergabe", "Start", "Pause", "Stop", "Auto-Pause", "Auto-Stop", "Standardm\u00e4\u00dfigen Wiedergabezustand setzen", "Qualit\u00e4t", "Auto", "Original", "Standard Video Qualit\u00e4t setzen", "Gr\u00f6\u00dfe", "Auto", "Breit", "Standard Player Gr\u00f6\u00dfe setzen", "Einstellungen", "Hilfe"]; // Portuguese - Pitukinha case 'pt': return [null, null, null, null, null, null, null, "Qualidade", "AUTOM\u00c1TICO", null, "Defini\u00e7\u00e3o padr\u00e3o de v\u00eddeo", null, null, null, null, "Configura\u00e7\u00e3o do usu\u00e1rio", null]; // Greek - TastyTeo case 'el': return ["\u0391\u03bd\u03b1\u03c0\u03b1\u03c1\u03b1\u03b3\u03c9\u03b3\u03ae", null, "\u03a0\u0391\u03a5\u03a3\u0397", "\u03a3\u03a4\u0391\u039c\u0391\u03a4\u0397\u039c\u0391", "\u0391\u03a5\u03a4\u039f\u039c\u0391\u03a4\u0397 \u03a0\u0391\u03a5\u03a3\u0397", "\u0391\u03a5\u03a4\u039f\u039c\u0391\u03a4\u039f \u03a3\u03a4\u0391\u039c\u0391\u03a4\u0397\u039c\u0391", "\u039f\u03c1\u03b9\u03c3\u03bc\u03cc\u03c2 \u03c0\u03c1\u03bf\u03b5\u03c0\u03b9\u03bb\u03b5\u03b3\u03bc\u03ad\u03bd\u03b7\u03c2 \u03ba\u03b1\u03c4\u03ac\u03c3\u03c4\u03b1\u03c3\u03b7\u03c2 \u03b1\u03bd\u03b1\u03c0\u03b1\u03c1\u03b1\u03b3\u03c9\u03b3\u03ae\u03c2", "\u03a0\u03bf\u03b9\u03cc\u03c4\u03b7\u03c4\u03b1", "\u0391\u03a5\u03a4\u039f\u039c\u0391\u03a4\u039f", "\u03a0\u03a1\u039f\u0395\u03a0\u0399\u039b\u039f\u0393\u0397", "\u039f\u03c1\u03b9\u03c3\u03bc\u03cc\u03c2 \u03c0\u03c1\u03bf\u03b5\u03c0\u03b9\u03bb\u03b5\u03b3\u03bc\u03ad\u03bd\u03b7\u03c2 \u03c0\u03bf\u03b9\u03cc\u03c4\u03b7\u03c4\u03b1\u03c2 \u03b2\u03af\u03bd\u03c4\u03b5\u03bf", "\u039c\u03ad\u03b3\u03b5\u03b8\u03bf\u03c2", "\u0391\u03a5\u03a4\u039f\u039c\u0391\u03a4\u039f", "\u03a0\u039b\u0391\u03a4\u03a5", "\u039f\u03c1\u03b9\u03c3\u03bc\u03cc\u03c2 \u03c0\u03c1\u03bf\u03b5\u03c0\u03b9\u03bb\u03b5\u03b3\u03bc\u03ad\u03bd\u03b7\u03c2 \u03b1\u03bd\u03ac\u03bb\u03c5\u03c3\u03b7\u03c2 \u03b1\u03bd\u03b1\u03c0\u03b1\u03c1\u03b1\u03b3\u03c9\u03b3\u03ad\u03b1", "\u0395\u03c0\u03b9\u03bb\u03bf\u03b3\u03ad\u03c2 \u03b1\u03bd\u03b1\u03c0\u03b1\u03c1\u03b1\u03b3\u03c9\u03b3\u03ad\u03b1", "\u0392\u03bf\u03ae\u03b8\u03b5\u03b9\u03b1"]; // French - eXa case 'fr': return [null, null, null, null, null, null, null, "Qualit\u00e9", "AUTO", "ORIGINAL", "Qualit\u00e9 par d\u00e9faut", "Taille", "AUTO", "LARGE", "Taille par d\u00e9faut du lecteur", "Options du lecteur", "Aide"]; // Slovenian - Paranoia.Com case 'sl': return [null, null, null, null, null, null, null, "Kakovost", "Samodejno", null, "Nastavi privzeto kakovost videa", null, null, null, null, "Nastavitve predvajalnika", "Pomo\u010d"]; // Russian - an1k3y case 'ru': return [null, null, null, null, null, null, null, "\u041a\u0430\u0447\u0435\u0441\u0442\u0432\u043e", "\u0410\u0412\u0422\u041e", null, "\u041a\u0430\u0447\u0435\u0441\u0442\u0432\u043e \u0432\u0438\u0434\u0435\u043e \u043f\u043e \u0443\u043c\u043e\u043b\u0447\u0430\u043d\u0438\u044e", "\u0420\u0410\u0417\u041c\u0415\u0420", null, "\u0420\u0410\u0417\u0412\u0415\u0420\u041d\u0423\u0422\u042c", "\u0420\u0430\u0437\u043c\u0435\u0440 \u0432\u0438\u0434\u0435\u043e \u043f\u043e \u0443\u043c\u043e\u043b\u0447\u0430\u043d\u0438\u044e", "\u041d\u0430\u0441\u0442\u0440\u043e\u0439\u043a\u0438 \u043f\u043b\u0435\u0435\u0440\u0430", "\u041f\u043e\u043c\u043e\u0449\u044c"]; // Hebrew - baryoni case 'iw': return [null, null, null, null, null, null, null, "\u05d0\u05d9\u05db\u05d5\u05ea", "\u05d0\u05d5\u05d8\u05d5\u05de\u05d8\u05d9\u05ea", null, "\u05d4\u05d2\u05d3\u05e8 \u05d0\u05ea \u05d0\u05d9\u05db\u05d5\u05ea \u05d1\u05e8\u05d9\u05e8\u05ea \u05d4\u05de\u05d7\u05d3\u05dc \u05e9\u05dc \u05d4\u05d5\u05d9\u05d3\u05d0\u05d5", "\u05d2\u05d5\u05d3\u05dc", null, "\u05e8\u05d7\u05d1", "\u05d4\u05d2\u05d3\u05e8 \u05d0\u05ea \u05d2\u05d5\u05d3\u05dc \u05d1\u05e8\u05d9\u05e8\u05ea \u05d4\u05de\u05d7\u05d3\u05dc \u05e9\u05dc \u05d4\u05e0\u05d2\u05df", "\u05d4\u05d2\u05d3\u05e8\u05d5\u05ea \u05e0\u05d2\u05df", "\u05e2\u05d6\u05e8\u05d4"]; // Chinese - blankhang case 'zh': return ["\u64ad\u653e\u6a21\u5f0f", "\u64ad\u653e", "\u6682\u505c", "\u505c\u6b62", "\u81ea\u52a8\u6682\u505c", "\u81ea\u52a8\u505c\u6b62", "\u8bbe\u7f6e\u9ed8\u8ba4\u64ad\u653e\u6a21\u5f0f", "\u89c6\u9891\u8d28\u91cf", "\u81ea\u52a8", "\u539f\u753b", "\u8bbe\u7f6e\u9ed8\u8ba4\u89c6\u9891\u8d28\u91cf", "\u64ad\u653e\u5668\u5927\u5c0f", "\u81ea\u52a8", "\u5bbd\u5c4f", "\u8bbe\u7f6e\u64ad\u653e\u5668\u9ed8\u8ba4\u5927\u5c0f", "\u64ad\u653e\u5668\u8bbe\u7f6e", "\u5e2e\u52a9"]; // Polish - mkvs case 'pl': return ["Odtwarzanie", "URUCHOM", "WSTRZYMAJ", "ZATRZYMAJ", "AUTOMATYCZNIE WSTRZYMAJ", "AUTOMATYCZNIE ZATRZYMAJ", "Ustaw domy\u015blny stan odtwarzania", "Jako\u015b\u0107", "AUTOMATYCZNA", "ORYGINALNA", "Ustaw domy\u015bln\u0105 jako\u015b\u0107 film\u00f3w", "Rozmiar", "AUTOMATYCZNY", "SZEROKI", "Ustaw domy\u015blny rozmiar odtwarzacza", "Ustawienia odtwarzacza", "Pomoc"]; // Swedish - eson case 'sv': return ["Uppspelning", "START", "PAUS", "STOPP", "AUTOPAUS", "AUTOSTOPP", "Ange uppspelningsl\u00e4ge", "Kvalitet", "AUTO", "ORIGINAL", "Ange standardkvalitet", "Storlek", "AUTO", "BRED", "Ange standardstorlek", "Inst\u00e4llningar", "Hj\u00e4lp"]; // Ukrainian - mukolah case 'uk': return ["\u0421\u0442\u0430\u043d \u043f\u0440\u043e\u0433\u0440\u0430\u0432\u0430\u0447\u0430", "\u0412\u0406\u0414\u0422\u0412\u041e\u0420\u0418\u0422\u0418", "\u041f\u0420\u0418\u0417\u0423\u041f\u0418\u041d\u0418\u0422\u0418", "\u0417\u0423\u041f\u0418\u041d\u0418\u0422\u0418", "\u0410\u0412\u0422\u041e\u041c\u0410\u0422\u0418\u0427\u041d\u0415 \u041f\u0420\u0418\u0417\u0423\u041f\u0418\u041d\u0415\u041d\u041d\u042f", "\u0410\u0412\u0422\u041e\u041c\u0410\u0422\u0418\u0427\u041d\u0415 \u0417\u0423\u041f\u0418\u041d\u0415\u041d\u041d\u042f", "\u0412\u0438\u0431\u0435\u0440\u0456\u0442\u044c \u0441\u0442\u0430\u043d\u0434\u0430\u0440\u0442\u043d\u0438\u0439 \u0441\u0442\u0430\u043d \u0432\u0456\u0434\u0442\u0432\u043e\u0440\u0435\u043d\u043d\u044f", "\u042f\u043a\u0456\u0441\u0442\u044c", "\u0410\u0412\u0422\u041e\u041c\u0410\u0422\u0418\u0427\u041d\u041e", "\u041e\u0420\u0418\u0413\u0406\u041d\u0410\u041b\u042c\u041d\u0410", "\u0412\u0438\u0431\u0435\u0440\u0456\u0442\u044c \u0441\u0442\u0430\u043d\u0434\u0430\u0440\u0442\u043d\u0443 \u044f\u043a\u0456\u0441\u0442\u044c \u0432\u0456\u0434\u0442\u0432\u043e\u0440\u0435\u043d\u043d\u044f", "\u0420\u043e\u0437\u043c\u0456\u0440", "\u0410\u0412\u0422\u041e\u041c\u0410\u0422\u0418\u0427\u041d\u0418\u0419", "\u0428\u0418\u0420\u041e\u041a\u0418\u0419", "\u0412\u0438\u0431\u0435\u0440\u0456\u0442\u044c \u0441\u0442\u0430\u043d\u0434\u0430\u0440\u0442\u043d\u0438\u0439 \u0440\u043e\u0437\u043c\u0456\u0440 \u043f\u0440\u043e\u0433\u0440\u0430\u0432\u0430\u0447\u0430", "\u041d\u0430\u043b\u0430\u0448\u0442\u0443\u0432\u0430\u043d\u043d\u044f \u043f\u0440\u043e\u0433\u0440\u0430\u0432\u0430\u0447\u0430", "\u0414\u043e\u043f\u043e\u043c\u043e\u0433\u0430"]; } return []; } var dictionary = combine(vocabulary, translation((document.documentElement.lang || 'en').substr(0, 2))); return function(text) { return dictionary[text] || text; }; })(); /** * @class ScopedStorage */ function ScopedStorage(scope, namespace) { this._scope = scope; this._ns = namespace; } ScopedStorage.prototype = { _scope: null, _ns: null, getItem: function(key) { return this._scope.getItem(this._ns + '.' + key); }, setItem: function(key, value) { this._scope.setItem(this._ns + '.' + key, value); }, removeItem: function(key) { this._scope.removeItem(this._ns + '.' + key); } }; var scriptStorage = new ScopedStorage(unsafeWindow.localStorage, Meta.ns); /* * Configuration handler singleton. */ var Config = (function() { // Greasemonkey compatible if (typeof GM_getValue == 'function') { return { get: GM_getValue, set: GM_setValue, del: GM_deleteValue }; } var configStorage = new ScopedStorage(scriptStorage, 'config'); // HTML5 return { get: function(key) { return configStorage.getItem(key); }, set: function(key, value) { configStorage.setItem(key, value); }, del: function(key) { configStorage.removeItem(key); } }; })(); /** * @class JSONRequest * Create XHR or JSONP requests. */ var JSONRequest = (function() { var Request = null; // XHR if (typeof GM_xmlhttpRequest == 'function') { Request = function(url, parameters, callback) { this._callback = callback; GM_xmlhttpRequest({ method: 'GET', url: buildURL(url, parameters), onload: bind(this._onLoad, this) }); }; Request.prototype = { _onLoad: function(response) { this._callback(parseJSON(response.responseText)); } }; } // JSONP else { Request = function(url, parameters, callback) { this._callback = callback; this._id = 'jsonp_' + Request.counter++; parameters.callback = scriptContext.publish(this._id, bind(this._onLoad, this)); this._scriptNode = document.body.appendChild(DH.build({ tag: 'script', attributes: { 'type': 'text/javascript', 'src': buildURL(url, parameters) } })); }; Request.counter = 0; Request.prototype = { _callback: null, _id: null, _scriptNode: null, _onLoad: function(response) { this._callback(response); scriptContext.revoke(this._id); document.body.removeChild(this._scriptNode); } }; } return Request; })(); /* * Update checker. */ (function() { if (new Date().valueOf() - Number(scriptStorage.getItem('update_checked_at')) < 86400000) { // 1 day return; } var popup = null; new JSONRequest(Meta.site + '/changelog', {version: Meta.version}, function(changelog) { scriptStorage.setItem('update_checked_at', new Date().valueOf().toFixed()); if (changelog && changelog.length) { popup = renderPopup(changelog); } }); function renderPopup(changelog) { return document.body.appendChild(DH.build({ style: { 'position': 'fixed', 'bottom': '0', 'width': '100%', 'z-index': '1000', 'background-color': '#f1f1f1', 'border-top': '1px solid #cccccc' }, children: { style: { 'margin': '15px' }, children: [{ tag: 'strong', children: ['There is an update available for ', Meta.title, '.'] }, { tag: 'p', style: { 'margin': '10px 0' }, children: [ 'You are using version ', { tag: 'strong', children: Meta.version }, ', released on ', { tag: 'em', children: Meta.releasedate }, '. Please consider updating to the latest version.' ] }, { style: { 'margin': '10px 0', 'max-height': '150px', 'overflow-y': 'auto' }, children: { tag: 'a', children: 'Show changes', listeners: { click: function(e) { e.preventDefault(); DH.insertAfter(e.target, map(function(entry) { return { style: { 'margin-bottom': '5px' }, children: [{ tag: 'strong', children: entry.version }, ' ', { tag: 'em', children: ['(', entry.date, ')'] }, { style: { 'padding': '0 0 2px 10px', 'white-space': 'pre' }, children: [].concat(entry.note).join('\n') }] }; }, [].concat(changelog))); DH.remove(e.target); } } } }, { children: map(function(text, handler) { return DH.build({ tag: 'button', attributes: { 'type': 'button', 'class': 'yt-uix-button yt-uix-button-default' }, style: { 'margin-right': '10px', 'padding': '5px 15px' }, children: text, listeners: { 'click': handler } }); }, ['Update', 'Dismiss'], [openDownloadSite, removePopup]) }] } })); } function removePopup() { document.body.removeChild(popup); } function openDownloadSite() { removePopup(); unsafeWindow.open(buildURL(Meta.site + '/download', {version: Meta.version})); } })(); /* * Migrations. */ (function(currentVersion) { var previousVersion = Config.get('version') || scriptStorage.getItem('version') || '1.0'; if (previousVersion == currentVersion) { return; } previousVersion = map(Number, previousVersion.split('.')); each([ { // Added "144p" to the quality levels. version: '1.7', apply: function() { var videoQuality = Number(Config.get('video_quality')); if (videoQuality > 0 && videoQuality < 7) { Config.set('video_quality', ++videoQuality); } } }, { // Autoplay reworked. version: '1.8', apply: function() { switch (Number(Config.get('auto_play'))) { case 1: // OFF > PAUSE Config.set('video_playback', 1); break; case 2: // AUTO > AUTO PAUSE Config.set('video_playback', 3); break; } Config.del('auto_play'); } }, { // Added "1440p" to the quality levels. version: '1.10', apply: function() { var videoQuality = Number(Config.get('video_quality')); if (videoQuality > 6) { Config.set('video_quality', ++videoQuality); } } }, { // Introduced the ScopedStorage class. version: '1.14', apply: function() { // Using a unique ScopedStorage for config outside of GM. each(['video_playback', 'video_quality', 'player_size', 'version'], function(i, key) { var value = scriptStorage.getItem(key); if (value) { Config.set(key, value); scriptStorage.removeItem(key); } }); // Removed from the config. Config.del('update_checked_at'); } }, { // Removed "FIT" from player sizes. version: '1.15', apply: function() { if (Number(Config.get('player_size')) == 2) { Config.set('player_size', 1); } } } ], function(i, migration) { var migrationVersion = map(Number, migration.version.split('.')); for (var j = 0, parts = Math.max(previousVersion.length, migrationVersion.length); j < parts; ++j) { if ((previousVersion[j] || 0) < (migrationVersion[j] || 0)) { Console.debug('Applying migration', migration.version); migration.apply(); break; } } }); Config.set('version', currentVersion); })(Meta.version); /** * @class Player */ function Player(element) { this._element = element; this._context = new Context(scriptContext, 'player' + Player._elements.indexOf(element)); this._exportApiInterface(); Console.debug('Player ready'); this._muted = Number(this.isMuted() && ! Number(Player._storage.getItem('muted'))); this._addStateChangeListener(); } merge(Player, { UNSTARTED: -1, ENDED: 0, PLAYING: 1, PAUSED: 2, BUFFERING: 3, CUED: 5, _elements: [], _storage: new ScopedStorage(scriptStorage, 'player'), test: function(element) { return typeof element.getApiInterface == 'function'; }, create: function(element) { switch (element.tagName) { case 'EMBED': return new FlashPlayer(element); case 'DIV': return new HTML5Player(element); } throw 'Unknown player type'; }, initialize: function(element) { if (this._elements.indexOf(element) > -1) { throw 'Player already initialized'; } var index = this._elements.indexOf(null); if (index > -1) { this._elements[index] = element; } else { this._elements.push(element); } return this.create(element); }, invalidate: function(player) { this._elements[this._elements.indexOf(player._element)] = null; player.invalidate(); } }); Player.prototype = { _element: null, _context: null, _muted: 0, _exportApiInterface: function() { each(this._element.getApiInterface(), function(i, method) { if (! (method in this)) { this[method] = bind(this._element[method], this._element); } }, this); }, _unexportApiInterface: function() { each(this._element.getApiInterface(), function(i, method) { if (this.hasOwnProperty(method)) { delete this[method]; } }, this); }, _onStateChange: function(state) { Console.debug('State changed to', ['unstarted', 'ended', 'playing', 'paused', 'buffering', undefined, 'cued'][state + 1]); this.onStateChange(state); }, _addStateChangeListener: function() { this.addEventListener('onStateChange', this._context.publish('onPlayerStateChange', asyncProxy(bind(this._onStateChange, this)))); }, _removeStateChangeListener: function() { this.removeEventListener('onStateChange', this._context.publish('onPlayerStateChange', noop)); }, invalidate: function() { this._removeStateChangeListener(); this._unexportApiInterface(); delete this.onStateChange; delete this._element; Console.debug('Player invalidated'); this.invalidate = noop; }, onStateChange: noop, isPlayerState: function() { return Array.prototype.indexOf.call(arguments, this.getPlayerState()) > -1; }, getVideoId: function() { try { return this.getVideoData().video_id; } catch (e) { return (this.getVideoUrl().match(/\bv=([\w-]+)/) || [, undefined])[1]; } }, restartPlayback: function() { if (this.getCurrentTime() > 60) { Console.debug('Restart threshold exceeded'); return; } var code = (location.hash + location.search).match(/\bt=(?:(\d+)h)?(?:(\d+)m)?(?:(\d+)s?)?/) || new Array(4), seconds = (Number(code[1]) || 0) * 3600 + (Number(code[2]) || 0) * 60 + (Number(code[3]) || 0); this.seekTo(seconds, true); }, resetState: function() { this.seekTo(this.getCurrentTime(), true); }, mute: function() { if (! this._muted++) { this._element.mute(); Player._storage.setItem('muted', '1'); Console.debug('Player muted'); } }, unMute: function() { if (! --this._muted) { this._element.unMute(); Player._storage.setItem('muted', '0'); Console.debug('Player unmuted'); } }, playVideo: function() { this._element.playVideo(); Console.debug('Playback started'); }, pauseVideo: function() { this._element.pauseVideo(); Console.debug('Playback paused'); }, stopVideo: function() { this._element.stopVideo(); Console.debug('Playback stopped'); }, setPlaybackQuality: function(quality) { this._element.setPlaybackQuality(quality); Console.debug('Quality changed to', quality); }, setStageMode: noop }; /** * @class FlashPlayer */ function FlashPlayer(element) { Player.call(this, element); } FlashPlayer.prototype = extend(Player, { _exportApiInterface: function() { try { Player.prototype._exportApiInterface.call(this); } catch (e) { throw 'Player has not loaded yet'; } }, _unexportApiInterface: function() { try { Player.prototype._unexportApiInterface.call(this); } catch (e) { Console.debug('Player has unloaded'); } }, _removeStateChangeListener: function() { try { Player.prototype._removeStateChangeListener.call(this); } catch (e) { Console.debug('Player has unloaded'); } } }); /** * @class HTML5Player */ function HTML5Player(element) { Player.call(this, element); } HTML5Player.prototype = extend(Player, { restartPlayback: function() { Player.prototype.restartPlayback.call(this); this.restartPlayback = noop; }, setStageMode: function(enable) { this.setSizeStyle(true, enable); } }); /** * @class Button */ function Button(label, tooltip) { this._node = DH.build(this._def(tooltip, label, this._indicator = DH.build('-'))); } Button.prototype = { _indicator: null, _node: null, _def: function(tooltip, label, indicator) { return { tag: 'button', attributes: { 'type': 'button', 'class': 'yt-uix-button yt-uix-button-default yt-uix-tooltip', 'title': tooltip }, listeners: { 'click': bind(this._onClick, this) }, style: { 'margin': '0 0.5%' }, children: [{ tag: 'span', attributes: { 'class': 'yt-uix-button-content' }, children: label }, { tag: 'span', style: { 'font-size': '14px', 'margin-left': '5px' }, attributes: { 'class': 'yt-uix-button-content' }, children: indicator }] }; }, _onClick: function() { this.handler(); this.refresh(); }, refresh: function() { this._indicator.data = this.display(); }, render: function() { this.refresh(); return this._node; }, handler: noop, display: noop }; /** * @class PlayerOption */ function PlayerOption(player, key) { this._player = player; this._key = key; } PlayerOption.prototype = { _player: null, _key: null, get: function() { return Number(Config.get(this._key) || '0'); }, set: function(value) { Config.set(this._key, Number(value)); }, apply: noop, cease: noop }; /** * @class PlayerOption.Button */ PlayerOption.Button = function(option) { Button.call(this, this.label, this.tooltip); this._option = option; }; PlayerOption.Button.extend = function(attributes) { var superclass = this; function Button(option) { superclass.call(this, option); } Button.prototype = extend(superclass, attributes); return Button; }; PlayerOption.Button.prototype = extend(Button, { _option: null, label: null, tooltip: null, states: null, handler: function() { this._option.set((this._option.get() + 1) % this.states.length); }, display: function() { return this.states[this._option.get()]; } }); /** * @class SilentPlayerOption */ function SilentPlayerOption(player, key) { PlayerOption.call(this, player, key); } SilentPlayerOption.prototype = extend(PlayerOption, { _muted: false, mute: function(state) { if (this._muted != state) { if (state) { this._player.mute(); } else { this._player.unMute(); } this._muted = state; } }, cease: function() { this.mute(false); } }); /** * @class VideoPlayback */ function VideoPlayback(player) { SilentPlayerOption.call(this, player, 'video_playback'); switch (this.get()) { case 0: // PLAY this._applied = true; break; case 3: // AUTO PAUSE case 4: // AUTO STOP // Video is visible or opened in the same window. if (this._isVisible() || unsafeWindow.history.length > 1) { this._applied = true; } // Video is opened in a background tab. else { this._handler = pageContext.protect(bind(this._handler, this)); DH.on(unsafeWindow, 'focus', this._handler); DH.on(unsafeWindow, 'blur', this._handler); } break; } } VideoPlayback.prototype = extend(SilentPlayerOption, { _applied: false, _timer: null, _handler: function(e) { switch (e.type) { case 'focus': if (this._timer === null) { this._timer = window.setTimeout(bind(function() { if (this._applied) { this._player.resetState(); this._player.playVideo(); Console.debug('Playback autostarted'); } else { this._applied = true; Console.debug('Playback not affected'); this.mute(false); } DH.un(unsafeWindow, 'focus', this._handler); DH.un(unsafeWindow, 'blur', this._handler); this._timer = null; }, this), 500); } break; case 'blur': if (this._timer !== null) { clearTimeout(this._timer); this._timer = null; } break; } }, // @see http://www.w3.org/TR/page-visibility/ _isVisible: function() { var doc = unsafeWindow.document; return doc.hidden === false || doc.mozHidden === false || doc.webkitHidden === false; }, apply: function() { if (! this._applied) { this.mute(true); if (this._player.isPlayerState(Player.PLAYING)) { this._applied = true; this._player.restartPlayback(); if (this.get() % 2) { // (AUTO) PAUSE this._player.pauseVideo(); } else { // (AUTO) STOP this._player.stopVideo(); } this.mute(false); } } } }); /** * @class VideoPlayback.Button */ VideoPlayback.Button = PlayerOption.Button.extend({ label: _('Playback'), tooltip: _('Set default playback state'), states: [_('START'), _('PAUSE'), _('STOP'), _('AUTO PAUSE'), _('AUTO STOP')] }); /** * @class VideoQuality */ function VideoQuality(player) { SilentPlayerOption.call(this, player, 'video_quality'); this._applied = ! this.get(); } VideoQuality.prototype = extend(SilentPlayerOption, { _applied: false, apply: function() { if (! this._applied) { this.mute(true); if (this._player.isPlayerState(Player.PLAYING, Player.PAUSED)) { this._applied = true; var quality = ['tiny', 'small', 'medium', 'large', 'hd720', 'hd1080', 'hd1440', 'highres'][this.get() - 1]; if (quality != this._player.getPlaybackQuality()) { this._player.restartPlayback(); this._player.setPlaybackQuality(quality); } asyncCall(this.apply, this); } } else if (this._player.isPlayerState(Player.PLAYING, Player.CUED)) { this.mute(false); } } }); /** * @class VideoQuality.Button */ VideoQuality.Button = PlayerOption.Button.extend({ label: _('Quality'), tooltip: _('Set default video quality'), states: [_('AUTO'), '144p', '240p', '360p', '480p', '720p', '1080p', '1440p', _('ORIGINAL')] }); /** * @class PlayerSize */ function PlayerSize(player) { PlayerOption.call(this, player, 'player_size'); } PlayerSize.prototype = extend(PlayerOption, { apply: function() { var mode = this.get(), rules = []; switch (mode) { case 4: // 1080p rules.push( '.watch-stage-mode .player-width {', 'left: -960.0px !important;', 'width: 1920px !important;', '}', '.watch-stage-mode .player-height {', 'height: 1110px !important;', '}', '.watch-stage-mode .html5-video-content[style*="640"], .watch-stage-mode .html5-video-content[style*="360"], .watch-stage-mode .html5-main-video[style*="640"], .watch-stage-mode .html5-main-video[style*="360"]', '{', 'transform: matrix(3, 0, 0, 3, 640, 360) !important; -o-transform: matrix(3, 0, 0, 3, 640, 360) !important; -moz-transform: matrix(3, 0, 0, 3, 640, 360) !important; -webkit-transform: matrix(3, 0, 0, 3, 640, 360) !important;', '}', '.watch-stage-mode .html5-video-content[style*="854"], .watch-stage-mode .html5-video-content[style*="480"], .watch-stage-mode .html5-main-video[style*="854"], .watch-stage-mode .html5-main-video[style*="480"]', '{', 'transform: matrix(2.24824, 0, 0, 2.24824, 533, 299.578) !important; -o-transform: matrix(2.24824, 0, 0, 2.24824, 533, 299.578) !important; -moz-transform: matrix(2.24824, 0, 0, 2.24824, 533, 299.578) !important; -webkit-transform: matrix(2.24824, 0, 0, 2.24824, 533, 299.578) !important;', '}', '.watch-stage-mode .html5-video-content[style*="1280"], .watch-stage-mode .html5-video-content[style*="720"], .watch-stage-mode .html5-main-video[style*="1280"], .watch-stage-mode .html5-main-video[style*="720"]', '{', 'transform: matrix(1.5, 0, 0, 1.5, 320, 180) !important; -o-transform: matrix(1.5, 0, 0, 1.5, 320, 180) !important; -moz-transform: matrix(1.5, 0, 0, 1.5, 320, 180) !important; -webkit-transform: matrix(1.5, 0, 0, 1.5, 320, 180) !important;', '}' ); break; case 3: // 720p rules.push( '.watch-stage-mode .player-width {', 'left: -640.0px !important;', 'width: 1280px !important;', '}', '.watch-stage-mode .player-height {', 'height: 750px !important;', '}', '.watch-stage-mode .html5-video-content[style*="640"], .watch-stage-mode .html5-video-content[style*="360"], .watch-stage-mode .html5-main-video[style*="640"], .watch-stage-mode .html5-main-video[style*="360"]', '{', 'transform: matrix(2, 0, 0, 2, 320, 180) !important; -o-transform: matrix(2, 0, 0, 2, 320, 180) !important; -moz-transform: matrix(2, 0, 0, 2, 320, 180) !important; -webkit-transform: matrix(2, 0, 0, 2, 320, 180) !important;', '}', '.watch-stage-mode .html5-video-content[style*="854"], .watch-stage-mode .html5-video-content[style*="480"], .watch-stage-mode .html5-main-video[style*="854"], .watch-stage-mode .html5-main-video[style*="480"]', '{', 'transform: matrix(1.49883, 0, 0, 1.49883, 213, 119.719) !important; -o-transform: matrix(1.49883, 0, 0, 1.49883, 213, 119.719) !important; -moz-transform: matrix(1.49883, 0, 0, 1.49883, 213, 119.719) !important; -webkit-transform: matrix(1.49883, 0, 0, 1.49883, 213, 119.719) !important;', '}' ); break; case 2: // 480p rules.push( '.watch-stage-mode .player-width {', 'left: -427.0px !important;', 'width: 854px !important;', '}', '.watch-stage-mode .player-height {', 'height: 510px !important;', '}', '.watch-stage-mode .html5-video-content[style*="640"], .watch-stage-mode .html5-video-content[style*="360"], .watch-stage-mode .html5-main-video[style*="640"], .watch-stage-mode .html5-main-video[style*="360"]', '{', 'transform: matrix(1.33438, 0, 0, 1.33438, 107, 60.1875) !important; -o-transform: matrix(1.33438, 0, 0, 1.33438, 107, 60.1875) !important; -moz-transform: matrix(1.33438, 0, 0, 1.33438, 107, 60.1875) !important; -webkit-transform: matrix(1.33438, 0, 0, 1.33438, 107, 60.1875) !important;', '}', '.watch-stage-mode .html5-video-content[style*="1280"], .watch-stage-mode .html5-video-content[style*="720"], .watch-stage-mode .html5-main-video[style*="1280"], .watch-stage-mode .html5-main-video[style*="720"]', '{', 'transform: matrix(0.667188, 0, 0, 0.667188, -213, -119.812) !important; -o-transform: matrix(0.667188, 0, 0, 0.667188, -213, -119.812) !important; -moz-transform: matrix(0.667188, 0, 0, 0.667188, -213, -119.812) !important; -webkit-transform: matrix(0.667188, 0, 0, 0.667188, -213, -119.812) !important;', '}' ); break; case 1: // WIDE break; default: return; } if (rules.length) { rules.push( '.watch-stage-mode .html5-main-video {', 'z-index: -1;', '}' ); DH.id('yays-player-size') || DH.append(document.body, { tag: 'style', attributes: { 'id': 'yays-player-size', 'type': 'text/css' }, children: rules }); } var page = DH.id('page'); DH.delClass(page, 'watch-non-stage-mode'); DH.addClass(page, 'watch-stage-mode watch-wide'); this._player.setStageMode(true); Console.debug('Size set to', ['wide', '480p', '720p', '1080p'][mode - 1]); } }); /** * @class PlayerSize.Button */ PlayerSize.Button = PlayerOption.Button.extend({ label: _('Size'), tooltip: _('Set default player size'), states: [_('AUTO'), _('WIDE'), '480p', '720p', '1080p'] }); /** * @class UI * Abstract UI class. */ function UI(content) { this.content = content; this.button = DH.build(this._def.button(bind(this.toggle, this))); this.panel = DH.build(this._def.panel(content)); } merge(UI, { instance: null, initialize: function(type, content) { if (this.instance) { this.instance.destroy(); } return this.instance = new type(content); } }); UI.prototype = { _def: { icon: function(def) { def = merge({tag: 'img', attributes: {}}, def); def.attributes.src = ''; return def; }, button: function(click) { return { listeners: { 'click': click } }; }, panel: function(content) { return [{ style: { 'margin-bottom': '10px' }, children: [{ tag: 'strong', children: _('Player settings') }, { tag: 'a', attributes: { 'href': Meta.site, 'target': '_blank' }, style: { 'margin-left': '4px', 'vertical-align': 'super', 'font-size': '10px' }, children: _('Help') }] }, { style: { 'text-align': 'center' }, children: content.render() }]; } }, content: null, button: null, panel: null, destroy: function() { DH.remove(this.button); DH.remove(this.panel); }, toggle: function() { this.content.refresh(); } }; /** * @class UI.Content */ UI.Content = function(buttons) { this._buttons = buttons; }; UI.Content.prototype = { _buttons: null, render: function() { return map(function(button) { return button.render(); }, this._buttons); }, refresh: function() { each(this._buttons, function(i, button) { button.refresh(); }); } }; /** * @class UI.Requirement */ UI.Requirement = function(queries) { this._queries = [].concat(queries); }; UI.Requirement.prototype = { _queries: null, test: function() { return DH.query(this._queries.join(', ')).length >= this._queries.length; } }; /** * @class WatchUI */ function WatchUI(buttons) { UI.call(this, new UI.Content(buttons)); } WatchUI.prototype = extend(UI, { _def: { panel: function(content) { return { attributes: { 'id': 'action-panel-yays', 'class': 'action-panel-content hid', 'data-panel-loaded': 'true' }, children: UI.prototype._def.panel(content) }; } } }); /** * @class Watch8UI */ function Watch8UI(buttons) { WatchUI.call(this, buttons); DH.append(DH.id('watch8-secondary-actions'), this.button); DH.append(DH.id('watch-action-panels'), this.panel); } Watch8UI.requirement = new UI.Requirement(['#page.watch #watch8-secondary-actions', '#page.watch #watch-action-panels']); Watch8UI.prototype = extend(WatchUI, { _def: { button: function(click) { return { tag: 'span', children: { tag: 'button', attributes: { 'type': 'button', 'class': 'action-panel-trigger yt-uix-button yt-uix-button-empty yt-uix-button-has-icon yt-uix-button-opacity yt-uix-button-size-default yt-uix-tooltip', 'data-button-toggle': 'true', 'data-trigger-for': 'action-panel-yays', 'data-tooltip-text': _('Player settings') }, listeners: { 'click': click }, children: { tag: 'span', attributes: { 'class': 'yt-uix-button-icon-wrapper' }, children: UI.prototype._def.icon({ attributes: { 'class': 'yt-uix-button-icon' } }) } } }; }, panel: WatchUI.prototype._def.panel } }); /** * @class ChannelUI */ function ChannelUI(buttons) { UI.call(this, new UI.Content(buttons)); DH.append(DH.id('channel-navigation-menu'), { tag: 'li', children: [this.button, this.panel] }); } ChannelUI.requirement = new UI.Requirement('#channel-navigation-menu'); ChannelUI.prototype = extend(UI, { _def: { button: function(click) { return { tag: 'button', attributes: { 'type': 'button', 'role': 'button', 'class': 'yt-uix-button yt-uix-button-empty yt-uix-button-epic-nav-item yt-uix-tooltip flip', 'data-button-menu-id': 'yays-panel-dropdown', 'data-tooltip-text': _('Player settings') }, style: { 'position': 'absolute', 'right': '20px', 'width': '30px' }, listeners: { 'click': click }, children: { tag: 'span', attributes: { 'class': 'yt-uix-button-icon-wrapper' }, style: { 'opacity': '0.75' }, children: UI.prototype._def.icon() } }; }, panel: function(content) { return { attributes: { 'id': 'yays-panel-dropdown', 'class': 'epic-nav-item-dropdown hid' }, style: { 'padding': '5px 10px 10px', 'width': '450px' }, children: UI.prototype._def.panel(content) }; } } }); /* * Ready callbacks. */ function onReady(player) { var videoPlayback = new VideoPlayback(player), videoQuality = new VideoQuality(player), playerSize = new PlayerSize(player), previousVideo = player.getVideoId(); player.onStateChange = function() { try { var currentVideo = player.getVideoId(); if (currentVideo == previousVideo) { videoQuality.apply(); videoPlayback.apply(); } else { videoQuality.cease(); videoPlayback.cease(); throw null; } } catch (e) { Player.invalidate(player); asyncCall(onPlayerReady); } }; videoQuality.apply(); videoPlayback.apply(); if (Watch8UI.requirement.test()) { playerSize.apply(); UI.initialize(Watch8UI, [ new VideoQuality.Button(videoQuality), new PlayerSize.Button(playerSize), new VideoPlayback.Button(videoPlayback) ]); } else if (ChannelUI.requirement.test()) { UI.initialize(ChannelUI, [ new VideoQuality.Button(videoQuality), new VideoPlayback.Button(videoPlayback) ]); } } function onPlayerReady() { each(DH.query('video, embed'), function(i, node) { var player = DH.closest(node, function(node) { return Player.test(DH.unwrap(node)); }); if (player) { try { player = Player.initialize(DH.unwrap(player)); onReady(player); Console.debug('Initialization finished'); } catch (e) { Console.debug(e); } } }); } onPlayerReady(); // FIXME: The call to an exported function is rejected if a function (or an // object with methods) is passed as argument. // // This restriction can be lifted in FF 33+ by setting the 'allowCallbacks' // option when exporting the function. var node = DH.build({ tag: 'script', attributes: { 'type': 'text/javascript' } }); node.text = 'function onYouTubePlayerReady() { ' + scriptContext.publish('onPlayerReady', intercept(unsafeWindow.onYouTubePlayerReady, asyncProxy(onPlayerReady))) + '(); }'; document.documentElement.appendChild(node); document.documentElement.removeChild(node); } // YAYS if (window.top === window.self) { if (this.unsafeWindow) { // Greasemonkey. YAYS(unsafeWindow); } else { var node = document.createElement('script'); node.setAttribute('type', 'text/javascript'); node.text = '(' + YAYS.toString() + ')(window);'; document.documentElement.appendChild(node); document.documentElement.removeChild(node); } }