| Index: safari/include.youtube.js |
| =================================================================== |
| --- a/safari/include.youtube.js |
| +++ b/safari/include.youtube.js |
| @@ -22,11 +22,13 @@ |
| if (!ext.backgroundPage.sendMessageSync({type: "get-domain-enabled-state"}).enabled) |
| return; |
| + var badArgumentsRegex = /^((.*_)?(ad|ads|afv|adsense)(_.*)?|(ad3|st)_module|prerolls|interstitial|infringe|iv_cta_url)$/; |
| + |
| function rewriteFlashvars(flashvars) |
| { |
| var pairs = flashvars.split("&"); |
| for (var i = 0; i < pairs.length; i++) |
| - if (/^((ad|afv|adsense)(_.*)?|(ad3|st)_module|prerolls|interstitial|infringe|iv_cta_url)=/.test(pairs[i])) |
| + if (badArgumentsRegex.test(pairs[i].split("=")[0])) |
| pairs.splice(i--, 1); |
| return pairs.join("&"); |
| } |
| @@ -66,21 +68,94 @@ |
| player.parentNode.replaceChild(newPlayer, player); |
| } |
| + function runInPage(fn, arg) |
| + { |
| + var script = document.createElement("script"); |
| + script.type = "application/javascript"; |
| + script.async = false; |
| + script.textContent = "(" + fn + ")(" + arg + ");"; |
| + document.documentElement.appendChild(script); |
| + document.documentElement.removeChild(script); |
| + } |
| + |
| document.addEventListener("beforeload", function(event) |
| { |
| if ((event.target.localName == "object" || event.target.localName == "embed") && /:\/\/[^\/]*\.ytimg\.com\//.test(event.url)) |
| patchPlayer(event.target); |
| }, true); |
| - // if history.pushState is available, YouTube uses the history API |
| - // when navigation from one video to another, and tells the flash |
| - // player with JavaScript which video and which ads to show next, |
| - // bypassing our flashvars rewrite code. So we disable |
| - // history.pushState before YouTube's JavaScript runs. |
| - var script = document.createElement("script"); |
| - script.type = "application/javascript"; |
| - script.async = false; |
| - script.textContent = "History.prototype.pushState = undefined;"; |
| - document.documentElement.appendChild(script); |
| - document.documentElement.removeChild(script); |
| + runInPage(function(badArgumentsRegex) |
| + { |
| + // if history.pushState is available, YouTube uses the history API |
| + // when navigation from one video to another, and tells the flash |
| + // player with JavaScript which video and which ads to show next, |
| + // bypassing our flashvars rewrite code. So we disable |
| + // history.pushState before YouTube's JavaScript runs. |
| + History.prototype.pushState = undefined; |
| + |
| + // the HMLT5 player is configured via ytplayer.config.args. We have |
|
Wladimir Palant
2014/08/28 16:43:38
Nit: "The HTML5 player" (properly capitalized and
Sebastian Noack
2014/08/29 14:51:24
Done.
|
| + // to make sure that ad-related arguments are ignored as they are set. |
| + var ytplayer = undefined; |
|
Wladimir Palant
2014/08/28 16:43:38
What if ytplayer variable was set already? I'd rec
Sebastian Noack
2014/08/29 14:51:24
This hack only works anyway, if our JavaScript run
|
| + Object.defineProperty(window, "ytplayer", |
| + { |
| + get: function() |
| + { |
| + return ytplayer; |
| + }, |
| + set: function(rawYtplayer) |
| + { |
| + if (typeof rawYtplayer != "object") |
|
Wladimir Palant
2014/08/28 16:43:38
console.log(typeof null);
This should be:
if (
Sebastian Noack
2014/08/29 14:51:24
Thanks for reminding me that null is an object. I'
|
| + { |
| + ytplayer = rawYtplayer; |
| + return; |
| + } |
| + |
| + var config = undefined; |
|
Wladimir Palant
2014/08/28 16:43:38
Here as well: what if the config property is alrea
Sebastian Noack
2014/08/29 14:51:24
This was handled by the loop below:
for (var pr
|
| + ytplayer = { |
| + get config() |
| + { |
| + return config; |
| + }, |
| + set config(rawConfig) |
| + { |
| + if (typeof rawConfig != "object") |
|
Wladimir Palant
2014/08/28 16:43:38
Here as well: please consider the possibility that
Sebastian Noack
2014/08/29 14:51:24
Done.
|
| + { |
| + config = rawConfig; |
| + return; |
| + } |
| + |
| + var args = undefined; |
| + config = { |
| + get args() |
| + { |
| + return args; |
| + }, |
| + set args(rawArgs) |
| + { |
| + if (typeof rawArgs != "object") |
| + { |
| + args = rawArgs; |
| + return; |
| + } |
| + |
| + args = {}; |
| + for (var arg in rawArgs) |
| + { |
| + if (!badArgumentsRegex.test(arg)) |
| + args[arg] = rawArgs[arg]; |
| + } |
| + } |
| + }; |
|
Wladimir Palant
2014/08/28 16:43:38
There is a recurring pattern here: you wrap object
Sebastian Noack
2014/08/29 14:51:24
Of course, I already considered that. However, the
|
| + |
| + for (var prop in rawConfig) |
| + config[prop] = rawConfig[prop]; |
| + } |
| + }; |
| + |
| + for (var prop in rawYtplayer) |
| + ytplayer[prop] = rawYtplayer[prop]; |
|
Wladimir Palant
2014/08/28 16:43:38
I think that inheriting from rawYtplayer would be
Sebastian Noack
2014/08/29 14:51:24
Done. However, I'm not convinced that it is a bett
|
| + }, |
| + configurable: true |
| + }); |
| + }, badArgumentsRegex); |
| })(); |