Add AutoplayStopper v1.8.7
This commit is contained in:
parent
f5620cbc84
commit
69eb447c28
5
AutoplayStopper/README.txt
Normal file
5
AutoplayStopper/README.txt
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
This is an extracted copy of https://chrome.google.com/webstore/detail/autoplaystopper/ejddcgojdblidajhngkogefpkknnebdh version 1.8.7. I then loaded it into Chrome as an unpacked extension.
|
||||||
|
This was done to bypass auto extension updates that may eventually be injected with malware.
|
||||||
|
The extraction was done using the "Chrome extension source viewer" addon (https://github.com/Rob--W/crxviewer).
|
||||||
|
|
||||||
|
I reviewed the code for malicious changes and made other small tweaks.
|
201
AutoplayStopper/_locales/en/messages.json
Normal file
201
AutoplayStopper/_locales/en/messages.json
Normal file
|
@ -0,0 +1,201 @@
|
||||||
|
{
|
||||||
|
"extensionDescription": {
|
||||||
|
"message": "Stops video autoplay gracefully."
|
||||||
|
},
|
||||||
|
|
||||||
|
"allowSite": {
|
||||||
|
"message": "Allow autoplay for %S"
|
||||||
|
},
|
||||||
|
|
||||||
|
"allowSession": {
|
||||||
|
"message": "Allow session autoplay for %S"
|
||||||
|
},
|
||||||
|
|
||||||
|
"disableSite": {
|
||||||
|
"message": "Block flash detection for %S"
|
||||||
|
},
|
||||||
|
|
||||||
|
"allowAll": {
|
||||||
|
"message": "Disable everywhere"
|
||||||
|
},
|
||||||
|
|
||||||
|
"settings": {
|
||||||
|
"message": "Settings"
|
||||||
|
},
|
||||||
|
|
||||||
|
"version": {
|
||||||
|
"message": "Version"
|
||||||
|
},
|
||||||
|
|
||||||
|
"autoplayTabLabel": {
|
||||||
|
"message": "Autoplay"
|
||||||
|
},
|
||||||
|
|
||||||
|
"permissions": {
|
||||||
|
"message": "Websites Permissions:"
|
||||||
|
},
|
||||||
|
|
||||||
|
"manageExceptions": {
|
||||||
|
"message": "Exceptions..."
|
||||||
|
},
|
||||||
|
|
||||||
|
"defaultMode": {
|
||||||
|
"message": "Default Mode:"
|
||||||
|
},
|
||||||
|
|
||||||
|
"disableAutoplay": {
|
||||||
|
"message": "Block Autoplay"
|
||||||
|
},
|
||||||
|
|
||||||
|
"allowAutoplay": {
|
||||||
|
"message": "Allow Autoplay"
|
||||||
|
},
|
||||||
|
|
||||||
|
"flashTabLabel": {
|
||||||
|
"message": "Flash"
|
||||||
|
},
|
||||||
|
|
||||||
|
"allowFlash": {
|
||||||
|
"message": "Allow Detection"
|
||||||
|
},
|
||||||
|
|
||||||
|
"disableFlash": {
|
||||||
|
"message": "Block Detection"
|
||||||
|
},
|
||||||
|
|
||||||
|
"autoplayExceptions": {
|
||||||
|
"message": "Autoplay exceptions"
|
||||||
|
},
|
||||||
|
|
||||||
|
"flashExceptions": {
|
||||||
|
"message": "Flash detection exceptions"
|
||||||
|
},
|
||||||
|
|
||||||
|
"exceptionHostnameHeader": {
|
||||||
|
"message": "Hostname"
|
||||||
|
},
|
||||||
|
|
||||||
|
"exceptionBehaviorHeader": {
|
||||||
|
"message": "Behavior"
|
||||||
|
},
|
||||||
|
|
||||||
|
"exceptionAllow": {
|
||||||
|
"message": "Allow"
|
||||||
|
},
|
||||||
|
|
||||||
|
"exceptionBlock": {
|
||||||
|
"message": "Block"
|
||||||
|
},
|
||||||
|
|
||||||
|
"exceptionPrompt": {
|
||||||
|
"message": "Block (strict)"
|
||||||
|
},
|
||||||
|
|
||||||
|
"exceptionSession": {
|
||||||
|
"message": "Session"
|
||||||
|
},
|
||||||
|
|
||||||
|
"exceptionUndefined": {
|
||||||
|
"message": "Undefined"
|
||||||
|
},
|
||||||
|
|
||||||
|
"apply": {
|
||||||
|
"message": "Apply"
|
||||||
|
},
|
||||||
|
|
||||||
|
"removeAll": {
|
||||||
|
"message": "Remove All"
|
||||||
|
},
|
||||||
|
|
||||||
|
"done": {
|
||||||
|
"message": "Done"
|
||||||
|
},
|
||||||
|
|
||||||
|
"play": {
|
||||||
|
"message": "Play"
|
||||||
|
},
|
||||||
|
|
||||||
|
"pause": {
|
||||||
|
"message": "Pause"
|
||||||
|
},
|
||||||
|
|
||||||
|
"selector": {
|
||||||
|
"message": "Selector"
|
||||||
|
},
|
||||||
|
|
||||||
|
"text": {
|
||||||
|
"message": "Text"
|
||||||
|
},
|
||||||
|
|
||||||
|
"reset": {
|
||||||
|
"message": "Reset"
|
||||||
|
},
|
||||||
|
|
||||||
|
"script": {
|
||||||
|
"message": "Script"
|
||||||
|
},
|
||||||
|
|
||||||
|
"userScript": {
|
||||||
|
"message": "User Script"
|
||||||
|
},
|
||||||
|
|
||||||
|
"path": {
|
||||||
|
"message": "Path"
|
||||||
|
},
|
||||||
|
|
||||||
|
"actions": {
|
||||||
|
"message": "Actions"
|
||||||
|
},
|
||||||
|
|
||||||
|
"edit": {
|
||||||
|
"message": "Edit"
|
||||||
|
},
|
||||||
|
|
||||||
|
"load": {
|
||||||
|
"message": "Load"
|
||||||
|
},
|
||||||
|
|
||||||
|
"export": {
|
||||||
|
"message": "Export..."
|
||||||
|
},
|
||||||
|
|
||||||
|
"setPath": {
|
||||||
|
"message": "Set Path"
|
||||||
|
},
|
||||||
|
|
||||||
|
"selectFile": {
|
||||||
|
"message": "File..."
|
||||||
|
},
|
||||||
|
|
||||||
|
"advanced": {
|
||||||
|
"message": "Advanced"
|
||||||
|
},
|
||||||
|
|
||||||
|
"devtools": {
|
||||||
|
"message": "Devtools Panel"
|
||||||
|
},
|
||||||
|
|
||||||
|
"overwrite": {
|
||||||
|
"message": "Disable overwrite on update"
|
||||||
|
},
|
||||||
|
|
||||||
|
"debug": {
|
||||||
|
"message": "Log debug info"
|
||||||
|
},
|
||||||
|
|
||||||
|
"loadError": {
|
||||||
|
"message": "Load script error: "
|
||||||
|
},
|
||||||
|
|
||||||
|
"allowFileUrls": {
|
||||||
|
"message": "Error: Cannot use user script without permission to access file URLs."
|
||||||
|
},
|
||||||
|
|
||||||
|
"fileNotFound": {
|
||||||
|
"message": "AutoplayStopper - error: File not found.\n Check that the script folder is in the workspace and verify the path."
|
||||||
|
},
|
||||||
|
|
||||||
|
"evalError": {
|
||||||
|
"message": "AutoplayStopper - exception: "
|
||||||
|
}
|
||||||
|
}
|
186
AutoplayStopper/background.js
Normal file
186
AutoplayStopper/background.js
Normal file
|
@ -0,0 +1,186 @@
|
||||||
|
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var Background = new function() {
|
||||||
|
|
||||||
|
console.log("loading background.js ...");
|
||||||
|
|
||||||
|
const {js, css} = chrome.runtime.getManifest().content_scripts[0];
|
||||||
|
const badgeColor = "#646464";
|
||||||
|
const playMsg = chrome.i18n.getMessage("play");
|
||||||
|
const pauseMsg = chrome.i18n.getMessage("pause");
|
||||||
|
const sArea = "local";
|
||||||
|
const permsAutoplayKey = Permissions.key("autoplay");
|
||||||
|
const permsAutoplayDefaultKey = Permissions.defaultKey("autoplay");
|
||||||
|
const permsFlashKey = Permissions.key("flash");
|
||||||
|
const permsFlashDefaultKey = Permissions.defaultKey("flash");
|
||||||
|
const sKeys = ["disabled", "debug", permsFlashKey, permsFlashDefaultKey, permsAutoplayKey,
|
||||||
|
permsAutoplayDefaultKey, "selector-css", "uhandler", "handler", "disableOverwrite", "devtools"];
|
||||||
|
|
||||||
|
var selector = null;
|
||||||
|
var permissions = null;
|
||||||
|
var permsAutoplayCache = {};
|
||||||
|
var permsFlashCache = {};
|
||||||
|
var tabs = { get(id) { return this[id] || (this[id] = {id, count: 0})}};
|
||||||
|
var menuitem = null;
|
||||||
|
|
||||||
|
var storage = new Storage(sKeys, sArea);
|
||||||
|
storage.addChangeListener(function(changes){
|
||||||
|
if (changes.indexOf("selector-css") != -1) selector = storage.data["selector-css"].split("{")[0];
|
||||||
|
if (changes.indexOf(permsAutoplayKey) != -1 || changes.indexOf(permsAutoplayDefaultKey) != -1) permsAutoplayCache = {};
|
||||||
|
if (changes.indexOf(permsFlashKey) != -1 || changes.indexOf(permsFlashDefaultKey) != -1) permsFlashCache = {};
|
||||||
|
if (changes.indexOf("disabled") != -1) updateIcon();
|
||||||
|
});
|
||||||
|
|
||||||
|
storage.ready.then(function() {
|
||||||
|
console.log("storage.ready");
|
||||||
|
updateIcon();
|
||||||
|
permissions = new Permissions(storage.data);
|
||||||
|
if (permissions.default("autoplay") === undefined) permissions.setDefault("autoplay", Permission.DENY_ACTION);
|
||||||
|
if (permissions.default("flash") === undefined) permissions.setDefault("flash", Permission.ALLOW_ACTION);
|
||||||
|
chrome.webNavigation.onCommitted.addListener(onCommitted, {url: [{schemes: ["http", "https", "about"]}]});
|
||||||
|
if (!storage.data.uhandler)
|
||||||
|
loadScript("uhandler", chrome.extension.getURL("script/userhandler.js"));
|
||||||
|
if (!storage.data.handler)
|
||||||
|
loadScript("handler", chrome.extension.getURL("script/handler.js"));
|
||||||
|
if (!storage.data["selector-css"])
|
||||||
|
loadScript("selector-css", chrome.extension.getURL("script/selector.css"));
|
||||||
|
selector = storage.data["selector-css"].split("{")[0];
|
||||||
|
js.forEach(function(a) { execScript(a, null, 0)});
|
||||||
|
});
|
||||||
|
|
||||||
|
chrome.browserAction.setBadgeBackgroundColor({color: badgeColor});
|
||||||
|
chrome.tabs.onRemoved.addListener(function(tabid){ delete tabs[tabid]; });
|
||||||
|
chrome.tabs.onActivated.addListener(resetMenuitem);
|
||||||
|
chrome.runtime.onMessage.addListener(handleMessage);
|
||||||
|
chrome.runtime.onInstalled.addListener(function(details) {
|
||||||
|
if (details.reason == "update" && !storage.data.disableOverwrite) {
|
||||||
|
loadScript("handler", chrome.extension.getURL("script/handler.js"));
|
||||||
|
loadScript("selector-css", chrome.extension.getURL("script/selector.css"));
|
||||||
|
selector = storage.data["selector-css"].split("{")[0];
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return {
|
||||||
|
get i18n() { return i18n; },
|
||||||
|
get storage() { return storage; },
|
||||||
|
get permissions() { return permissions; },
|
||||||
|
openOptionsPage: function openOptionsPage(hash) {
|
||||||
|
chrome.runtime.openOptionsPage();
|
||||||
|
if (hash) addEventListener("message", function(e) {
|
||||||
|
if (location.origin == e.origin && e.data == "optionsPageActive")
|
||||||
|
e.source.location.hash = hash;
|
||||||
|
}, { once: true});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
function resetMenuitem(){ if (menuitem) menuitem = chrome.contextMenus.remove("play-menuitem"); }
|
||||||
|
|
||||||
|
function updateIcon() { chrome.browserAction.setIcon({ path: "/icons/" + (storage.data.disabled ? "icon32d.png" : "icon32.png")})};
|
||||||
|
|
||||||
|
function handleMessage(request, sender, sendResponse)
|
||||||
|
{
|
||||||
|
if (!storage.data.handler) {
|
||||||
|
console.log("handleMessage delayed -", request, sender);
|
||||||
|
storage.ready.then(function() { handleMessage(request, sender, sendResponse)});
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
if (request == "permission") {
|
||||||
|
var data = {debug: !!storage.data.debug};
|
||||||
|
var allow = true, host = sender.tab && new URL(sender.tab.url).origin;
|
||||||
|
if (host && !storage.data.disabled) {
|
||||||
|
var permission = permsAutoplayCache[host] ||
|
||||||
|
(permsAutoplayCache[host] = permissions.testPermission("autoplay", sender.tab.url));
|
||||||
|
allow = permission == Permission.ALLOW_ACTION ||
|
||||||
|
(permission == Permission.ACCESS_SESSION && tabs.get(sender.tab.id).last == host);
|
||||||
|
data.strict = permission == Permission.PROMPT_ACTION;
|
||||||
|
}
|
||||||
|
var msg = {msg: "permission", data, allow, selector, handler: storage.data.handler, uhandler: storage.data.uhandler};
|
||||||
|
// console.log("background send:", msg, sender.url, sender.tab.url);
|
||||||
|
sendResponse(msg);
|
||||||
|
}
|
||||||
|
if (request == "count")
|
||||||
|
chrome.browserAction.setBadgeText({text: String(++tabs.get(sender.tab.id).count), tabId: sender.tab.id});
|
||||||
|
if (request.msg == "contextmenu") {
|
||||||
|
resetMenuitem();
|
||||||
|
if (request.media)
|
||||||
|
menuitem = chrome.contextMenus.create({ id: "play-menuitem", title: request.paused ? playMsg : pauseMsg,
|
||||||
|
contexts: ["all"], onclick: function onClick(info, tab) { if (info.menuItemId == "play-menuitem") sendResponse();}
|
||||||
|
})
|
||||||
|
return !!request.media;
|
||||||
|
}
|
||||||
|
if (request.msg == "select") {
|
||||||
|
fetch(request.url, {cache: "no-store"}).then(function(res) {
|
||||||
|
res.blob().then(function(file){ sendResponse({url: URL.createObjectURL(file), name: file.name}); });
|
||||||
|
});
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
if (request.msg == "load") {
|
||||||
|
var res = loadScript(request.script, request.file);
|
||||||
|
sendResponse(res);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
function onCommitted(details)
|
||||||
|
{
|
||||||
|
var allow = true, url = new URL(details.url);
|
||||||
|
if (url.protocol.search("^chrome") != -1) return;
|
||||||
|
|
||||||
|
var tab = tabs.get(details.tabId);
|
||||||
|
if (details.frameId == 0) {
|
||||||
|
tab.last = tab.host;
|
||||||
|
tab.host = url.origin;
|
||||||
|
tab.url = details.url;
|
||||||
|
tab.count = 0;
|
||||||
|
chrome.browserAction.setBadgeText({text:'', tabId: tab.id});
|
||||||
|
chrome.tabs.query({ active: true, currentWindow: true}, function(tabs) {
|
||||||
|
if (tabs[0] && tabs[0].id == tab.id) resetMenuitem();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
if (!storage.data.disabled && tab.url) {
|
||||||
|
var permission = permsFlashCache[tab.host] ||
|
||||||
|
(permsFlashCache[tab.host] = permissions.testPermission("flash", tab.url));
|
||||||
|
if (permission == Permission.DENY_ACTION)
|
||||||
|
execScript("content/navigator.js", details.tabId, details.frameId);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
function execScript(file, tabId, frameId) {
|
||||||
|
chrome.tabs.executeScript(tabId, {file, frameId, matchAboutBlank: true, runAt: "document_start"}, function() {
|
||||||
|
if (chrome.runtime.lastError && storage.data.debug)
|
||||||
|
console.error("chrome.tabs.executeScript - error", chrome.runtime.lastError, {tabId, frameId, file});
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
function insertCSS(file, tabId, frameId) {
|
||||||
|
chrome.tabs.insertCSS(tabId, {file, frameId, matchAboutBlank: true, runAt: "document_start"}, function() {
|
||||||
|
if (chrome.runtime.lastError && storage.data.debug)
|
||||||
|
console.error("chrome.tabs.insertCSS - error", chrome.runtime.lastError, {tabId, frameId, file});
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
function loadScript(script, url){
|
||||||
|
try {
|
||||||
|
var request = new XMLHttpRequest();
|
||||||
|
request.overrideMimeType("text/plain");
|
||||||
|
request.open("GET", url, false);
|
||||||
|
request.setRequestHeader("pragma", "no-cache");
|
||||||
|
request.send();
|
||||||
|
if (request.statusText == "Not Found" || !request.responseURL)
|
||||||
|
throw new DOMException(`Failed to load '${url}'.`, "NetworkError");
|
||||||
|
|
||||||
|
storage.data[script] = request.responseText;
|
||||||
|
storage.commit([script]);
|
||||||
|
} catch (e) {
|
||||||
|
chrome.extension.isAllowedFileSchemeAccess(function(allowed) {
|
||||||
|
if(!allowed && url.search(/^blob:/) == -1)
|
||||||
|
return alert(chrome.i18n.getMessage("allowFileUrls"));
|
||||||
|
alert(`${chrome.i18n.getMessage("loadError")} ${e.name}\n ${e.message}`);
|
||||||
|
});
|
||||||
|
return e;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
//</>
|
310
AutoplayStopper/content/content.js
Normal file
310
AutoplayStopper/content/content.js
Normal file
|
@ -0,0 +1,310 @@
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
const ContentScript = new function() {
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
const start = Date.now();
|
||||||
|
const any = (window.wrappedJSObject ? "-moz" : "-webkit") + "-any";
|
||||||
|
const frameSelector = `iframe:${any}(:not([src]), [src^='javascript:' i], [src^='about:blank' i]):not([flashstopped])`;
|
||||||
|
|
||||||
|
var data = {debug: false};
|
||||||
|
const _trace = function TRACE(format, ...etc) {
|
||||||
|
if (!data.debug) return TRACE._noop || (TRACE._noop = function(){});
|
||||||
|
return console.log.bind(console, "### " + format + " ###", ...etc);
|
||||||
|
};
|
||||||
|
|
||||||
|
var selector = null;
|
||||||
|
var handler = null; //loadURL(chrome.extension.getURL("script/handler.js"));
|
||||||
|
var href = frameElement !== null ? frameElement.src || "about:blank" : location.href;
|
||||||
|
var ret = 0;
|
||||||
|
|
||||||
|
_trace("ready(%s) content.js - iframe: %s href: %s loc: %s fs: %s")(start - Date.now(),
|
||||||
|
!!frameElement && frameElement.id, href == location.href || href, location.href, window.flashstopped);
|
||||||
|
|
||||||
|
if (window.flashstopped) return;
|
||||||
|
window.flashstopped = true;
|
||||||
|
|
||||||
|
chrome.runtime.sendMessage("permission", handlePermission);
|
||||||
|
function handlePermission(response) {
|
||||||
|
if (chrome.runtime.lastError && ret <= 3) {
|
||||||
|
console.error(`handlePermission - error ret: ${ret} msg: ${chrome.runtime.lastError.message}`);
|
||||||
|
window.setTimeout(function() { chrome.runtime.sendMessage("permission", handlePermission)}, ret++ * 250);
|
||||||
|
};
|
||||||
|
if (!response) return;
|
||||||
|
data = response.data;
|
||||||
|
selector = response.selector;
|
||||||
|
handler = `${response.uhandler}\n\n${response.handler}`;
|
||||||
|
if (!response.allow) load(window);
|
||||||
|
addEventListener("message", function(e) {
|
||||||
|
if (e.data && e.data.id == "userinput") e.stopImmediatePropagation();
|
||||||
|
}, true);
|
||||||
|
};
|
||||||
|
|
||||||
|
if (parent && parent != window)
|
||||||
|
parent.postMessage({id: "userinput"}, "*");
|
||||||
|
|
||||||
|
function load({window, frameElement, location})
|
||||||
|
{
|
||||||
|
var href = frameElement !== null ? frameElement.src || "about:blank" : location.href;
|
||||||
|
_trace("loading(%s) content.js - iframe: %s href: %s loc: %s")(start - Date.now(),
|
||||||
|
!!frameElement && frameElement.id, href == location.href || href, location.href);
|
||||||
|
|
||||||
|
var match = 0, query = 0;
|
||||||
|
var injected = false;
|
||||||
|
var body = window.document.body;
|
||||||
|
var userInput = null;
|
||||||
|
|
||||||
|
var n = handleDocument();
|
||||||
|
_trace("body: %s n: %s")(!!body, n);
|
||||||
|
|
||||||
|
var docObserver = new MutationObserver(function(mutations) {
|
||||||
|
|
||||||
|
_trace("!!! docObserver - body: %s neb: %s !!!")(!!body, body != window.document.body);
|
||||||
|
if (window.document.body && body != window.document.body) {
|
||||||
|
if (body) observer.observe(window.document.body, { subtree: true, childList: true});
|
||||||
|
body = window.document.body;
|
||||||
|
requestAnimationFrame(function() {
|
||||||
|
var n = handleDocument();
|
||||||
|
_trace(">>>>>>> requestAnimationFrame n: %s <<<<<<<")(n);
|
||||||
|
});
|
||||||
|
window.setTimeout(function() {
|
||||||
|
var n = handleDocument();
|
||||||
|
_trace(">>>>>>> setting body observer - n: %s <<<<<<<<")(n);
|
||||||
|
}, 500);
|
||||||
|
var n = handleDocument();
|
||||||
|
_trace("!!! docObserver - out n: %s !!!")(n);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
var observer = new MutationObserver(function(mutations) {
|
||||||
|
|
||||||
|
for (var i = 0; i < mutations.length; i++)
|
||||||
|
for (var j = 0; j < mutations[i].addedNodes.length; j++) {
|
||||||
|
var addedNode = mutations[i].addedNodes[j];
|
||||||
|
if (addedNode.nodeType === 1){
|
||||||
|
match++;
|
||||||
|
if (addedNode.children.length) query++;
|
||||||
|
if (handleNodesDeep(addedNode, true)) {
|
||||||
|
_trace(">>>>>>> MutationObserver: tag: %s id: %s <<<<<<<")(addedNode.localName, addedNode.id);
|
||||||
|
}
|
||||||
|
var frames = addedNode.matches(frameSelector) ? [addedNode] : [];
|
||||||
|
if (frames.push(...addedNode.querySelectorAll(frameSelector)))
|
||||||
|
handleFrames(frames);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
if (window.document.body)
|
||||||
|
observer.observe((href.search("^https?:") == -1) ? window.document : window.document.body
|
||||||
|
, { subtree: true, childList: true});
|
||||||
|
else
|
||||||
|
docObserver.observe(window.document.documentElement, { childList: true});
|
||||||
|
|
||||||
|
// workaround for chrome no events on blank iframe ...
|
||||||
|
setTimeout( function() {
|
||||||
|
window.addEventListener("mousedown", onContextMenu, true);
|
||||||
|
["mousedown","click"].forEach((a) => window.addEventListener(a, function(e) {
|
||||||
|
if (e.isTrusted && e.button == 0) {
|
||||||
|
userInput = {e: e, time: Date.now()};
|
||||||
|
if (injected) dispatchUserInput(userInput);
|
||||||
|
}
|
||||||
|
}, true));
|
||||||
|
if (data.debug) chrome.storage.local.get(["selector-css"], function(data) {
|
||||||
|
window.document.head.appendChild(document.createElement("style"))
|
||||||
|
.textContent = data["selector-css"];
|
||||||
|
});
|
||||||
|
}, 1000);
|
||||||
|
|
||||||
|
window.addEventListener("DOMContentLoaded", function() {
|
||||||
|
observer.observe(window.document.body, { subtree: true, childList: true});
|
||||||
|
var n = handleDocument();
|
||||||
|
_trace(">>>>>>> DOMContentLoaded n: %s match: %s query: %s iframe: %s loc: %s <<<<<<<")
|
||||||
|
(n , match, query, !!frameElement && frameElement.id, location.href);
|
||||||
|
}, true);
|
||||||
|
|
||||||
|
window.addEventListener("message", function(e) {
|
||||||
|
if (!e.data || e.data.id != "userinput") return;
|
||||||
|
if (e.source == window.parent) {
|
||||||
|
_trace("@@@ handleMessage(userinput) @@@@")();
|
||||||
|
var si = new MouseEvent("siminput", Object.assign({view: window}, e.data));
|
||||||
|
userInput = {e: Object.defineProperty(si, "target", {value: window}), time: e.data.time};
|
||||||
|
if (injected) dispatchUserInput(userInput);
|
||||||
|
}
|
||||||
|
if (e.source && e.source.parent == window && userInput) {
|
||||||
|
var iframe, iframes = [...window.document.querySelectorAll("iframe")];
|
||||||
|
if (userInput.e.target.shadowRoot) iframes.push(...userInput.e.target.shadowRoot.querySelectorAll("iframe"));
|
||||||
|
if (iframe = iframes.find((a) => a.contentWindow == e.source)) {
|
||||||
|
var ue = userInput.e, time = userInput.time, r = iframe.getBoundingClientRect();
|
||||||
|
var data = {id: "userinput", clientX: ue.pageX - r.left - scrollX, clientY: ue.pageY - r.top - scrollY, time};
|
||||||
|
e.source.postMessage(data, "*");
|
||||||
|
_trace("@@@ iframe.postMessage(userinput) @@@")();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, true);
|
||||||
|
|
||||||
|
function handleDocument()
|
||||||
|
{
|
||||||
|
var nodes = handleNodesDeep(window.document.body || window.document.documentElement);
|
||||||
|
var frames = (window.document.body || window.document.documentElement).querySelectorAll(frameSelector);
|
||||||
|
if (frames.length) handleFrames(frames);
|
||||||
|
return nodes;
|
||||||
|
};
|
||||||
|
|
||||||
|
function handleNodes(nodes, delayed)
|
||||||
|
{
|
||||||
|
if (!injected) {
|
||||||
|
if (href.search("^https?:") == -1 && !delayed)
|
||||||
|
return window.setTimeout(function(){ handleNodes(nodes, true)});
|
||||||
|
injected = loadScript();
|
||||||
|
if (userInput) dispatchUserInput(userInput);
|
||||||
|
}
|
||||||
|
nodes.forEach(function(node) {
|
||||||
|
node.setAttribute("flashstopped", true);
|
||||||
|
try { node.parentNode.setAttribute("flashstopped_p", true)} catch(e){};
|
||||||
|
if (!node.flashstopped) {
|
||||||
|
node.dispatchEvent(new Event("flashstop:bind", {bubbles: true, composed: true}));
|
||||||
|
chrome.runtime.sendMessage("count");
|
||||||
|
_trace("match: %s query: %s")(match, query);
|
||||||
|
}
|
||||||
|
node.flashstopped = true;
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
function handleFrames(frames)
|
||||||
|
{
|
||||||
|
if (!window.wrappedJSObject) frames.forEach(function(frame) {
|
||||||
|
if (!frame.contentWindow) return;
|
||||||
|
_trace("!!! iframe - id: %s src: %s fs: %s !!!")(frame.id, frame.src, frame.contentWindow.flashstopped);
|
||||||
|
if (!frame.contentWindow.flashstopped) {
|
||||||
|
frame.setAttribute("flashstopped", true);
|
||||||
|
frame.contentWindow.flashstopped = true;
|
||||||
|
frame.contentWindow.Function('parent.postMessage({id: "userinput"}, "*")')();
|
||||||
|
window.setTimeout(function(){ load(frame.contentWindow)});
|
||||||
|
};
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
function handleNodesDeep(node, match)
|
||||||
|
{
|
||||||
|
var nodes = match && (node.matches(selector) || node.shadowRoot) ? [node] : [];
|
||||||
|
var count = nodes.push(...node.querySelectorAll(selector));
|
||||||
|
for (var el of nodes.filter((a) => a.shadowRoot)) {
|
||||||
|
nodes.splice(nodes.indexOf(el), 1)[0].setAttribute("flashstopped", true);
|
||||||
|
count += handleNodesDeep(el.shadowRoot) - 1;
|
||||||
|
observer.observe(el.shadowRoot, { subtree: true, childList: true});
|
||||||
|
}
|
||||||
|
if (nodes.length) handleNodes(nodes);
|
||||||
|
return count;
|
||||||
|
};
|
||||||
|
|
||||||
|
function loadScript()
|
||||||
|
{
|
||||||
|
var code = `(function(data){ ${_trace} ${Handlers} ${handler} (${init})()})(${JSON.stringify(data)})`;
|
||||||
|
var script = window.document.createElement("script");
|
||||||
|
script.textContent = code;
|
||||||
|
if (window.wrappedJSObject)
|
||||||
|
script.textContent += "\ndocument.currentScript.dispatchEvent(new Event('load'));";
|
||||||
|
script.onload = function() { this.executed = true};
|
||||||
|
(window.document.head || window.document.documentElement).appendChild(script);
|
||||||
|
script.remove();
|
||||||
|
|
||||||
|
if (window.wrappedJSObject && !script.executed) // fff
|
||||||
|
try { window.eval(code); } catch(e) {};
|
||||||
|
|
||||||
|
return true;
|
||||||
|
};
|
||||||
|
// initilaizer & adapter for the injection code...
|
||||||
|
function init()
|
||||||
|
{
|
||||||
|
var handlingUserInput = false
|
||||||
|
var lastUserInput = null;
|
||||||
|
var handlers = new Handlers(TRACE);
|
||||||
|
function isHandlingUserInput() { return handlingUserInput = handlingUserInput && Date.now() - lastUserInput.time < 100 };
|
||||||
|
|
||||||
|
registerUserHandlers(handlers, TRACE, isHandlingUserInput, () => TRACE("setHandlingUserInput")(), () => lastUserInput, data);
|
||||||
|
registerHandlers(handlers, TRACE, isHandlingUserInput, () => TRACE("setHandlingUserInput")(), () => lastUserInput, data);
|
||||||
|
|
||||||
|
window.addEventListener("userinput", function(e) {
|
||||||
|
handlingUserInput = true;
|
||||||
|
setTimeout(function() { handlingUserInput = false;});
|
||||||
|
lastUserInput = {e: Object.assign({target: e.target}, e.detail.e), time: e.detail.time};
|
||||||
|
}, true);
|
||||||
|
window.addEventListener("flashstop:bind", function onbind(e){
|
||||||
|
var aElement = e.target, path = e.composedPath();
|
||||||
|
if (path && path[0] != aElement) aElement = path[0];
|
||||||
|
TRACE("onElementBinding - tag: %s id: %s loc: %s")(aElement.localName, aElement.id, window.location.href);
|
||||||
|
window.wrappedJSObject = window;
|
||||||
|
try {
|
||||||
|
handlers.apply(wrapper(aElement));
|
||||||
|
} catch(e) {
|
||||||
|
aElement.addEventListener("playing", () => {
|
||||||
|
if (!lastUserInput || Date.now() - lastUserInput.time > 1500) aElement.pause();
|
||||||
|
})};
|
||||||
|
delete window.wrappedJSObject;
|
||||||
|
}, true);
|
||||||
|
|
||||||
|
function wrapper(node)
|
||||||
|
{
|
||||||
|
var proxy = new Proxy(node, { get: function(target, prop, receiver) {
|
||||||
|
var proto = Object.getPrototypeOf(node);
|
||||||
|
var res = Reflect.get(proto, prop, node);
|
||||||
|
return typeof res == "function" && res.bind ? res.bind(node) : res;
|
||||||
|
}, set: function(target, prop, value, receiver) {
|
||||||
|
var proto = Object.getPrototypeOf(node);
|
||||||
|
return Reflect.set(proto, prop, value, Reflect.has(proto, prop) ? node : receiver);
|
||||||
|
}});
|
||||||
|
return Object.create(proxy, {wrappedJSObject: { value: node}});
|
||||||
|
};
|
||||||
|
|
||||||
|
TRACE("addEventListener - flashstop:bind")();
|
||||||
|
};
|
||||||
|
|
||||||
|
function inRect(x, y, el){ var b = el.getBoundingClientRect(); return (x > b.left && x < b.right && y > b.top && y < b.bottom) ? el : null};
|
||||||
|
function matchDeep(el, x, y, res) {
|
||||||
|
if (!el.shadowRoot) return el.matches("video, audio") ? el : null;
|
||||||
|
for (var n of [...el.shadowRoot.elementsFromPoint(x, y)].filter((a) => a.getRootNode().host == el))
|
||||||
|
if (res = matchDeep(n, x, y)) return res;
|
||||||
|
};
|
||||||
|
function getMediaElement(aWin, aPrev, x, y)
|
||||||
|
{
|
||||||
|
try {aWin.document} catch(e) { _trace("getMediaElement - %s")(e); return };
|
||||||
|
var res, sx = x + aWin.mozInnerScreenX, sy = y + aWin.mozInnerScreenY, round = Math.round;
|
||||||
|
_trace("sc(%s,%s) win(%s,%s) win: %s prev: %s")(round(sx), round(sy), round(x), round(y), aWin.location.host, aPrev && aPrev.location.host);
|
||||||
|
for (var node of aWin.document.elementsFromPoint(x, y))
|
||||||
|
if (res = (node.localName != "iframe" || node.contentWindow == aPrev) ? matchDeep(node, x, y) :
|
||||||
|
getMediaElement(node.contentWindow, null, x - node.getBoundingClientRect().left, y - node.getBoundingClientRect().top))
|
||||||
|
return (res.localName != "iframe" && _trace("getMediaElement - success id: %s")(res.id), res);
|
||||||
|
if (res = [...aWin.document.querySelectorAll("video:not([src^='data:'])")].find((node) => inRect(x, y, node)))
|
||||||
|
return (_trace("getMediaElement - success id: %s")(res.id), res);
|
||||||
|
return (aPrev && aWin != aWin.parent && aWin.frameElement) && getMediaElement(aWin.parent, aWin, x +
|
||||||
|
aWin.frameElement.getBoundingClientRect().left, y + aWin.frameElement.getBoundingClientRect().top);
|
||||||
|
};
|
||||||
|
|
||||||
|
function onContextMenu(e)
|
||||||
|
{ // before
|
||||||
|
if (e.isTrusted && e.button == 2){
|
||||||
|
var media = e.target instanceof HTMLMediaElement ? e.target : getMediaElement(e.view, e.view, e.clientX, e.clientY);
|
||||||
|
var paused = media && media.paused;
|
||||||
|
if (media) {
|
||||||
|
chrome.runtime.sendMessage({ msg: "contextmenu", media: true, paused },
|
||||||
|
function(){ if (!chrome.runtime.lastError) paused ? media.play() : media.pause(); });
|
||||||
|
if (e.ctrlKey)
|
||||||
|
e.stopImmediatePropagation();
|
||||||
|
}
|
||||||
|
else
|
||||||
|
chrome.runtime.sendMessage({ msg: "contextmenu", media: false});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
function dispatchUserInput({e, time, e: {view: window}})
|
||||||
|
{ // fff!
|
||||||
|
with(e) var Obj = window.Object, data = Object.assign(new Obj, {clientX, clientY, pageX, pageY});
|
||||||
|
var target = e.target != window && window.document.contains(e.target) ? e.target : window;
|
||||||
|
target.dispatchEvent(new CustomEvent("userinput", {detail: Object.assign(new Obj,{e: data, time})}));
|
||||||
|
};
|
||||||
|
|
||||||
|
// </>
|
49
AutoplayStopper/content/handler-utils.js
Normal file
49
AutoplayStopper/content/handler-utils.js
Normal file
|
@ -0,0 +1,49 @@
|
||||||
|
|
||||||
|
// handler-utils.js See license.txt for terms of usage and credits
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Handlers - content handlers manager...
|
||||||
|
* ----------------------------------------
|
||||||
|
* @bYO!
|
||||||
|
*/
|
||||||
|
|
||||||
|
function Handlers(TRACE){
|
||||||
|
|
||||||
|
var _handlers = [], _mapRemoved = new Map();
|
||||||
|
return {
|
||||||
|
unload: function unload(){
|
||||||
|
for (var win of _mapRemoved.keys()) {
|
||||||
|
win.removeEventListener("unload", onUnload, false);
|
||||||
|
TRACE("Handlers.unload - remove wnd loc: %s")(win.location);
|
||||||
|
};
|
||||||
|
_handlers = [];
|
||||||
|
_mapRemoved = new Map();
|
||||||
|
},
|
||||||
|
add: function add(handler) { _handlers.push(handler); },
|
||||||
|
remove: function remove(win, handler) {
|
||||||
|
if (!_mapRemoved.has(win)){
|
||||||
|
win.addEventListener("unload", onUnload, false);
|
||||||
|
_mapRemoved.set(win, []);
|
||||||
|
}
|
||||||
|
_mapRemoved.get(win).push(handler);
|
||||||
|
TRACE("Handlers.remove - handler: %s")(handler.name);
|
||||||
|
},
|
||||||
|
apply: function apply(element){
|
||||||
|
var removed = _mapRemoved.get(element.ownerDocument.defaultView);
|
||||||
|
for (var handler, i = 0; i < _handlers.length; i++)
|
||||||
|
if ((handler = _handlers[i]) && (!removed || removed.indexOf(handler) == -1))
|
||||||
|
try {
|
||||||
|
if (handler(element)) break;
|
||||||
|
} catch (e) { TRACE("!!!!!!!!! handler: %s Exception: %s !!!!!!!!!")(handler.name, e); };
|
||||||
|
},
|
||||||
|
get length() { return _handlers.length; }
|
||||||
|
};
|
||||||
|
|
||||||
|
function onUnload(e){
|
||||||
|
var win = e.target.defaultView;
|
||||||
|
TRACE("Handlers.onUnload - remove wnd has: %s loc: %s")(_mapRemoved.has(win), win.location);
|
||||||
|
_mapRemoved.delete(win);
|
||||||
|
}
|
||||||
|
};
|
39
AutoplayStopper/content/navigator.js
Normal file
39
AutoplayStopper/content/navigator.js
Normal file
|
@ -0,0 +1,39 @@
|
||||||
|
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
(function() {
|
||||||
|
|
||||||
|
|
||||||
|
// setTimeout(function() {
|
||||||
|
var script = document.createElement("script");
|
||||||
|
script.textContent = `(${setNavigator})()`;
|
||||||
|
(document.head || document.documentElement || document).appendChild(script);
|
||||||
|
script.remove();
|
||||||
|
// });
|
||||||
|
|
||||||
|
function setNavigator(nav = { plugins: navigator.plugins, mimeTypes: navigator.mimeTypes}) {
|
||||||
|
|
||||||
|
console.log("setNavigator - uri: " + location);
|
||||||
|
|
||||||
|
var plugins = Object.assign(Object.create(PluginArray.prototype), [...nav.plugins]);
|
||||||
|
Object.defineProperties(plugins, { item: {value: (a) => plugins[a]}, namedItem: {value: (a) => plugins[a]},
|
||||||
|
refresh: {value: () => { nav.plugins.refresh(); setNavigator(nav)}},
|
||||||
|
length: { writable: true, value: nav.plugins.length}});
|
||||||
|
for (var a of plugins) { Object.defineProperty(plugins, a.name, {configurable: true, value: a}); };
|
||||||
|
[].splice.call(plugins, [].findIndex.call(plugins, (a) => a.name == "Shockwave Flash"), 1);
|
||||||
|
delete plugins["Shockwave Flash"];
|
||||||
|
Object.defineProperty(navigator, "plugins", {configurable: true, value: plugins});
|
||||||
|
|
||||||
|
var mimetypes = Object.assign(Object.create(MimeTypeArray.prototype), [...nav.mimeTypes]);
|
||||||
|
Object.defineProperties(mimetypes, { item: {value: (a) => mimetypes[a]}, namedItem: {value: (a) => mimetypes[a]},
|
||||||
|
refresh: {value: () => { plugins.refresh()}}, length: { writable: true, value: nav.mimeTypes.length}});
|
||||||
|
for (var a of mimetypes) { Object.defineProperty(mimetypes, a.type, {configurable: true, value: a}); };
|
||||||
|
[].forEach.call(nav.plugins["Shockwave Flash"] || [], ({type}) =>
|
||||||
|
delete mimetypes[[].splice.call(mimetypes, [].findIndex.call(mimetypes, (a) => a.type == type), 1)[0].type]);
|
||||||
|
Object.defineProperty(navigator, "mimeTypes", {configurable: true, value: mimetypes});
|
||||||
|
};
|
||||||
|
})();
|
||||||
|
|
||||||
|
// </>
|
||||||
|
|
144
AutoplayStopper/devtools.js
Normal file
144
AutoplayStopper/devtools.js
Normal file
|
@ -0,0 +1,144 @@
|
||||||
|
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
const DevTools = new function() {
|
||||||
|
|
||||||
|
const sArea = "local";
|
||||||
|
const sKeys = ["debug", "devtools", "codefile", "ucodefile", "codefilename", "ucodefilename", "disableOverwrite", "selector-css", "uhandler", "handler"];
|
||||||
|
|
||||||
|
var storage = new Storage(sKeys, sArea);
|
||||||
|
|
||||||
|
storage.ready.then(function() {
|
||||||
|
|
||||||
|
if (!storage.data.devtools)
|
||||||
|
return;
|
||||||
|
// devtools page - create panel ...
|
||||||
|
if (location.search == "?view=page")
|
||||||
|
return chrome.devtools.panels.create("AutoplayStopper", "/icons/icon48.png", "/skin/devtools.html?view=panel");
|
||||||
|
|
||||||
|
// devtools panel ...
|
||||||
|
var fileinput = Object.assign(document.createElement("input"), {type: "file", hidden: true});
|
||||||
|
document.body.appendChild(fileinput);
|
||||||
|
document.readyState == "complete" ? load() : addEventListener("load", load);
|
||||||
|
|
||||||
|
function load() {
|
||||||
|
|
||||||
|
i18n.process(document);
|
||||||
|
document.getElementById("ver").textContent = chrome.runtime.getManifest().version;
|
||||||
|
setCheckbox("overwrite", "disableOverwrite");
|
||||||
|
setCheckbox("debug");
|
||||||
|
setSelector();
|
||||||
|
setup("");
|
||||||
|
setup("u");
|
||||||
|
};
|
||||||
|
|
||||||
|
function setSelector()
|
||||||
|
{
|
||||||
|
var selector = document.getElementById("selector");
|
||||||
|
var apply = document.getElementById("apply");
|
||||||
|
selector.value = storage.data["selector-css"];
|
||||||
|
selector.oninput = function() { apply.disabled = selector.value == storage.data["selector-css"]; };
|
||||||
|
apply.onclick = function() {
|
||||||
|
storage.data["selector-css"] = selector.value;
|
||||||
|
storage.commit(["selector-css"]);
|
||||||
|
apply.disabled = true;
|
||||||
|
};
|
||||||
|
document.getElementById("reset").onclick = function() {
|
||||||
|
var url = chrome.extension.getURL("script/selector.css");
|
||||||
|
chrome.runtime.sendMessage({ msg: "load", script: "selector-css", file: url });
|
||||||
|
};
|
||||||
|
storage.addChangeListener(function(changes){
|
||||||
|
if (changes.includes("selector-css")) selector.value = storage.data["selector-css"];
|
||||||
|
apply.disabled = true;
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
function setup(pfx)
|
||||||
|
{
|
||||||
|
var file = pfx + "codefile";
|
||||||
|
var filename = file + "name";
|
||||||
|
|
||||||
|
document.getElementById(pfx + "path-input").textContent = storage.data[file];
|
||||||
|
document.getElementById(pfx + "filename").textContent = storage.data[filename];
|
||||||
|
updateState();
|
||||||
|
|
||||||
|
document.getElementById(pfx + "load-button").onclick = function() {
|
||||||
|
var url = storage.data[file] || chrome.extension.getURL("script/") + (pfx && "user") + "handler.js";
|
||||||
|
chrome.runtime.sendMessage({ msg: "load", script: pfx + "handler", file: url});
|
||||||
|
};
|
||||||
|
document.getElementById(pfx + "export-button").onclick = function() {
|
||||||
|
var a = document.getElementById("export-link");
|
||||||
|
a.download = (pfx && "user") + "handler.js";
|
||||||
|
a.href = "data:application/javascript," + encodeURIComponent(storage.data[pfx + "handler"]);
|
||||||
|
a.click();
|
||||||
|
};
|
||||||
|
document.getElementById(pfx + "edit-button").onclick = function() {
|
||||||
|
var open = chrome.devtools.panels.openResource || function(){};
|
||||||
|
open(getURL(file), null, function(res){
|
||||||
|
if (res.isError && res.code == "E_NOTFOUND")
|
||||||
|
alert(chrome.i18n.getMessage("fileNotFound"));
|
||||||
|
});
|
||||||
|
};
|
||||||
|
var committed = chrome.devtools && chrome.devtools.inspectedWindow.onResourceContentCommitted;
|
||||||
|
committed && committed.addListener(function(resource){
|
||||||
|
if (resource.url == getURL(file))
|
||||||
|
resource.getContent(function(content, encoding) {
|
||||||
|
chrome.devtools.inspectedWindow.eval(content, function(result, isException) {
|
||||||
|
if (isException)
|
||||||
|
alert(chrome.i18n.getMessage("evalError") + isException.value);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
document.getElementById(pfx + "path-input").oninput = updateState;
|
||||||
|
document.getElementById(pfx + "path-button").onclick = function(e) {
|
||||||
|
storage.data[filename] = document.getElementById(pfx + "filename").textContent = "";
|
||||||
|
storage.data[file] = document.getElementById(pfx + "path-input").textContent;
|
||||||
|
storage.commit([file, filename]);
|
||||||
|
updateState();
|
||||||
|
};
|
||||||
|
document.getElementById(pfx + "select-button").onclick = function(e) {
|
||||||
|
fileinput.onchange = function() {
|
||||||
|
if (!chrome.extension.getBackgroundPage) { // fff...
|
||||||
|
var data = {msg: "select", url: URL.createObjectURL(this.files[0])};
|
||||||
|
chrome.runtime.sendMessage(data, function(res) { updateSelected(res.url, res.name); });
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
var url = chrome.extension.getBackgroundPage().URL.createObjectURL(this.files[0]);
|
||||||
|
updateSelected(url, this.files[0].name);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
fileinput.click();
|
||||||
|
};
|
||||||
|
|
||||||
|
function updateSelected(url, name) {
|
||||||
|
storage.data[file] = document.getElementById(pfx + "path-input").textContent = url;
|
||||||
|
storage.data[filename] = document.getElementById(pfx + "filename").textContent = name;
|
||||||
|
storage.commit([file, filename]);
|
||||||
|
updateState();
|
||||||
|
};
|
||||||
|
|
||||||
|
function updateState()
|
||||||
|
{
|
||||||
|
var a = document.getElementById(pfx + "path-input").textContent == (storage.data[file] || "");
|
||||||
|
document.getElementById(pfx + "path-button").disabled = a;
|
||||||
|
document.getElementById(pfx + "load-button").disabled = !a;
|
||||||
|
document.getElementById(pfx + "edit-button").disabled = !a || !storage.data[file];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
function getURL(file) { try { return new URL(storage.data[file]).href } catch(e) {}; };
|
||||||
|
|
||||||
|
function setCheckbox(id, key = id)
|
||||||
|
{
|
||||||
|
var node = document.getElementById(id);
|
||||||
|
node.checked = !!storage.data[key];
|
||||||
|
node.onchange = function() {
|
||||||
|
storage.data[key] = this.checked;
|
||||||
|
storage.commit([key]);
|
||||||
|
};
|
||||||
|
};
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
//</>
|
BIN
AutoplayStopper/icons/icon32.png
Normal file
BIN
AutoplayStopper/icons/icon32.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 2.0 KiB |
BIN
AutoplayStopper/icons/icon32d.png
Normal file
BIN
AutoplayStopper/icons/icon32d.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 1.7 KiB |
BIN
AutoplayStopper/icons/icon48.png
Normal file
BIN
AutoplayStopper/icons/icon48.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 4.8 KiB |
53
AutoplayStopper/manifest.json
Normal file
53
AutoplayStopper/manifest.json
Normal file
|
@ -0,0 +1,53 @@
|
||||||
|
{
|
||||||
|
"description": "[MODIFIED BY ME] Stops video autoplay gracefully.",
|
||||||
|
"manifest_version": 2,
|
||||||
|
"name": "AutoplayStopper",
|
||||||
|
"version": "1.8.7",
|
||||||
|
|
||||||
|
"default_locale": "en",
|
||||||
|
|
||||||
|
"icons": {
|
||||||
|
"32": "icons/icon32.png",
|
||||||
|
"48": "icons/icon48.png"
|
||||||
|
},
|
||||||
|
|
||||||
|
"permissions": [
|
||||||
|
"http://*/*", "https://*/*",
|
||||||
|
"tabs",
|
||||||
|
"contextMenus",
|
||||||
|
"storage",
|
||||||
|
"webNavigation"
|
||||||
|
],
|
||||||
|
|
||||||
|
"background": {
|
||||||
|
"scripts": ["utils.js", "background.js"]
|
||||||
|
},
|
||||||
|
|
||||||
|
"browser_action": {
|
||||||
|
"default_icon": {
|
||||||
|
"32": "icons/icon32.png"
|
||||||
|
},
|
||||||
|
"default_title": "AutoplayStopper",
|
||||||
|
"default_popup": "skin/popup.html",
|
||||||
|
"browser_style": true
|
||||||
|
},
|
||||||
|
|
||||||
|
"content_scripts": [
|
||||||
|
{
|
||||||
|
"run_at": "document_start",
|
||||||
|
"all_frames": true,
|
||||||
|
"match_about_blank": true,
|
||||||
|
"matches": ["http://*/*", "https://*/*"],
|
||||||
|
"js": ["content/handler-utils.js", "content/content.js"]
|
||||||
|
}],
|
||||||
|
|
||||||
|
"options_ui": {
|
||||||
|
"chrome_style": true,
|
||||||
|
"browser_style": true,
|
||||||
|
"page": "skin/options.html"
|
||||||
|
},
|
||||||
|
|
||||||
|
"devtools_page": "skin/devtools.html?view=page",
|
||||||
|
|
||||||
|
"optional_permissions": ["file://*/*"]
|
||||||
|
}
|
292
AutoplayStopper/options.js
Normal file
292
AutoplayStopper/options.js
Normal file
|
@ -0,0 +1,292 @@
|
||||||
|
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
const Options = new function() {
|
||||||
|
|
||||||
|
const {i18n, storage, permissions: perms} = chrome.extension.getBackgroundPage().Background;
|
||||||
|
const permsAutoplayKey = perms.key("autoplay");
|
||||||
|
const permsAutoplayDefaultKey = perms.defaultKey("autoplay");
|
||||||
|
const permsFlashKey = perms.key("flash");
|
||||||
|
const permsFlashDefaultKey = perms.defaultKey("flash");
|
||||||
|
const data = { flash: { type: "flash", dirty: false}, autoplay: { type: "autoplay", dirty: false}};
|
||||||
|
const { autoplay, flash } = data;
|
||||||
|
|
||||||
|
var tabId = null;
|
||||||
|
var selected = null;
|
||||||
|
|
||||||
|
// chrome.tabs.getCurrent(function(tab) { tabId = tab.id; }); // chrome workaround ...
|
||||||
|
chrome.tabs.query({ active: true, currentWindow: true}, function(tabs) { tabId = tabs[0].id; });
|
||||||
|
chrome.tabs.onActivated.addListener(function(info){ if (tabId == info.tabId) postActive(); });
|
||||||
|
addEventListener("load", init);
|
||||||
|
addEventListener("hashchange", function() { updateState(location.hash.slice(1)); });
|
||||||
|
|
||||||
|
function init()
|
||||||
|
{
|
||||||
|
i18n.process(document);
|
||||||
|
var listener = storage.addChangeListener(function(changes){
|
||||||
|
if (changes.indexOf(permsAutoplayKey) != -1) {
|
||||||
|
autoplay.exceptions.init(perms.entries("autoplay"));
|
||||||
|
autoplay.dirty = false;
|
||||||
|
updateApply();
|
||||||
|
}
|
||||||
|
if(changes.indexOf(permsAutoplayDefaultKey) != -1) {
|
||||||
|
document.getElementById("autoplay-default").value = perms.default("autoplay");
|
||||||
|
}
|
||||||
|
if (changes.indexOf(permsFlashKey) != -1) {
|
||||||
|
flash.exceptions.init(perms.entries("flash"));
|
||||||
|
flash.dirty = false;
|
||||||
|
updateApply();
|
||||||
|
}
|
||||||
|
if(changes.indexOf(permsFlashDefaultKey) != -1) {
|
||||||
|
document.getElementById("flash-default").value = perms.default("flash");
|
||||||
|
}
|
||||||
|
});
|
||||||
|
addEventListener("unload", function(){ storage.removeChangeListener(listener); });
|
||||||
|
|
||||||
|
var manifest = chrome.runtime.getManifest();
|
||||||
|
document.getElementById("extension-version").appendChild(document.createTextNode(manifest.version));
|
||||||
|
document.getElementById("autoplay-default").value = perms.default("autoplay");
|
||||||
|
document.getElementById("flash-default").value = perms.default("flash");
|
||||||
|
document.getElementById("devtools").checked = storage.data.devtools;
|
||||||
|
autoplay.exceptions = new ExceptionsList(document.getElementById("autoplay-list"));
|
||||||
|
autoplay.exceptions.init(perms.entries("autoplay"));
|
||||||
|
document.getElementById("autoplay-default").addEventListener("change", function(e) {
|
||||||
|
perms.setDefault("autoplay", e.target.value);
|
||||||
|
storage.commit([permsAutoplayDefaultKey]);
|
||||||
|
});
|
||||||
|
flash.exceptions = new ExceptionsList(document.getElementById("flash-list"));
|
||||||
|
flash.exceptions.init(perms.entries("flash"));
|
||||||
|
document.getElementById("flash-default").addEventListener("change", function(e) {
|
||||||
|
perms.setDefault("flash", e.target.value);
|
||||||
|
storage.commit([permsFlashDefaultKey]);
|
||||||
|
});
|
||||||
|
document.querySelectorAll(".exceptions-list-button").forEach(function(button) {
|
||||||
|
button.onclick = function(e) { location.hash = `${button.getAttribute("contenttype")}`; };
|
||||||
|
});
|
||||||
|
autoplay.exceptions.addDirtyListener(function(d) { autoplay.dirty = d; updateApply(); });
|
||||||
|
flash.exceptions.addDirtyListener(function(d) { flash.dirty = d; updateApply(); });
|
||||||
|
document.querySelector(".close-button").onclick = function() { location.hash = ""; };
|
||||||
|
document.getElementById("exceptions-clear").onclick = function() { selected.exceptions.clear(); };
|
||||||
|
document.getElementById("exceptions-apply").onclick = apply;
|
||||||
|
document.getElementById("exceptions-confirm").onclick = function() { apply(); location.hash = ""; };
|
||||||
|
document.getElementById("devtools").onclick = function(e) {
|
||||||
|
storage.data.devtools = this.checked;
|
||||||
|
storage.commit(["devtools"]);
|
||||||
|
};
|
||||||
|
if (location.hash) updateState(location.hash.slice(1));
|
||||||
|
postActive();
|
||||||
|
};
|
||||||
|
|
||||||
|
function postActive() { chrome.extension.getBackgroundPage().postMessage("optionsPageActive", "*"); };
|
||||||
|
function updateApply() { document.getElementById("exceptions-apply").disabled = !selected || !selected.dirty; };
|
||||||
|
|
||||||
|
function apply()
|
||||||
|
{
|
||||||
|
if (selected.dirty) {
|
||||||
|
perms.clear(selected.type);
|
||||||
|
for (var [url, perm] of selected.exceptions.entries()) { perms.set(selected.type, url, perm); };
|
||||||
|
storage.commit([perms.key(selected.type)], function(err){
|
||||||
|
if (err) { selected.dirty = true; };
|
||||||
|
updateApply();
|
||||||
|
});
|
||||||
|
selected.dirty = false;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
function updateState(state)
|
||||||
|
{
|
||||||
|
// console.log(`updateState(${state})`);
|
||||||
|
if (state) {
|
||||||
|
document.querySelector("div[contenttype]:not([hidden])").hidden = true;
|
||||||
|
document.querySelector("h2[contenttype]:not([hidden])").hidden = true;
|
||||||
|
document.querySelector(`div[contenttype=${state}]`).hidden = false;
|
||||||
|
document.querySelector(`h2[contenttype=${state}]`).hidden = false;
|
||||||
|
var overlay = document.querySelector(".overlay");
|
||||||
|
overlay.hidden = false;
|
||||||
|
setTimeout(function() { overlay.classList.remove("transparent"); }, 20);
|
||||||
|
selected = data[state];
|
||||||
|
updateApply();
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
var overlay = document.querySelector(".overlay");
|
||||||
|
overlay.classList.add("transparent");
|
||||||
|
setTimeout(function() { overlay.hidden = true; }, 500);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const ExceptionsList = function ExceptionsList(list){
|
||||||
|
|
||||||
|
var itemid = 0;
|
||||||
|
var baseitem = list.querySelector("#listitem");
|
||||||
|
var inputitem = list.querySelector("#inputitem");
|
||||||
|
var selected = list.querySelector("[selected]");
|
||||||
|
var listeners = [];
|
||||||
|
var dirty = false;
|
||||||
|
|
||||||
|
baseitem.remove();
|
||||||
|
list.addEventListener("mousedown", handleMousedown.bind(this));
|
||||||
|
list.addEventListener("click", handleClick.bind(this));
|
||||||
|
list.addEventListener("keydown", handleKeyDown.bind(this));
|
||||||
|
list.addEventListener("change", handleChange.bind(this));
|
||||||
|
list.addEventListener("focus", function(e) {
|
||||||
|
if (!e.target.classList.contains("row-delete-button")) {
|
||||||
|
list.setAttribute("has-element-focus", "hasElementFocus");
|
||||||
|
handleSelection(e.target.closest("[role=listitem]"));
|
||||||
|
if (e.target.getAttribute("displaymode") == "static") e.target.nextSibling.focus();
|
||||||
|
}
|
||||||
|
}, true);
|
||||||
|
list.addEventListener("blur", function(e) {
|
||||||
|
if (!list.contains(e.relatedTarget) && !e.target.classList.contains("row-delete-button")) {
|
||||||
|
list.removeAttribute("has-element-focus");
|
||||||
|
handleSelection();
|
||||||
|
if (!e.relatedTarget) list.parentNode.closest("[tabindex]").focus();
|
||||||
|
}
|
||||||
|
}, true);
|
||||||
|
|
||||||
|
return {
|
||||||
|
init: function init(entries) {
|
||||||
|
this.clear();
|
||||||
|
entries.sort(function(a, b) { return a[0].localeCompare(b[0]); });
|
||||||
|
for (var [key, type] of entries) { appendItem(key, type); };
|
||||||
|
dirty = false;
|
||||||
|
},
|
||||||
|
entries: function * entries()
|
||||||
|
{
|
||||||
|
for (var node of list.children)
|
||||||
|
if (node.id.search("listitem") == 0 && node != inputitem)
|
||||||
|
yield [node.querySelector("#perms-host").textContent, parseInt(node.querySelector("#perms-type-select").value)];
|
||||||
|
},
|
||||||
|
clear: function clear()
|
||||||
|
{
|
||||||
|
for (var node of [].slice.apply(list.children))
|
||||||
|
if (node.id.search("listitem") == 0 && node != inputitem) node.remove();
|
||||||
|
inputitem.querySelector("#perms-host-input").focus();
|
||||||
|
fireDirty(true);
|
||||||
|
},
|
||||||
|
get dirty() { return dirty; },
|
||||||
|
addDirtyListener: function(listener) { listeners.push(listener); },
|
||||||
|
removeDirtyListener: function(listener) { var idx = listeners.indexOf(listener); if (idx != -1) listeners.splice(idx); }
|
||||||
|
};
|
||||||
|
|
||||||
|
function appendItem(host, type, scroll)
|
||||||
|
{
|
||||||
|
var item = baseitem.cloneNode(true);
|
||||||
|
item.id = item.id + "-" + (++itemid);
|
||||||
|
item.querySelector("#perms-host-input").value = item.querySelector("#perms-host").textContent = host;
|
||||||
|
var select = item.querySelector("#perms-type-select");
|
||||||
|
select.value = type;
|
||||||
|
if (type != 0) item.querySelector("#perms-type").textContent = select[select.selectedIndex].text;
|
||||||
|
list.insertBefore(item, inputitem);
|
||||||
|
if (scroll) scrollToBottom(inputitem);
|
||||||
|
};
|
||||||
|
|
||||||
|
function scrollToBottom(node)
|
||||||
|
{
|
||||||
|
if (node.clientHeight < node.scrollHeight)
|
||||||
|
return node.scrollTop = node.scrollHeight - node.clientHeight;
|
||||||
|
if (node.parentNode) scrollToBottom(node.parentNode);
|
||||||
|
};
|
||||||
|
|
||||||
|
function handleMousedown(e)
|
||||||
|
{
|
||||||
|
var target = e.target;
|
||||||
|
var item = target.closest("[role=listitem]");
|
||||||
|
if (target.classList.contains("row-delete-button")) return;
|
||||||
|
if (target.id == "perms-host") {
|
||||||
|
var x = e.clientX, y = e.clientY, offset = document.caretPositionFromPoint ?
|
||||||
|
document.caretPositionFromPoint(x, y).offset : document.caretRangeFromPoint(x, y).startOffset;
|
||||||
|
var input = item.querySelector("#perms-host-input");
|
||||||
|
var inside = x < inputitem.querySelector("#perms-host-input").getBoundingClientRect().right;
|
||||||
|
input.selectionStart = input.selectionEnd = inside ? 0 : input.value.length;
|
||||||
|
setTimeout(function() { input.selectionStart = input.selectionEnd = offset; }, 25);
|
||||||
|
}
|
||||||
|
if (item && !target.hasAttribute("tabindex") && !item.hasAttribute("editing")) {
|
||||||
|
item.querySelector("[id^=perms-host]").focus();
|
||||||
|
e.preventDefault();
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
function handleSelection(item)
|
||||||
|
{
|
||||||
|
if (selected && item != selected) {
|
||||||
|
selected.removeAttribute("lead");
|
||||||
|
selected.removeAttribute("editing");
|
||||||
|
if (item) selected.removeAttribute("selected");
|
||||||
|
if (selected == inputitem) handleInput(!item);
|
||||||
|
selected.querySelectorAll("[id^=perms]").forEach(function(a) { a.tabIndex = item ? -1 : 0; });
|
||||||
|
}
|
||||||
|
if (item && !item.hasAttribute("editing")) {
|
||||||
|
item.setAttribute("selected", "selected");
|
||||||
|
item.setAttribute("lead", "lead");
|
||||||
|
item.setAttribute("editing", "");
|
||||||
|
item.querySelectorAll("[id^=perms]")
|
||||||
|
.forEach(function(a) { a.tabIndex = a.getAttribute("displaymode") == "static" ? -1 : 0; });
|
||||||
|
selected = item;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
function handleClick(e)
|
||||||
|
{
|
||||||
|
var target = e.target;
|
||||||
|
if (target.classList.contains("row-delete-button")) {
|
||||||
|
var item = target.closest("[role=listitem]");
|
||||||
|
if (item == selected) handleSelection(inputitem);
|
||||||
|
item.remove();
|
||||||
|
list.focus({preventScroll: true});
|
||||||
|
fireDirty(true);
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
function handleKeyDown(e)
|
||||||
|
{
|
||||||
|
if (selected && selected != inputitem && e.target.id != "perms-host-input" && e.keyCode == 46)
|
||||||
|
selected.querySelector(".row-delete-button").click();
|
||||||
|
if (selected && e.key == "Enter") selected == inputitem ? handleInput(true) : list.focus({preventScroll: true});
|
||||||
|
if (selected && e.key == "ArrowUp" && e.target.id != "perms-type-select" && selected != list.querySelector("[role=listitem]"))
|
||||||
|
selected.previousSibling.querySelector("[id^=perms-host]").focus();
|
||||||
|
if (selected && e.key == "ArrowDown" && e.target.id != "perms-type-select" && selected != inputitem)
|
||||||
|
selected.nextSibling.querySelector("[id^=perms-host]").focus();
|
||||||
|
};
|
||||||
|
|
||||||
|
function handleChange(e)
|
||||||
|
{
|
||||||
|
var target = e.target;
|
||||||
|
var item = target.closest("[role=listitem]");
|
||||||
|
if (item != inputitem) {
|
||||||
|
if (target.id == "perms-type-select") {
|
||||||
|
item.querySelector("#perms-type").textContent =
|
||||||
|
target.value != 0 ? target[target.selectedIndex].text : "";
|
||||||
|
fireDirty(true);
|
||||||
|
}
|
||||||
|
if (target.id == "perms-host-input"){
|
||||||
|
try {
|
||||||
|
var url = new URL(target.value);
|
||||||
|
target.value = url.origin;
|
||||||
|
item.querySelector("#perms-host").textContent = target.value;
|
||||||
|
fireDirty(true);
|
||||||
|
}
|
||||||
|
catch (e) { target.value = item.querySelector("#perms-host").textContent; };
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
function handleInput(scroll)
|
||||||
|
{
|
||||||
|
try {
|
||||||
|
var target = inputitem.querySelector("#perms-host-input")
|
||||||
|
var url = new URL(target.value);
|
||||||
|
target.value = url.origin;
|
||||||
|
appendItem(target.value, inputitem.querySelector("#perms-type-select").value, scroll);
|
||||||
|
target.value = null;
|
||||||
|
fireDirty(true);
|
||||||
|
}
|
||||||
|
catch (e) {};
|
||||||
|
};
|
||||||
|
|
||||||
|
function fireDirty(d) { dirty = d; for (var listener of listeners) listener(d); };
|
||||||
|
};
|
||||||
|
|
||||||
|
// </>
|
94
AutoplayStopper/popup.js
Normal file
94
AutoplayStopper/popup.js
Normal file
|
@ -0,0 +1,94 @@
|
||||||
|
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
const Popup = new function(){
|
||||||
|
|
||||||
|
const start = Date.now();
|
||||||
|
const {i18n, storage, permissions: perms, openOptionsPage} = chrome.extension.getBackgroundPage().Background;
|
||||||
|
|
||||||
|
while (Date.now() - start < 50);
|
||||||
|
setTimeout(function() {
|
||||||
|
while (Date.now() - start < 150);
|
||||||
|
addEventListener("unload", function() {
|
||||||
|
if (Date.now() - start < 250) {
|
||||||
|
storage.data.disabled = !storage.data.disabled;
|
||||||
|
storage.commit(["disabled"]);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
const permsAutoplayKey = perms.key("autoplay");
|
||||||
|
const permsFlashKey = perms.key("flash");
|
||||||
|
|
||||||
|
var url = null;
|
||||||
|
var ready = new Promise( function(resolve) {
|
||||||
|
chrome.tabs.query({ active: true, currentWindow: true}, function(tabs) {
|
||||||
|
url = new URL(tabs[0].url);
|
||||||
|
resolve();
|
||||||
|
})
|
||||||
|
});
|
||||||
|
|
||||||
|
addEventListener("load", function() { ready.then(init); });
|
||||||
|
|
||||||
|
function init()
|
||||||
|
{
|
||||||
|
i18n.process(document);
|
||||||
|
var permission = perms.testPermission("autoplay", url);
|
||||||
|
var permissionFlash = perms.testPermission("flash", url);
|
||||||
|
initItem( document.getElementById("allow_site"), !!permission.origin && permission != perms.ACCESS_SESSION && permission, [url.origin]);
|
||||||
|
initItem( document.getElementById("allow_session"), !!permission.origin && permission == perms.ACCESS_SESSION, [url.origin]);
|
||||||
|
initItem( document.getElementById("disable_site"), !!permissionFlash.origin && permissionFlash, [url.origin]);
|
||||||
|
initItem( document.getElementById("allow_all"), storage.data.disabled);
|
||||||
|
initItem( document.getElementById("settings"));
|
||||||
|
document.getElementById("popupmenu")
|
||||||
|
.classList.toggle("restricted", url.protocol.search("^http") == -1 || !!storage.data.disabled);
|
||||||
|
|
||||||
|
window.onclick = function(e) {
|
||||||
|
|
||||||
|
switch (e.target.id) {
|
||||||
|
case "settings":
|
||||||
|
openOptionsPage();
|
||||||
|
close();
|
||||||
|
break;
|
||||||
|
case "allow_site":
|
||||||
|
if ( [undefined, perms.UNKNOWN_ACTION].indexOf(perms.get("autoplay", url)) != -1
|
||||||
|
&& permission != perms.default("autoplay")) { openOptionsPage("#autoplay"); close(); break; };
|
||||||
|
perms.set("autoplay", url, (permission == perms.ALLOW_ACTION) ? perms.DENY_ACTION : perms.ALLOW_ACTION);
|
||||||
|
storage.commit([permsAutoplayKey]);
|
||||||
|
init();
|
||||||
|
break;
|
||||||
|
case "allow_session":
|
||||||
|
if ( [undefined, perms.UNKNOWN_ACTION].indexOf(perms.get("autoplay", url)) != -1
|
||||||
|
&& permission != perms.default("autoplay")) { openOptionsPage("#autoplay"); close(); break; };
|
||||||
|
(permission == perms.ACCESS_SESSION) ? perms.remove("autoplay", url) :
|
||||||
|
perms.set("autoplay", url, perms.ACCESS_SESSION);
|
||||||
|
storage.commit([permsAutoplayKey]);
|
||||||
|
init();
|
||||||
|
break;
|
||||||
|
case "disable_site":
|
||||||
|
if ( [undefined, perms.UNKNOWN_ACTION].indexOf(perms.get("flash", url)) != -1
|
||||||
|
&& permissionFlash != perms.default("flash")) { openOptionsPage("#flash"); close(); break; };
|
||||||
|
perms.set("flash", url, (permissionFlash == perms.ALLOW_ACTION) ? perms.DENY_ACTION : perms.ALLOW_ACTION);
|
||||||
|
storage.commit([permsFlashKey]);
|
||||||
|
init();
|
||||||
|
break;
|
||||||
|
case "allow_all":
|
||||||
|
storage.data.disabled = !storage.data.disabled;
|
||||||
|
storage.commit(["disabled"]);
|
||||||
|
init();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
function initItem(item, check, args)
|
||||||
|
{
|
||||||
|
if (check !== undefined) check ? item.setAttribute("checked", check) : item.removeAttribute("checked");
|
||||||
|
if (args) { var i = 0; item.textContent = item.textContent.replace(/%S/g, function () { return args[i++]; }); };
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
//</>
|
289
AutoplayStopper/script/handler.js
Normal file
289
AutoplayStopper/script/handler.js
Normal file
|
@ -0,0 +1,289 @@
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
function registerHandlers(handlers, TRACE, isHandlingUserInput, setHandlingUserInput, lastUserInput, data = {})
|
||||||
|
{
|
||||||
|
var dummyid = Math.floor(Math.random() * 100);
|
||||||
|
|
||||||
|
function testUserInput(el, delay)
|
||||||
|
{
|
||||||
|
var win = el.ownerDocument.defaultView, _last = lastUserInput(win), e = _last && _last.e;
|
||||||
|
if (!e || Date.now() - _last.time > delay) return false;
|
||||||
|
var px = e.pageX - win.scrollX, py = win.scrollY ? e.pageY - win.scrollY : e.clientY, r = el.getBoundingClientRect(), {width ,height} = r;
|
||||||
|
for (var i = 0; i < 5 && el.parentNode && (r.top + win.scrollY < 0 || r.left + win.scrollX < 0 || r.width < 10 || r.height < 10); i++) r = (el = el.parentNode).getBoundingClientRect();
|
||||||
|
r.width = Math.max(r.width, width), r.height = Math.max(r.height, height);
|
||||||
|
return px >= r.left && px <= r.right && py >= r.top && py <= r.bottom;
|
||||||
|
};
|
||||||
|
// htmlmedia
|
||||||
|
handlers.add(function handleHtmlMedia(aElement){
|
||||||
|
|
||||||
|
const simEvents = {play: 0, playing: 50, pause: 350}, eventTypes = Object.keys(simEvents), props = ["preload","autoplay","poster","setAttribute","removeAttribute"];
|
||||||
|
const document = aElement.ownerDocument, window = document.defaultView;
|
||||||
|
if (aElement.localName == "video" || aElement.localName == "audio")
|
||||||
|
{
|
||||||
|
aElement.id || (aElement.id = "dummyid" + (++dummyid % 100));
|
||||||
|
TRACE('handleElement html5 media - tag: %s id: %s autoplay: %s preload: "%s" poster: %s paused: %s')(aElement.localName, aElement.id, aElement.autoplay, aElement.preload, !!aElement.poster, aElement.paused);
|
||||||
|
if (aElement.wrappedJSObject.play !== window.wrappedJSObject.HTMLMediaElement.prototype.play) TRACE('handleElement html5 media - %s.play() != HTMLMediaElement.play !!!')(aElement.id);
|
||||||
|
// Yendifplayer .autoplay = 1 => ...
|
||||||
|
var autoplay = aElement.autoplay || !aElement.paused, count = 0, released = false;
|
||||||
|
Object.defineProperties(aElement.wrappedJSObject, { preload: {configurable: true, set: (a) => TRACE('%s.preload = "%s"')(aElement.id, a)/* || (aElement.preload = "metadata")*/, get: () => aElement.preload}
|
||||||
|
, autoplay: {configurable: true, set: (a) => TRACE("%s.autoplay = %s")(aElement.id, a) || (a && aElement.wrappedJSObject.play()), get: () => aElement.autoplay}});
|
||||||
|
aElement.wrappedJSObject.setAttribute = (a, b) => (props.indexOf(a) != -1) ? TRACE("%s.setAttribute(%s, %s)")(aElement.id, a, b) || (aElement.wrappedJSObject[a] = b): aElement.setAttribute(a, b);
|
||||||
|
aElement.wrappedJSObject.removeAttribute = (a) => (props.indexOf(a) != -1) ? TRACE("%s.removeAttribute(%s)")(aElement.id, a) : aElement.removeAttribute(a);
|
||||||
|
|
||||||
|
function simPlay(e) {
|
||||||
|
e && aElement.removeEventListener(e.type, simPlay);
|
||||||
|
if (aElement.simPlay !== false && aElement.paused)
|
||||||
|
eventTypes.forEach((key) => window.setTimeout(() => !released && (TRACE("simPlay - %s.%s")(aElement.id, key) || aElement.dispatchEvent(new window.Event(key))), simEvents[key]));
|
||||||
|
};
|
||||||
|
|
||||||
|
aElement.wrappedJSObject.play = () => {
|
||||||
|
var force = aElement.allowPlay || testUserInput(aElement, 5000);
|
||||||
|
var userInput = isHandlingUserInput(window);
|
||||||
|
TRACE("%s.play(%s) - force: %s state: %s user: %s")(aElement.id, count, !!force, aElement.readyState, userInput);
|
||||||
|
if (!userInput && !force) {
|
||||||
|
count++ == 0 && (aElement.preload = "metadata") && (aElement.simPlay || aElement.readyState >= 2 ? simPlay() : aElement.addEventListener("loadeddata", simPlay));
|
||||||
|
return count < 5 ? Promise.reject(new window.DOMException("The play method is not allowed by the user agent.", "NotAllowedError")) : simPlay() || Promise.resolve();
|
||||||
|
}
|
||||||
|
released = !data.strict && delete aElement.wrappedJSObject.play; // jwplayer: zapiks.fr rottentomatoes baeblemusic
|
||||||
|
force && !userInput && setHandlingUserInput(window);
|
||||||
|
return aElement.play();
|
||||||
|
};
|
||||||
|
window.wrappedJSObject.Object.defineProperty(aElement.wrappedJSObject.play, "toString", window.Object.assign(window.Object(), {value: () => "[native code]"}));
|
||||||
|
aElement.autoplay = false;
|
||||||
|
aElement.preload != "none" && (aElement.preload = "metadata"); // mediaelement.js
|
||||||
|
!data.strict && aElement.addEventListener("play", function cleanup(e){ e.isTrusted && !aElement.paused && (aElement.removeEventListener("play", cleanup), aElement.muted = false,
|
||||||
|
TRACE("%s cleanup...")(aElement.id), props.forEach((prop) => delete aElement.wrappedJSObject[prop]), delete aElement.wrappedJSObject.play)});
|
||||||
|
autoplay && aElement.wrappedJSObject.play(); // flowplayer
|
||||||
|
!aElement.paused ? !released && aElement.pause() : window.setTimeout(() => !released && aElement.pause(), 0); // dbtv.no html5box
|
||||||
|
};
|
||||||
|
});
|
||||||
|
// simplay
|
||||||
|
const hosts = [{host: "cnn.com$", val: true}, {host: "twitch.tv$"}, {host: "yastatic.net$"},
|
||||||
|
{host: "bbc.(com|co.uk)$"}, {host: "(yahoo|yimg).com$", class: "html5-video", val: true}, {class: "\\b(video-js|vjs-tech)\\b"}];
|
||||||
|
handlers.add(function handleSimplay(aElement){
|
||||||
|
if (aElement.localName == "video")
|
||||||
|
for (var a of hosts) if (aElement.ownerDocument.location.hostname.match(a.host) && aElement.className.match(a.class)) aElement.simPlay = !!a.val;
|
||||||
|
});
|
||||||
|
// YouTube
|
||||||
|
handlers.add(function handleYouTube(aElement){
|
||||||
|
|
||||||
|
const document = aElement.ownerDocument, window = document.defaultView, jsWin = window.wrappedJSObject;
|
||||||
|
if (document.location.hostname.search("\.youtube(-nocookie)?\.com$") != -1)
|
||||||
|
{
|
||||||
|
TRACE("handleElement ytplayer - tag: %s player: %s")(aElement.localName, aElement.id);
|
||||||
|
if (document.location.search.search("feature=youtube-anywhere-player") == -1)
|
||||||
|
{
|
||||||
|
function stopPlayer(ytplayer){
|
||||||
|
if (testUserInput(ytplayer, 5000)) return !TRACE("ytplayer released...")();
|
||||||
|
var vid = ytplayer.getVideoData && ytplayer.getVideoData().video_id;
|
||||||
|
TRACE("ytplayer stopPlayer(%s) - videoId: %s t: %s")(ytplayer.id || "", vid, ytplayer.getCurrentTime());
|
||||||
|
ytplayer.cueVideoById(vid, ytplayer.getCurrentTime());
|
||||||
|
var nop = false, _seekTo = ytplayer.seekTo, _playVideo = ytplayer.playVideo;
|
||||||
|
ytplayer.playVideo = function doNothing() { nop || (nop = testUserInput(ytplayer, 1000)) ? _playVideo.apply(this) : TRACE("@@@@@@ ytplayer doNothing @@@@@")()};
|
||||||
|
ytplayer.seekTo = function seekTo(tm) { nop || (nop = testUserInput(ytplayer, 1000)) ? _seekTo.apply(this, arguments) : ytplayer.cueVideoById(vid, tm), TRACE("ytplayer seekTo(%s)")(tm); };
|
||||||
|
function onstate(a) { if (a == 1) { nop = true; ytplayer.playVideo = _playVideo; ytplayer.seekTo = _seekTo; TRACE("ytplayer - removed doNothing")(); ytplayer.removeEventListener("onStateChange", onstate); }};
|
||||||
|
ytplayer.addEventListener("onStateChange", onstate);
|
||||||
|
}
|
||||||
|
if (aElement.localName == "video" && aElement.parentNode.classList.contains("html5-video-container")) {
|
||||||
|
if (!data.strict && aElement.parentNode.parentNode.id == "movie_player")
|
||||||
|
window.setTimeout(() => aElement.allowPlay = true, document.location.search.search("feature=youtube-anywhere-player") == -1 ? 5000 : 0);
|
||||||
|
aElement.simPlay = false;
|
||||||
|
if (aElement.parentNode.parentNode.classList.contains("html5-video-player"))
|
||||||
|
window.setTimeout(() => stopPlayer(aElement.wrappedJSObject.parentNode.parentNode));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
// googleads
|
||||||
|
handlers.add(function handleGoogleAds(aElement){
|
||||||
|
const document = aElement.ownerDocument, window = document.defaultView;
|
||||||
|
var host = aElement.getRootNode().host;
|
||||||
|
if (host && host.localName == "lima-video") {
|
||||||
|
TRACE("handleElement googleads - tag: %s")(aElement.localName);
|
||||||
|
window.addEventListener("visibilitychange", (e) => e.stopPropagation(), true);
|
||||||
|
aElement.wrappedJSObject.addEventListener("play", function(){ if (!testUserInput(this, 100)) this.pause()}, {once: true});
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
// ifunny.co
|
||||||
|
handlers.add(function handleIFunny(aElement){
|
||||||
|
const document = aElement.ownerDocument, window = document.defaultView;
|
||||||
|
if (document.location.hostname.match("ifunny.co$") && aElement.paused) {
|
||||||
|
(isHandlingUserInput(window) || testUserInput(aElement.parentNode, 1000)) ? aElement.play() : aElement.src = "";
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
// msn.com
|
||||||
|
handlers.add(function handleMsn(aElement){
|
||||||
|
|
||||||
|
const document = aElement.ownerDocument, window = document.defaultView, jsWin = window.wrappedJSObject;
|
||||||
|
if (document.location.hostname.search(/\.msn\.com$/) != -1)
|
||||||
|
{
|
||||||
|
var obj = aElement.wrappedJSObject;
|
||||||
|
if (aElement.className.indexOf("vxFlashPlayer") != -1 && aElement.localName == "embed"){
|
||||||
|
obj.MsnVideoCallback = function(msg) { msg == "playVideo" ? delete obj.MsnVideoCallback : obj.__proto__.MsnVideoCallback.apply(obj, arguments)};
|
||||||
|
var flashvars = (aElement.getAttribute("flashvars") || "").replace(/(&ap=)true/gi, "$1false")
|
||||||
|
aElement.setAttribute("flashvars", flashvars);
|
||||||
|
TRACE("handleElement msnplayer - tag: %s id: %s ")(aElement.localName, aElement.id);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
// BBC
|
||||||
|
handlers.add(function handleBBC(aElement){
|
||||||
|
|
||||||
|
const document = aElement.ownerDocument, window = document.defaultView, jsWin = window.wrappedJSObject;
|
||||||
|
if (jsWin.embeddedMedia && jsWin.embeddedMedia.players)
|
||||||
|
for (var player of jsWin.embeddedMedia.players || []) if (player._swf && player._swf.id == aElement.id) {
|
||||||
|
TRACE("handleElement BBC - embeddedMedia.players id: %s autoplay: %s")(player._swf.id, player._settings.autoplay);
|
||||||
|
Object.defineProperty(player._settings, "autoplay", {get: () => false});
|
||||||
|
player.setData = (info) => ( info.data && (info.data.autoPlayFirstItem = false), player.__proto__.setData.call(player, info));
|
||||||
|
window.setTimeout(function(){ player._settings.autoplay = true; handlers.remove(window, handleBBC); }, 2500);
|
||||||
|
return true;
|
||||||
|
};
|
||||||
|
});
|
||||||
|
// metacafe
|
||||||
|
handlers.add(function handleMetacafe(aElement){
|
||||||
|
|
||||||
|
if (aElement.localName == "object" && (aElement.data || "").search(/s.mcstatic.com\/Flash\/.*\.swf/) != -1)
|
||||||
|
{
|
||||||
|
var param = aElement.querySelector("#" + aElement.id + ">param[name=flashvars]");
|
||||||
|
param.value = param.value.replace(/&beacons=.*(?=&)/,"");
|
||||||
|
var id = param.value.match(/itemID=([^&]*)(?=&)/)[1];
|
||||||
|
var data = aElement.data;
|
||||||
|
aElement.data = "http://www.metacafe.com/fplayer/" + id + "/.swf";
|
||||||
|
function onClick(aEvent){
|
||||||
|
aEvent.stopImmediatePropagation();
|
||||||
|
aElement.removeEventListener(aEvent.type, onClick, true);
|
||||||
|
aElement.data = data;
|
||||||
|
}
|
||||||
|
aElement.addEventListener("mouseup", onClick, true);
|
||||||
|
TRACE("handleElement metacafe - data: %s")(aElement.data);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
// jwplayer
|
||||||
|
handlers.add(function handleJWPlayer(aElement){ // return;
|
||||||
|
|
||||||
|
const document = aElement.ownerDocument, window = document.defaultView, jsWin = window.wrappedJSObject;
|
||||||
|
aElement.id || (aElement.id = "dummyid" + (++dummyid % 100));
|
||||||
|
if (jsWin.jwplayer && !aElement.querySelector("[id='" + aElement.id + "']>param[name=flashvars]"))
|
||||||
|
{
|
||||||
|
if (testUserInput(aElement, 5000)) return !TRACE("jwplayer released...")();
|
||||||
|
function closest(node, selector) { while(node && !node.matches(selector)) node = node.parentElement; return node};
|
||||||
|
var res, ar, player = jsWin.jwplayer(aElement).config ?
|
||||||
|
jsWin.jwplayer(aElement) : (res = closest(aElement, ".jwplayer")) ? jsWin.jwplayer(res) : null;
|
||||||
|
|
||||||
|
if (player && (player.config || (player.config = player.getConfig())) && player.config.autostart != false)
|
||||||
|
{
|
||||||
|
TRACE("handleElement jwplayer setup - id: %s autostart: %s")(player.id, player.config.autostart);
|
||||||
|
|
||||||
|
player.config.autoStart = player.config.autostart = false;
|
||||||
|
(ar = player.config.aspectratio) && ar.indexOf("%") != -1 && (player.config.aspectratio = "100:" + ar.slice(0,-1));
|
||||||
|
aElement.localName == "video" && window.setTimeout(() => aElement.src = "");
|
||||||
|
player.setup(player.config);
|
||||||
|
}
|
||||||
|
TRACE("handleElement jwplayer - id: %s config: %s")((player && player.id), !!(player && player.config));
|
||||||
|
return player && player.config;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
// flowplayer
|
||||||
|
handlers.add(function handleFlowplayer(aElement){ //return;
|
||||||
|
|
||||||
|
aElement.id || (aElement.id = "dummyid" + (++dummyid % 100));
|
||||||
|
var param = aElement.querySelector("[id='" + aElement.id + "']>param[name=flashvars]")
|
||||||
|
var config, flashvars = param ? param.value : aElement.getAttribute("flashvars");
|
||||||
|
if (flashvars && flashvars.search(/^config={/) != -1)
|
||||||
|
if ((config = JSON.parse(flashvars.slice(7).replace(/'/g,'"').replace(/("autoPlay":)true/g,"$1false"))) && (config.playlist || config.clip))
|
||||||
|
{
|
||||||
|
var playlist = (config.playlist = (config.playlist || [config.clip]));
|
||||||
|
(typeof playlist[0] == 'string') && (playlist[0] = {url: playlist[0]});
|
||||||
|
playlist[0].autoPlay = (typeof playlist[0].url == 'string' && playlist[0].url.search(/(.png|.jpg)/) != -1);
|
||||||
|
(playlist[0].autoBuffering && (playlist[0].autoBuffering = false)) || (playlist[0].autoPlay && playlist[1] && (playlist[1].autoPlay = false));
|
||||||
|
flashvars = "config=" + JSON.stringify(config);
|
||||||
|
param ? param.value = flashvars : aElement.setAttribute("flashvars", flashvars);
|
||||||
|
if (aElement.data) aElement.data = aElement.data;
|
||||||
|
TRACE("handleElement flowplayer - id: %s flashvars: %s")(aElement.id, flashvars);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
// aol
|
||||||
|
handlers.add(function handleAol(aElement){
|
||||||
|
|
||||||
|
if (aElement.localName == "object" && (aElement.data || "").search(/cdn(-ssl)?.vidible.tv\/.*\.swf/) != -1)
|
||||||
|
{
|
||||||
|
aElement.id || (aElement.id = "dummyid" + (++dummyid % 100));
|
||||||
|
var param = aElement.querySelector("[id='" + aElement.id + "']>param[name=flashvars]");
|
||||||
|
if (param)
|
||||||
|
{
|
||||||
|
param.value = param.value.replace(/(initialization%22%3A%22)autoplay/, "$1click");
|
||||||
|
aElement.wrappedJSObject.doPlay = function() { TRACE("doNothing - doPlay")()};
|
||||||
|
aElement.ownerDocument.defaultView.setTimeout(() => {delete aElement.wrappedJSObject.doPlay; TRACE("handleElement aol - removed doPlay")()}, 5000);
|
||||||
|
return !TRACE("handleElement aol - data: %s")(aElement.data);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
// general
|
||||||
|
var colValues = {"true":"false", "1":"0", "yes":"no", "on":"off", "y":"n", "Y":"N"}
|
||||||
|
function replace(match,p0,p1,p2) {
|
||||||
|
var res = p0 + (!p1 && p2 && colValues[p2] ? colValues[p2] : !!p1);
|
||||||
|
TRACE(">>> handleElement replace - match: %s p0: %s p1: %s p2: %s res: %s <<<")(match, p0, p1, p2, res);
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
function modifyParams(str) { return (str || "").replace(/((no)?auto_?(?:play|start|run)\w*(?:["']?\s?[=:]\s?["']?|%22%3A(?!\W)))(true|1|yes|y|on|null|)(?=\W|$)/gi, replace)};
|
||||||
|
|
||||||
|
function handleGeneral(aElement){
|
||||||
|
|
||||||
|
if (aElement.localName != "object" && aElement.localName != "embed")
|
||||||
|
return;
|
||||||
|
|
||||||
|
var data = modifyParams(aElement.getAttribute("data"));
|
||||||
|
if (data != aElement.getAttribute("data") && data != "")
|
||||||
|
{
|
||||||
|
aElement.setAttribute("data", data);
|
||||||
|
TRACE("handleElement - %s.data: %s")(aElement.localName, data);
|
||||||
|
}
|
||||||
|
|
||||||
|
var flashvars = modifyParams(aElement.getAttribute("flashvars")); // cnn | (remove) neulion nhl embed
|
||||||
|
if (aElement.localName == "embed" )
|
||||||
|
{
|
||||||
|
aElement.setAttribute("menu","true");
|
||||||
|
|
||||||
|
if (flashvars == aElement.getAttribute("flashvars") || flashvars == "" )
|
||||||
|
{
|
||||||
|
flashvars += "&autoplay=false&autostart=false&autoPlay=0"; // youtube(autoplay) | ustream live
|
||||||
|
aElement.setAttribute("play","false"); // basic flash
|
||||||
|
}
|
||||||
|
aElement.setAttribute("flashvars", flashvars);
|
||||||
|
if (aElement.src)
|
||||||
|
{
|
||||||
|
var src = modifyParams(aElement.src); // embed dailymotion
|
||||||
|
if (src != aElement.src) {
|
||||||
|
aElement.src = src;
|
||||||
|
var parent = aElement.parentNode, next = aElement.nextSibling; // workaround for chrome embed.src =
|
||||||
|
if (aElement.src && parent) { aElement.remove(); parent.insertBefore(aElement.wrappedJSObject, next); };
|
||||||
|
TRACE("handleElement embed.src: %s")(src);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
aElement.id || (aElement.id = "dummyid" + (++dummyid % 100)); // [id =''] works with numeric ids!!!
|
||||||
|
var param = aElement.querySelector("[id='" + aElement.id + "']>param[name=flashvars], [id='" + aElement.id + "']>param[name=FlashVars], [id='" + aElement.id + "']>param[name=flashVars]");
|
||||||
|
if (param)
|
||||||
|
{
|
||||||
|
flashvars = modifyParams(param.value); // ustream(autoplay=false) | bbc / metacafe.embed(no) / tv.com(vid)
|
||||||
|
if (flashvars == param.value) {
|
||||||
|
if (flashvars.search(/auto(_?play(vid)?|start|run)["']?\s?[=:]/i) != -1) return; // false already exists...
|
||||||
|
flashvars += "&autoplay=false&autoPlay=0&autoStart=0&_autoPlay=no&auto_start=off"; // justin.tv(autoPlay=0)|nba.com(autostart)|espn(!autoplay)|discovery(_autoPlay=no)|56.com(auto_start)
|
||||||
|
}
|
||||||
|
param.value = flashvars;
|
||||||
|
if (aElement.data) aElement.data = aElement.data;
|
||||||
|
}
|
||||||
|
(param = aElement.querySelector("[id='" + aElement.id + "']>param[name=play]")) && (param.value = false);
|
||||||
|
if (param && aElement.data) aElement.data = aElement.data;
|
||||||
|
TRACE("handleElement - id: %s %s.flashvars: %s")(aElement.id, aElement.localName, flashvars);
|
||||||
|
};
|
||||||
|
|
||||||
|
handlers.add(handleGeneral);
|
||||||
|
};
|
24
AutoplayStopper/script/selector.css
Normal file
24
AutoplayStopper/script/selector.css
Normal file
|
@ -0,0 +1,24 @@
|
||||||
|
object[classid="clsid:D27CDB6E-AE6D-11cf-96B8-444553540000"]:not([flashstopped]),
|
||||||
|
object[classid*=":d27cdb6e-ae6d-11cf-96b8-444553540000"]:not([flashstopped]),
|
||||||
|
object[codebase*="swflash.cab"]:not([flashstopped]),
|
||||||
|
object[data*=".swf"]:not([flashstopped]),
|
||||||
|
object[type="application/x-shockwave-flash"]:not([flashstopped]),
|
||||||
|
object[src*=".swf"]:not([flashstopped]),
|
||||||
|
embed[type="application/x-shockwave-flash"]:not([flashstopped]),
|
||||||
|
embed[src*=".swf"]:not([flashstopped]),
|
||||||
|
embed[allowscriptaccess]:not([flashstopped]),
|
||||||
|
embed[flashvars]:not([flashstopped]),
|
||||||
|
embed[wmode]:not([flashstopped]),
|
||||||
|
lima-video:not([flashstopped]),
|
||||||
|
/*--------- !(html5media) ---------*/
|
||||||
|
*:not([flashstopped_p]) > video,
|
||||||
|
video:not([flashstopped]),
|
||||||
|
audio:not([flashstopped]) {
|
||||||
|
border: 5px red solid !important;
|
||||||
|
box-sizing: border-box;
|
||||||
|
}
|
||||||
|
|
||||||
|
*[flashstopped] {
|
||||||
|
border: 3px blue solid !important;
|
||||||
|
box-sizing: border-box;
|
||||||
|
}
|
9
AutoplayStopper/script/userhandler.js
Normal file
9
AutoplayStopper/script/userhandler.js
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
|
||||||
|
// userhandler.js - handler that doesn't get overwritten by new versions
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
function registerUserHandlers(handlers, TRACE)
|
||||||
|
{
|
||||||
|
// add your handlers here...
|
||||||
|
};
|
BIN
AutoplayStopper/skin/IDR_CLOSE_DIALOG
Normal file
BIN
AutoplayStopper/skin/IDR_CLOSE_DIALOG
Normal file
Binary file not shown.
After Width: | Height: | Size: 139 B |
BIN
AutoplayStopper/skin/IDR_CLOSE_DIALOG_H
Normal file
BIN
AutoplayStopper/skin/IDR_CLOSE_DIALOG_H
Normal file
Binary file not shown.
After Width: | Height: | Size: 214 B |
1
AutoplayStopper/skin/arrow_down.svg
Normal file
1
AutoplayStopper/skin/arrow_down.svg
Normal file
|
@ -0,0 +1 @@
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" width="24px" height="12px" viewBox="0 0 24 12" fill="#757575"><g><path d="M 0 0 L 24 0 L 12 12 z"/></g></svg>
|
After Width: | Height: | Size: 150 B |
157
AutoplayStopper/skin/devtools.html
Normal file
157
AutoplayStopper/skin/devtools.html
Normal file
|
@ -0,0 +1,157 @@
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<script src="/utils.js"></script>
|
||||||
|
<script src="/devtools.js"></script>
|
||||||
|
|
||||||
|
<style type="text/css">
|
||||||
|
body {
|
||||||
|
margin: 0.5% 10%;
|
||||||
|
font-family: "Segoe UI", Tahoma, sans-serif;
|
||||||
|
color: #4c4c4c;
|
||||||
|
}
|
||||||
|
|
||||||
|
label.settings-label {
|
||||||
|
min-width: 80px;
|
||||||
|
font-size: initial;
|
||||||
|
font-weight: lighter;
|
||||||
|
}
|
||||||
|
|
||||||
|
div.settings-row {
|
||||||
|
margin: 6px 18px;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
input.settings-checkbox {
|
||||||
|
align-self: flex-end;
|
||||||
|
-webkit-margin-start: 0px;
|
||||||
|
-webkit-margin-end: 10px;
|
||||||
|
margin-inline-start: 0px;
|
||||||
|
margin-inline-end: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
button {
|
||||||
|
-webkit-margin-end: 3px;
|
||||||
|
white-space: nowrap;
|
||||||
|
}
|
||||||
|
|
||||||
|
button:disabled {
|
||||||
|
color: #a0a0a0;
|
||||||
|
}
|
||||||
|
|
||||||
|
button:not([disabled]) {
|
||||||
|
color: #4c4c4c;
|
||||||
|
}
|
||||||
|
|
||||||
|
span.input {
|
||||||
|
min-width: 450px;
|
||||||
|
max-width: 85%;
|
||||||
|
border: 1px solid rgb(170, 170, 170);
|
||||||
|
padding: 1px 2px;
|
||||||
|
-webkit-appearance: textfield;
|
||||||
|
font-size: 13.3333px;
|
||||||
|
white-space: nowrap;
|
||||||
|
overflow: hidden;
|
||||||
|
height: 17px; /* fff */
|
||||||
|
}
|
||||||
|
|
||||||
|
hr {
|
||||||
|
border-color: rgb(210, 210, 210);
|
||||||
|
border-top-width: 0px;
|
||||||
|
margin-bottom: 1em;
|
||||||
|
margin-top: .9em;
|
||||||
|
}
|
||||||
|
|
||||||
|
h2 {
|
||||||
|
font-weight: 400;
|
||||||
|
margin-top: 15px; /* fff */
|
||||||
|
}
|
||||||
|
|
||||||
|
#splitter {
|
||||||
|
display: flex;
|
||||||
|
flex-flow: column nowrap;
|
||||||
|
overflow: hidden;
|
||||||
|
height: 0;
|
||||||
|
min-height: calc(40px + 6.5em);
|
||||||
|
max-height: calc(100vh - 10px);
|
||||||
|
margin-bottom: 1px;
|
||||||
|
resize: vertical;
|
||||||
|
background: linear-gradient(to right bottom, transparent 50%,
|
||||||
|
rgba(0, 0, 0, 0.1) 50%) 100% 100% / 16px 15px no-repeat;
|
||||||
|
}
|
||||||
|
|
||||||
|
#selector {
|
||||||
|
height: 100%;
|
||||||
|
width: 800px;
|
||||||
|
color: darkslateblue;
|
||||||
|
border: 1px solid rgb(170, 170, 170);
|
||||||
|
resize: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
:focus {
|
||||||
|
outline: rgb(170, 170, 170) auto 1px;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<div id="splitter">
|
||||||
|
<div class="settings-row" style="margin:0 0px;">
|
||||||
|
<img src="../icons/icon48.png" style="margin: 10px; height: 32px;" />
|
||||||
|
<h2>AutoplayStopper</h2>
|
||||||
|
<h2 id="ver" style="font-weight: 200; -webkit-margin-start: 10px; margin-inline-start: 10px;">1.0.0</h2>
|
||||||
|
</div>
|
||||||
|
<hr style="width: calc(100% - 2px); margin-top: 6px; margin-bottom: 0px"/>
|
||||||
|
<h2 i18n-content="selector">Selector</h2>
|
||||||
|
<div class="settings-row" style="height: 100%;">
|
||||||
|
<label class="settings-label" i18n-content="text" style="margin-bottom: auto;">Text</label>
|
||||||
|
<textarea id="selector" spellcheck="false"></textarea>
|
||||||
|
</div>
|
||||||
|
<div class="settings-row" style="margin-bottom: calc(.9em - 1px);">
|
||||||
|
<label class="settings-label" i18n-content="actions">Actions</label>
|
||||||
|
<button id="reset" i18n-content="reset">Reset</button>
|
||||||
|
<button id="apply" disabled="" i18n-content="apply"></button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<hr style="margin-top:0px"/>
|
||||||
|
<h2 i18n-content="script">Script</h2>
|
||||||
|
<div class="settings-row">
|
||||||
|
<label class="settings-label" i18n-content="path">Path:</label>
|
||||||
|
<span id="path-input" class="input" contenteditable="true" spellcheck="false"></span>
|
||||||
|
</div>
|
||||||
|
<div class="settings-row">
|
||||||
|
<label class="settings-label" i18n-content="actions">Actions:</label>
|
||||||
|
<button id="edit-button" i18n-content="edit">Edit</button>
|
||||||
|
<button id="load-button" i18n-content="load">Load</button>
|
||||||
|
<button id="export-button" i18n-content="export">Export...</button>
|
||||||
|
<button id="path-button" i18n-content="setPath" disabled>Set Path</button>
|
||||||
|
<button id="select-button" i18n-content="selectFile" >File...</button>
|
||||||
|
<label id="filename" class="settings-label" style="-webkit-margin-start:10px;margin-inline-start:10px;"/>
|
||||||
|
</div>
|
||||||
|
<hr />
|
||||||
|
<h2 i18n-content="userScript">User Script</h2>
|
||||||
|
<div class="settings-row">
|
||||||
|
<label class="settings-label" i18n-content="path">Path:</label>
|
||||||
|
<span id="upath-input" class="input" contenteditable="true" spellcheck="false"></span>
|
||||||
|
</div>
|
||||||
|
<div class="settings-row">
|
||||||
|
<label class="settings-label" i18n-content="actions">Actions:</label>
|
||||||
|
<button id="uedit-button" i18n-content="edit">Edit</button>
|
||||||
|
<button id="uload-button" i18n-content="load">Load</button>
|
||||||
|
<button id="uexport-button" i18n-content="export">Export...</button>
|
||||||
|
<button id="upath-button" i18n-content="setPath" disabled>Set Path</button>
|
||||||
|
<button id="uselect-button" i18n-content="selectFile">File...</button>
|
||||||
|
<label id="ufilename" class="settings-label" style="-webkit-margin-start:10px;margin-inline-start:10px;"/>
|
||||||
|
</div>
|
||||||
|
<hr />
|
||||||
|
<div class="settings-row">
|
||||||
|
<input id="overwrite" class="settings-checkbox" type="checkbox" />
|
||||||
|
<label class="settings-label" i18n-content="overwrite">Disable overwrite on update</label>
|
||||||
|
</div>
|
||||||
|
<div class="settings-row">
|
||||||
|
<input id="debug" class="settings-checkbox" type="checkbox" />
|
||||||
|
<label class="settings-label" i18n-content="debug">Log debug info</label>
|
||||||
|
</div>
|
||||||
|
<a id="export-link" hidden="true"></a>
|
||||||
|
</body>
|
||||||
|
</html>
|
615
AutoplayStopper/skin/options.css
Normal file
615
AutoplayStopper/skin/options.css
Normal file
|
@ -0,0 +1,615 @@
|
||||||
|
|
||||||
|
/* chrome://resources/css/list.css */
|
||||||
|
|
||||||
|
list,
|
||||||
|
grid {
|
||||||
|
display: block;
|
||||||
|
outline: none;
|
||||||
|
overflow: auto;
|
||||||
|
position: relative; /* Make sure that item offsets are relative to the
|
||||||
|
list. */
|
||||||
|
}
|
||||||
|
|
||||||
|
list > *,
|
||||||
|
grid > * {
|
||||||
|
-webkit-user-select: none;
|
||||||
|
background-color: rgba(255, 255, 255, 0);
|
||||||
|
border: 1px solid rgba(255, 255, 255, 0); /* transparent white */
|
||||||
|
border-radius: 2px;
|
||||||
|
cursor: default;
|
||||||
|
line-height: 20px;
|
||||||
|
margin: -1px 0;
|
||||||
|
overflow: hidden;
|
||||||
|
padding: 0 3px;
|
||||||
|
position: relative; /* to allow overlap */
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
white-space: pre;
|
||||||
|
}
|
||||||
|
|
||||||
|
list > * {
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
|
||||||
|
grid > * {
|
||||||
|
display: inline-block;
|
||||||
|
}
|
||||||
|
|
||||||
|
list > [lead],
|
||||||
|
grid > [lead] {
|
||||||
|
border-color: transparent;
|
||||||
|
}
|
||||||
|
|
||||||
|
list:focus > [lead],
|
||||||
|
grid:focus > [lead] {
|
||||||
|
border-color: hsl(214, 91%, 65%);
|
||||||
|
z-index: 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
list > [anchor],
|
||||||
|
grid > [anchor] {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
list:not([disabled]) > :hover,
|
||||||
|
grid:not([disabled]) > :hover {
|
||||||
|
background-color: hsl(214, 91%, 97%);
|
||||||
|
border-color: hsl(214, 91%, 85%);
|
||||||
|
z-index: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
list > [selected],
|
||||||
|
grid > [selected] {
|
||||||
|
background-color: hsl(0, 0%, 90%);
|
||||||
|
background-image: -webkit-linear-gradient(rgba(255, 255, 255, 0.8),
|
||||||
|
rgba(255, 255, 255, 0));
|
||||||
|
border-color: hsl(0, 0%, 85%);
|
||||||
|
z-index: 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
list:focus > [selected],
|
||||||
|
grid:focus > [selected] {
|
||||||
|
background-color: hsl(214, 91%, 89%);
|
||||||
|
border-color: hsl(214, 91%, 65%);
|
||||||
|
}
|
||||||
|
|
||||||
|
list:focus > [lead][selected],
|
||||||
|
list > [selected]:hover,
|
||||||
|
grid:focus > [lead][selected],
|
||||||
|
grid > [selected]:hover {
|
||||||
|
background-color: hsl(214, 91%, 87%);
|
||||||
|
border-color: hsl(214, 91%, 65%);
|
||||||
|
}
|
||||||
|
|
||||||
|
list > .spacer,
|
||||||
|
grid > .spacer {
|
||||||
|
border: 0;
|
||||||
|
box-sizing: border-box;
|
||||||
|
display: block;
|
||||||
|
margin: 0;
|
||||||
|
overflow: hidden;
|
||||||
|
visibility: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
list :-webkit-any(
|
||||||
|
input[type='input'],
|
||||||
|
input[type='password'],
|
||||||
|
input[type='search'],
|
||||||
|
input[type='text'],
|
||||||
|
input[type='url']),
|
||||||
|
list :-webkit-any(
|
||||||
|
button,
|
||||||
|
input[type='button'],
|
||||||
|
input[type='submit'],
|
||||||
|
select):not(.custom-appearance) {
|
||||||
|
line-height: normal;
|
||||||
|
margin: 0;
|
||||||
|
vertical-align: middle;
|
||||||
|
}
|
||||||
|
|
||||||
|
list input[type='text'] {
|
||||||
|
margin-left: -4px;
|
||||||
|
padding: 3px;
|
||||||
|
width: 190px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* chrome://resources/css/overlay.css */
|
||||||
|
|
||||||
|
[hidden] {
|
||||||
|
display: none !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.overlay {
|
||||||
|
-webkit-box-align: center;
|
||||||
|
-webkit-box-orient: vertical;
|
||||||
|
-webkit-box-pack: center;
|
||||||
|
-webkit-perspective: 1px;
|
||||||
|
-webkit-transition: 200ms opacity;
|
||||||
|
background-color: rgba(255, 255, 255, 0.75);
|
||||||
|
/* bottom: 50px; */
|
||||||
|
display: -webkit-box;
|
||||||
|
left: 0;
|
||||||
|
overflow: auto;
|
||||||
|
padding: 20px;
|
||||||
|
position: fixed;
|
||||||
|
right: 0;
|
||||||
|
top: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.overlay.transparent .page {
|
||||||
|
transform: scale(0.99) translateY(-20px);
|
||||||
|
}
|
||||||
|
|
||||||
|
.transparent {
|
||||||
|
opacity: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.overlay .page {
|
||||||
|
max-height: 335.5px;
|
||||||
|
/* display: -webkit-box;
|
||||||
|
-webkit-box-orient: vertical;
|
||||||
|
*/
|
||||||
|
-webkit-border-radius: 3px;
|
||||||
|
-webkit-box-orient: vertical;
|
||||||
|
-webkit-transition: 200ms transform;
|
||||||
|
background: white;
|
||||||
|
box-shadow: 0 4px 23px 5px rgba(0, 0, 0, 0.2), 0 2px 6px rgba(0,0,0,0.15);
|
||||||
|
color: #333;
|
||||||
|
display: -webkit-box;
|
||||||
|
min-width: 400px;
|
||||||
|
padding: 0px;
|
||||||
|
position: relative;
|
||||||
|
z-index: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.overlay .page .content-area {
|
||||||
|
-webkit-box-flex: 1;
|
||||||
|
overflow: auto;
|
||||||
|
overflow: overlay; /* auto */
|
||||||
|
padding: 0px 20px 0px; /* 0px 17px 6px */
|
||||||
|
position: relative;
|
||||||
|
min-height: 226px;
|
||||||
|
outline: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.overlay .page > .close-button {
|
||||||
|
background-image: url(IDR_CLOSE_DIALOG);
|
||||||
|
background-position: center;
|
||||||
|
background-repeat: no-repeat;
|
||||||
|
height: 14px;
|
||||||
|
position: absolute;
|
||||||
|
right: 7px;
|
||||||
|
top: 7px;
|
||||||
|
width: 14px;
|
||||||
|
z-index: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.overlay .page > .close-button:hover {
|
||||||
|
background-image: url(IDR_CLOSE_DIALOG_H);
|
||||||
|
}
|
||||||
|
|
||||||
|
.overlay .page > .close-button:focus:not(:hover) {
|
||||||
|
background-image: url(IDR_CLOSE_DIALOG_H);
|
||||||
|
filter: opacity(75%);
|
||||||
|
}
|
||||||
|
|
||||||
|
div.settings-row {
|
||||||
|
margin: 2px 18px;
|
||||||
|
display: flex;
|
||||||
|
padding: 2px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
label.settings-label {
|
||||||
|
width: 130px;
|
||||||
|
display: inherit;
|
||||||
|
}
|
||||||
|
|
||||||
|
.overlay .page h2 {
|
||||||
|
padding: 4px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* chrome://settings/contentExceptions#... */
|
||||||
|
|
||||||
|
body {
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
|
||||||
|
.hbox {
|
||||||
|
-webkit-box-orient: horizontal;
|
||||||
|
display: -webkit-box;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vbox {
|
||||||
|
-webkit-box-orient: vertical;
|
||||||
|
display: -webkit-box;
|
||||||
|
}
|
||||||
|
|
||||||
|
.box-align-center {
|
||||||
|
-webkit-box-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.stretch {
|
||||||
|
-webkit-box-flex: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.raw-button,
|
||||||
|
.raw-button:hover,
|
||||||
|
.raw-button:active {
|
||||||
|
background-color: transparent;
|
||||||
|
background-repeat: no-repeat;
|
||||||
|
border: none;
|
||||||
|
box-shadow: none !important;
|
||||||
|
min-width: 0;
|
||||||
|
min-height: 0;
|
||||||
|
padding: 1px 6px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.page list {
|
||||||
|
/* Min height is a multiple of the list item height (32) */
|
||||||
|
box-sizing: content-box;
|
||||||
|
min-height: 192px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.option {
|
||||||
|
margin-top: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.settings-list,
|
||||||
|
.settings-list-empty {
|
||||||
|
border: 1px solid #d9d9d9;
|
||||||
|
border-radius: 2px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Editable List properties */
|
||||||
|
list > * {
|
||||||
|
-webkit-box-align: center;
|
||||||
|
-webkit-transition: 150ms background-color;
|
||||||
|
border: none;
|
||||||
|
border-radius: 0; /* TODO(dbeam): Is this necessary? */
|
||||||
|
box-sizing: border-box;
|
||||||
|
display: -webkit-box;
|
||||||
|
height: 32px;
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
list > .spacer {
|
||||||
|
/* The above height rule should not apply to spacers. When redraw is called
|
||||||
|
on the list they will be given an explicit element height but this ensures
|
||||||
|
they have 0 height to begin with. */
|
||||||
|
height: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
list:not([disabled]) > :hover {
|
||||||
|
background-color: rgb(228, 236, 247);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Note: If this becomes the list style for other WebUI pages these rules can be
|
||||||
|
* simplified (since they wont need to override other rules). */
|
||||||
|
|
||||||
|
list:not([has-element-focus]) > [selected],
|
||||||
|
list:not([has-element-focus]) > [lead][selected] {
|
||||||
|
background-color: #d0d0d0;
|
||||||
|
background-image: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
list[has-element-focus] > [selected],
|
||||||
|
list[has-element-focus] > [lead][selected],
|
||||||
|
list:not([has-element-focus]) > [selected]:hover,
|
||||||
|
list:not([has-element-focus]) > [selected][lead]:hover {
|
||||||
|
background-color: rgb(187, 206, 233);
|
||||||
|
background-image: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.settings-list[has-element-focus] > [lead],
|
||||||
|
.settings-list[has-element-focus] > [lead][selected] {
|
||||||
|
border-bottom: 1px solid rgb(120, 146, 180);
|
||||||
|
border-top: 1px solid rgb(120, 146, 180);
|
||||||
|
}
|
||||||
|
|
||||||
|
.settings-list[has-element-focus] > [lead]:nth-child(2),
|
||||||
|
.settings-list[has-element-focus] > [lead][selected]:nth-child(2) {
|
||||||
|
border-top: 1px solid transparent;
|
||||||
|
}
|
||||||
|
|
||||||
|
.settings-list[has-element-focus] > [lead]:nth-last-child(2),
|
||||||
|
.settings-list[has-element-focus] > [lead][selected]:nth-last-child(2) {
|
||||||
|
border-bottom: 1px solid transparent;
|
||||||
|
}
|
||||||
|
|
||||||
|
.settings-list[disabled] > [lead][selected],
|
||||||
|
.settings-list[disabled]:focus > [lead][selected] {
|
||||||
|
border: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
list .deletable-item {
|
||||||
|
-webkit-box-align: center;
|
||||||
|
outline: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
list .deletable-item > :first-child {
|
||||||
|
-webkit-box-align: center;
|
||||||
|
-webkit-box-flex: 1;
|
||||||
|
-webkit-padding-end: 5px;
|
||||||
|
-moz-padding-end: 5px;
|
||||||
|
display: -webkit-box;
|
||||||
|
}
|
||||||
|
|
||||||
|
list .row-delete-button {
|
||||||
|
-webkit-transition: 150ms opacity;
|
||||||
|
background-color: transparent;
|
||||||
|
background-image:
|
||||||
|
url();
|
||||||
|
border: none;
|
||||||
|
display: block;
|
||||||
|
height: 16px;
|
||||||
|
opacity: 1;
|
||||||
|
width: 16px;
|
||||||
|
}
|
||||||
|
|
||||||
|
list > *:not(:hover):not([selected]):not([lead]) .row-delete-button,
|
||||||
|
list:not([has-element-focus]) > *:not(:hover):not([selected])
|
||||||
|
.row-delete-button,
|
||||||
|
list[disabled] .row-delete-button,
|
||||||
|
list .row-delete-button[disabled] {
|
||||||
|
opacity: 0;
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
list .row-delete-button:hover {
|
||||||
|
background-image:
|
||||||
|
url() !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
list .row-delete-button:active {
|
||||||
|
background-image:
|
||||||
|
url()
|
||||||
|
,
|
||||||
|
url()
|
||||||
|
;
|
||||||
|
}
|
||||||
|
|
||||||
|
list .static-text {
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
white-space: nowrap;
|
||||||
|
/* outline: none; bYO! inner rect on selection*/
|
||||||
|
}
|
||||||
|
|
||||||
|
list[type='text'][inlineeditable] input {
|
||||||
|
box-sizing: border-box;
|
||||||
|
margin: 0;
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
list > :not([editing]) [displaymode='edit'] {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
list > [editing] [displaymode='static'] {
|
||||||
|
/* Don't use display:none or visibility:hidden because we need to keep an
|
||||||
|
* element focusable.
|
||||||
|
* We shrink only height. We don't shrink width to avoid to change the size
|
||||||
|
* of containing boxes. */
|
||||||
|
border-bottom: 0 !important;
|
||||||
|
border-top: 0 !important;
|
||||||
|
height: 0 !important;
|
||||||
|
width: auto;
|
||||||
|
margin-bottom: 0 !important;
|
||||||
|
margin-top: 0 !important;
|
||||||
|
overflow: hidden;
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
list > [editing] input:invalid {
|
||||||
|
background-color: pink;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* UI Controls */
|
||||||
|
|
||||||
|
/* LIST */ /* bYO! top level outline... */
|
||||||
|
.settings-list[has-element-focus]:after {
|
||||||
|
outline: 1px solid rgba(0, 128, 256, 0.5);
|
||||||
|
outline-offset: -1px;
|
||||||
|
content: "";
|
||||||
|
position: absolute;
|
||||||
|
top: 0px;
|
||||||
|
left: 0px;
|
||||||
|
bottom: 0px;
|
||||||
|
right: 0px;
|
||||||
|
pointer-events: none;
|
||||||
|
z-index: 3;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* chrome://settings/contentExceptions#... */
|
||||||
|
|
||||||
|
#content-settings-exceptions-area {
|
||||||
|
min-width: 540px;
|
||||||
|
width: 540px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.exception-pattern {
|
||||||
|
-webkit-box-flex: 1;
|
||||||
|
-webkit-margin-end: 10px;
|
||||||
|
-moz-margin-end: 10px;
|
||||||
|
-webkit-margin-start: 14px;
|
||||||
|
-moz-margin-start: 14px;
|
||||||
|
width: 200px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.exception-setting {
|
||||||
|
display: inline-block;
|
||||||
|
width: 120px;
|
||||||
|
}
|
||||||
|
|
||||||
|
select.exception-setting {
|
||||||
|
vertical-align: middle;
|
||||||
|
-webkit-padding-start: 3px;
|
||||||
|
width: calc(120px + 4px);
|
||||||
|
}
|
||||||
|
|
||||||
|
.overruled .exception-setting {
|
||||||
|
width: calc(120px - 24px);
|
||||||
|
}
|
||||||
|
|
||||||
|
.overruled .overruleable:focus {
|
||||||
|
outline:auto !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.overruled .overruleable {
|
||||||
|
text-decoration: line-through;
|
||||||
|
}
|
||||||
|
|
||||||
|
#exception-column-headers {
|
||||||
|
margin-left: 17px;
|
||||||
|
-webkit-margin-start: 17px;
|
||||||
|
display: -webkit-box;
|
||||||
|
margin-top: 10px; /* bYO! 6 lines on FF */
|
||||||
|
line-height: 16px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#exception-column-headers > div {
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
|
||||||
|
#exception-pattern-column {
|
||||||
|
-webkit-box-flex: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.exception-value-column-header {
|
||||||
|
width: 145px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#content-settings-exceptions-area list {
|
||||||
|
/*margin-bottom: 10px; bYO! keep list area consistent with scrolling */
|
||||||
|
margin-top: 6px;
|
||||||
|
}
|
||||||
|
/****** new ui compat... ******/
|
||||||
|
button:not(.custom-appearance) {
|
||||||
|
--google-blue-500: #528de0; /* #3b72ce #4285f4 #0a84ff */
|
||||||
|
font-weight: 500;
|
||||||
|
line-height: 100%; /* 154% */
|
||||||
|
padding: 8px 10px;
|
||||||
|
color: white !important;
|
||||||
|
border: 1px solid var(--google-blue-500);
|
||||||
|
border-radius: 3px;
|
||||||
|
box-shadow: none !important;
|
||||||
|
font-family: inherit;
|
||||||
|
font-size: inherit;
|
||||||
|
text-shadow: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
button:enabled:not(.custom-appearance) {
|
||||||
|
background: var(--google-blue-500) !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
button:enabled:not(.custom-appearance):focus {
|
||||||
|
background: rgb(58, 117, 215) !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
select {
|
||||||
|
--md-arrow-width: 0.9em;
|
||||||
|
--md-arrow-offset: 0%;
|
||||||
|
--md-select-side-padding: 3px;
|
||||||
|
--md-select-width: 110px;
|
||||||
|
-webkit-appearance: none;
|
||||||
|
-moz-appearance: none;
|
||||||
|
/* -webkit-margin-end: calc(-1 * var(--md-select-side-padding)); */
|
||||||
|
/* -moz-margin-end: var(--md-select-side-padding); */
|
||||||
|
/* -webkit-padding-end: calc(var(--md-select-side-padding) + var(--md-arrow-offset) + var(--md-arrow-width) + 3px); */
|
||||||
|
-webkit-padding-start: var(--md-select-side-padding);
|
||||||
|
-moz-padding-start: calc(var(--md-select-side-padding) - 1px);
|
||||||
|
background: url(arrow_down.svg) calc(100% - var(--md-arrow-offset) - var(--md-select-side-padding)) center no-repeat !important;
|
||||||
|
background-image: url(arrow_down.svg) !important;
|
||||||
|
background-size: var(--md-arrow-width) !important;
|
||||||
|
border: none;
|
||||||
|
font-family: inherit;
|
||||||
|
font-size: inherit;
|
||||||
|
outline: none;
|
||||||
|
padding-bottom: 2px;
|
||||||
|
padding-top: 3px;
|
||||||
|
border-bottom: 1px solid rgb(224, 224, 224);
|
||||||
|
width: var(--md-select-width);
|
||||||
|
/* width: calc(var(--md-select-width, 200px) + 2 * var(--md-select-side-padding)); */
|
||||||
|
box-shadow: none !important;
|
||||||
|
text-shadow: none;
|
||||||
|
margin: 3px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
select:focus {
|
||||||
|
border-color: rgb(77, 144, 254);
|
||||||
|
}
|
||||||
|
|
||||||
|
select.exception-setting {
|
||||||
|
-webkit-padding-start: var(--md-select-side-padding);
|
||||||
|
width: calc(120px + var(--md-select-side-padding));
|
||||||
|
}
|
||||||
|
|
||||||
|
html > body {
|
||||||
|
height: 376px;
|
||||||
|
width: 500px;
|
||||||
|
margin: 16px 40px 14px 40px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.overlay {
|
||||||
|
background-color: rgba(255, 255, 255,.5);
|
||||||
|
bottom: 0px;
|
||||||
|
padding-top: 30px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.overlay .page {
|
||||||
|
max-height: 341.5px;
|
||||||
|
box-shadow: 0 0px 1px 0px rgba(0, 0, 0, 0.2), 0 1px 6px rgba(0,0,0,0.15);
|
||||||
|
}
|
||||||
|
|
||||||
|
body > div:first-child {
|
||||||
|
padding-bottom: 5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.action-area {
|
||||||
|
padding-top: .3em !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
label.settings-label {
|
||||||
|
width: 80px;
|
||||||
|
margin: 7px 0px;
|
||||||
|
}
|
||||||
|
|
||||||
|
input.settings-checkbox ~ label.settings-label {
|
||||||
|
-webkit-margin-start: 3px;
|
||||||
|
-moz-margin-start: 3px;
|
||||||
|
}
|
||||||
|
|
||||||
|
select ~ button.exceptions-list-button{
|
||||||
|
-webkit-margin-start: auto;
|
||||||
|
-moz-margin-start: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
input.settings-checkbox {
|
||||||
|
display: -moz-box;
|
||||||
|
}
|
||||||
|
|
||||||
|
button:enabled:not(.custom-appearance):not(:focus):hover {
|
||||||
|
border: 1px solid rgba(0, 0, 0, 0.1); /* #3b72ce */
|
||||||
|
}
|
||||||
|
|
||||||
|
button:disabled:not(.custom-appearance) {
|
||||||
|
border-color: rgba(80, 80, 80, 0.1);
|
||||||
|
}
|
||||||
|
|
||||||
|
list input[type="text"] {
|
||||||
|
font-size: inherit;
|
||||||
|
font-family: inherit;
|
||||||
|
}
|
||||||
|
|
||||||
|
#perms-type {
|
||||||
|
-moz-padding-start: 3px;
|
||||||
|
height: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
button::-moz-focus-inner {
|
||||||
|
border: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
select:-moz-focusring {
|
||||||
|
color: transparent;
|
||||||
|
text-shadow: 0 0 0 #000;
|
||||||
|
}
|
124
AutoplayStopper/skin/options.html
Normal file
124
AutoplayStopper/skin/options.html
Normal file
|
@ -0,0 +1,124 @@
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8" />
|
||||||
|
<link href="options.css" rel="stylesheet" type="text/css" />
|
||||||
|
<script src="/options.js"></script>
|
||||||
|
<style type="text/css">
|
||||||
|
|
||||||
|
body {
|
||||||
|
font-family: "Segoe UI", Tahoma, sans-serif;
|
||||||
|
width: 560px;
|
||||||
|
height: 400px;
|
||||||
|
}
|
||||||
|
|
||||||
|
h1 {
|
||||||
|
font-size: 1.5em;
|
||||||
|
}
|
||||||
|
|
||||||
|
h2 {
|
||||||
|
margin: 1em 17px;
|
||||||
|
font-size: 1.3em;
|
||||||
|
}
|
||||||
|
|
||||||
|
h1, h2, h3 {
|
||||||
|
user-select: none;
|
||||||
|
font-weight: normal;
|
||||||
|
line-height: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
hr {
|
||||||
|
border-style: solid;
|
||||||
|
color: #dddddd;
|
||||||
|
border-top-width: 0px;
|
||||||
|
margin-top: 1em;
|
||||||
|
margin-bottom: 1.5em;
|
||||||
|
margin-inline-start: initial;
|
||||||
|
margin-inline-end: initial;
|
||||||
|
}
|
||||||
|
|
||||||
|
input.settings-checkbox {
|
||||||
|
margin: auto 1px;
|
||||||
|
-webkit-margin-start: 0px;
|
||||||
|
margin-inline-start: 0px;
|
||||||
|
position: static;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<div style="text-align: center; display: block;">
|
||||||
|
<span i18n-content="version">Version</span>
|
||||||
|
<span id="extension-version"></span>
|
||||||
|
</div>
|
||||||
|
<h1> <span i18n-content="settings" style="position: relative; top: -.7em">Settings</span></h1>
|
||||||
|
<h2 i18n-content="autoplayTabLabel">Autoplay</h2>
|
||||||
|
|
||||||
|
<div class="settings-row">
|
||||||
|
<label for="autoplay-default" class="settings-label" i18n-content="defaultMode">Default Mode:</label>
|
||||||
|
<select id="autoplay-default">
|
||||||
|
<option i18n-content="disableAutoplay" value="2">Disable Autoplay</option>
|
||||||
|
<option i18n-content="allowAutoplay" value="1">Allow Autoplay</option>
|
||||||
|
</select>
|
||||||
|
<button class="exceptions-list-button" contenttype="autoplay" i18n-content="manageExceptions">Exceptions...</button>
|
||||||
|
</div>
|
||||||
|
<hr />
|
||||||
|
<h2 i18n-content="flashTabLabel">Flash</h2>
|
||||||
|
<div class="settings-row">
|
||||||
|
<label class="settings-label" for="flash-default" i18n-content="defaultMode">Default Mode:</label>
|
||||||
|
<select id="flash-default">
|
||||||
|
<option i18n-content="allowFlash" value="1">Allow Flash</option>
|
||||||
|
<option i18n-content="disableFlash" value="2">Disable Flash</option>
|
||||||
|
</select>
|
||||||
|
<button class="exceptions-list-button" contenttype="flash" i18n-content="manageExceptions">Exceptions...</button>
|
||||||
|
</div>
|
||||||
|
<div class="overlay transparent" hidden>
|
||||||
|
<div id="content-settings-exceptions-area" class="page">
|
||||||
|
<button class="close-button raw-button custom-appearance"></button>
|
||||||
|
<h2 contenttype="autoplay" i18n-content="autoplayExceptions">Autoplay exceptions</h2>
|
||||||
|
<h2 contenttype="flash" i18n-content="flashExceptions" hidden>Flash exceptions</h2>
|
||||||
|
<div class="content-area" tabindex="-1">
|
||||||
|
<div id="exception-column-headers">
|
||||||
|
<div id="exception-pattern-column" i18n-content="exceptionHostnameHeader">
|
||||||
|
Hostname</div>
|
||||||
|
<div id="exception-behavior-column" class="exception-value-column-header" i18n-content="exceptionBehaviorHeader">
|
||||||
|
Behavior</div>
|
||||||
|
</div>
|
||||||
|
<div contenttype="autoplay">
|
||||||
|
<list id="autoplay-list" aria-activedescendant="listitem-27" class="settings-list" inlineeditable="" mode="normal" role="list" tabindex="-1">
|
||||||
|
<div class="spacer" style="height: 0px;"></div>
|
||||||
|
<div role="listitem" id="listitem" class="deletable-item" aria-posinset="1" aria-setsize="3"><div><div class="exception-pattern weakrtl"><div id="perms-host" class="static-text overruleable" displaymode="static" tabindex="-1"></div><input id="perms-host-input" type="text" displaymode="edit" tabindex="0"></div><span id="perms-type" class="exception-setting overruleable" displaymode="static" tabindex="-1"></span><select id="perms-type-select" tabindex="0" class="exception-setting" aria-labelledby="exception-behavior-column" displaymode="edit"><option value="1" i18n-content="exceptionAllow">Allow</option><option value="2" i18n-content="exceptionBlock">Block</option><option value="3" i18n-content="exceptionPrompt">Block (strict)</option><option value="8" i18n-content="exceptionSession">Session</option><option value="0" i18n-content="exceptionUndefined">Undefined</option></select></div><button class="raw-button row-delete-button custom-appearance" tabindex="-1" title="Delete this item"></button></div>
|
||||||
|
<div role="listitem" id="inputitem" selected="selected" lead="lead" class="deletable-item"><div><div class="exception-pattern weakrtl"><input id="perms-host-input" type="text" tabindex="0" placeholder="http://example.com"></div><select id="perms-type-select" tabindex="0" class="exception-setting" aria-labelledby="exception-behavior-column"><option value="1" i18n-content="exceptionAllow">Allow</option><option value="2" i18n-content="exceptionBlock">Block</option><option value="3" i18n-content="exceptionPrompt">Block (strict)</option><option value="8" i18n-content="exceptionSession">Session</option><option value="0" i18n-content="exceptionUndefined">Undefined</option></select></div><button class="raw-button row-delete-button custom-appearance" tabindex="-1" title="Delete this item" disabled=""></button></div>
|
||||||
|
<div class="spacer" style="height: 0px;"></div>
|
||||||
|
</list>
|
||||||
|
</div>
|
||||||
|
<div contenttype="flash" hidden>
|
||||||
|
<list id="flash-list" aria-activedescendant="listitem-27" class="settings-list" inlineeditable="" mode="normal" role="list" tabindex="-1">
|
||||||
|
<div class="spacer" style="height: 0px;"></div>
|
||||||
|
<div role="listitem" id="listitem" class="deletable-item" aria-posinset="1" aria-setsize="3"><div><div class="exception-pattern weakrtl"><div id="perms-host" class="static-text overruleable" displaymode="static" tabindex="-1"></div><input id="perms-host-input" type="text" displaymode="edit" tabindex="0"></div><span id="perms-type" class="exception-setting overruleable" displaymode="static" tabindex="-1"></span><select id="perms-type-select" tabindex="0" class="exception-setting" aria-labelledby="exception-behavior-column" displaymode="edit"><option value="1" i18n-content="exceptionAllow">Allow</option><option value="2" i18n-content="exceptionBlock">Block</option><option value="0" i18n-content="exceptionUndefined">Undefined</option></select></div><button class="raw-button row-delete-button custom-appearance" tabindex="-1" title="Delete this item"></button></div>
|
||||||
|
<div role="listitem" id="inputitem" selected="selected" lead="lead" class="deletable-item"><div><div class="exception-pattern weakrtl"><input id="perms-host-input" type="text" tabindex="0" placeholder="http://example.com"></div><select id="perms-type-select" tabindex="0" class="exception-setting" aria-labelledby="exception-behavior-column"><option value="1" i18n-content="exceptionAllow">Allow</option><option value="2" i18n-content="exceptionBlock">Block</option><option value="0" i18n-content="exceptionUndefined">Undefined</option></select></div><button class="raw-button row-delete-button custom-appearance" tabindex="-1" title="Delete this item" disabled=""></button></div>
|
||||||
|
<div class="spacer" style="height: 0px;"></div>
|
||||||
|
</list>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="action-area" style="padding-top: .5em">
|
||||||
|
<div class="hbox stretch">
|
||||||
|
</div>
|
||||||
|
<button id="exceptions-clear" style="margin-left: 20px;" i18n-content="removeAll">Remove All</button>
|
||||||
|
<div class="action-area-right" style="-webkit-box-pack: end; display: -webkit-box;">
|
||||||
|
<div class="button-strip" reversed="" style="position:relative; top: -1.5em; right: 19px">
|
||||||
|
<button id="exceptions-apply" disabled i18n-content="apply">Apply</button>
|
||||||
|
<button id="exceptions-confirm" class="default-button" i18n-content="done">Done</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<hr />
|
||||||
|
<h2 i18n-content="advanced">Advanced</h2>
|
||||||
|
<div class="settings-row">
|
||||||
|
<input id="devtools" class="settings-checkbox" type="checkbox" />
|
||||||
|
<label class="settings-label" i18n-content="devtools">Devtools Panel</label>
|
||||||
|
</div>
|
||||||
|
</body>
|
||||||
|
</html>
|
53
AutoplayStopper/skin/popup.css
Normal file
53
AutoplayStopper/skin/popup.css
Normal file
|
@ -0,0 +1,53 @@
|
||||||
|
body {
|
||||||
|
margin: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.popupmenu {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
color: #303942;
|
||||||
|
background: #f9fafb;
|
||||||
|
-webkit-user-select: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.popupmenu .popupitem {
|
||||||
|
cursor: pointer;
|
||||||
|
display: block;
|
||||||
|
white-space: nowrap;
|
||||||
|
}
|
||||||
|
|
||||||
|
.popupmenu .popupitem:hover {
|
||||||
|
background: rgb(235, 235, 235); /*#565a63;
|
||||||
|
color: white;*/
|
||||||
|
}
|
||||||
|
|
||||||
|
.popupmenu .popupitem.separator {
|
||||||
|
background-color: rgba(0,0,0,0.15);
|
||||||
|
height: 1px;
|
||||||
|
pointer-events: none;
|
||||||
|
margin: 2px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.popupmenu .popupitem:not(.separator) {
|
||||||
|
padding: 7px 10px;
|
||||||
|
border-radius: 2px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.popupmenu .popupitem[checked] {
|
||||||
|
background: rgb(80, 135, 90); /*#54975f;*/
|
||||||
|
color: white;
|
||||||
|
}
|
||||||
|
|
||||||
|
.popupmenu .popupitem[checked="2"], .popupmenu .popupitem[checked="3"] {
|
||||||
|
background: rgb(135, 80, 90); /*#54975f;*/
|
||||||
|
color: white;
|
||||||
|
}
|
||||||
|
|
||||||
|
.popupmenu.restricted #allow_site, .popupmenu.restricted #allow_session, .popupmenu.restricted #disable_site {
|
||||||
|
opacity: 0.5;
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.popupmenu #allow_all[checked] {
|
||||||
|
background: darkgray;
|
||||||
|
}
|
24
AutoplayStopper/skin/popup.html
Normal file
24
AutoplayStopper/skin/popup.html
Normal file
|
@ -0,0 +1,24 @@
|
||||||
|
<!DOCTYPE html>
|
||||||
|
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8" />
|
||||||
|
<link href="popup.css" rel="stylesheet" />
|
||||||
|
<script src="/popup.js"></script>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<div id="popupmenu" class="popupmenu">
|
||||||
|
<section>
|
||||||
|
<div id="disable_site" class="popupitem" i18n-content="disableSite" >Disable flash for %S</div>
|
||||||
|
<div id="allow_site" class="popupitem" i18n-content="allowSite" >Allow autoplay for %S</div>
|
||||||
|
<div id="allow_session" class="popupitem" i18n-content="allowSession" >Allow session autoplay for %S</div>
|
||||||
|
</section>
|
||||||
|
<div class="popupitem separator"></div>
|
||||||
|
<section>
|
||||||
|
<!-- <popup-item id="script">Edit script...</popup-item> -->
|
||||||
|
<div id="allow_all" class="popupitem" i18n-content="allowAll" >Allow everywhere</div>
|
||||||
|
<div id="settings" class="popupitem" i18n-content="settings" >Settings</div>
|
||||||
|
</section>
|
||||||
|
</div>
|
||||||
|
</body>
|
||||||
|
</html>
|
103
AutoplayStopper/utils.js
Normal file
103
AutoplayStopper/utils.js
Normal file
|
@ -0,0 +1,103 @@
|
||||||
|
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
const i18n = new function(){
|
||||||
|
|
||||||
|
return {
|
||||||
|
process: function (doc) {
|
||||||
|
var elements = doc.querySelectorAll("[i18n-content]");
|
||||||
|
for (var i = 0; i < elements.length; ++i) {
|
||||||
|
var element = elements[i];
|
||||||
|
var message = chrome.i18n.getMessage(element.getAttribute("i18n-content"));
|
||||||
|
if (message) element.textContent = message;
|
||||||
|
else
|
||||||
|
console.warn("i18n: no message for " + element.getAttribute("i18n-content"));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
const Storage = function Storage(keys, areaname){
|
||||||
|
|
||||||
|
var listeners = [];
|
||||||
|
var data = {};
|
||||||
|
var ready = new Promise(function (resolve, reject) {
|
||||||
|
chrome.storage[areaname].get(keys, function(items){
|
||||||
|
Object.assign(data, items);
|
||||||
|
if (!chrome.runtime.lastError) resolve();
|
||||||
|
})
|
||||||
|
});
|
||||||
|
|
||||||
|
chrome.storage.onChanged.addListener(function(changes, area) {
|
||||||
|
if (area == areaname) {
|
||||||
|
var updated = [];
|
||||||
|
for (var key of Object.keys(changes))
|
||||||
|
if (keys.indexOf(key) != -1) { data[key] = changes[key].newValue; updated.push(key); };
|
||||||
|
fireChanged(updated);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return {
|
||||||
|
get ready() { return ready; },
|
||||||
|
get data() { return data; },
|
||||||
|
addChangeListener: function(listener) { listeners.push(listener); return listener; },
|
||||||
|
removeChangeListener: function(listener) { var idx = listeners.indexOf(listener); if (idx != -1) listeners.splice(idx); },
|
||||||
|
commit: function commit(keys, callback) {
|
||||||
|
var items = {};
|
||||||
|
for (var key of keys) items[key] = data[key];
|
||||||
|
chrome.storage[areaname].set(items, function(){
|
||||||
|
if (callback) callback(chrome.runtime.lastError);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
function fireChanged(updated) { for (var listener of listeners) listener(updated); };
|
||||||
|
};
|
||||||
|
|
||||||
|
const Permission = {
|
||||||
|
UNKNOWN_ACTION: 0,
|
||||||
|
ALLOW_ACTION: 1,
|
||||||
|
DENY_ACTION: 2,
|
||||||
|
PROMPT_ACTION: 3,
|
||||||
|
ACCESS_SESSION: 8
|
||||||
|
};
|
||||||
|
|
||||||
|
const Permissions = class Permissions {
|
||||||
|
|
||||||
|
constructor(data) {
|
||||||
|
|
||||||
|
function getData(type){ return data[Permissions.key(type)] || (data[Permissions.key(type)] = {}); };
|
||||||
|
function getDefault(type){ return data[Permissions.defaultKey(type)]; };
|
||||||
|
|
||||||
|
Object.assign(this, {
|
||||||
|
testPermission: function testPermission(type, uri) {
|
||||||
|
var origin, url = new window.URL(uri);
|
||||||
|
var hostTokens = url.host.split(".");
|
||||||
|
do {
|
||||||
|
var permission = getData(type)[origin = url.protocol + "//" + hostTokens.join(".")];
|
||||||
|
if (permission != undefined && permission != Permission.UNKNOWN_ACTION)
|
||||||
|
return Object.assign(Number(permission), {origin});
|
||||||
|
} while (hostTokens.shift());
|
||||||
|
return getDefault(type);
|
||||||
|
},
|
||||||
|
clear: function clear(type) { data[Permissions.key(type)] = {}; },
|
||||||
|
set: function set(type, url, perm) { getData(type)[new window.URL(url).origin] = perm; },
|
||||||
|
remove: function remove(type, url) { delete getData(type)[new window.URL(url).origin]; },
|
||||||
|
get: function get(type, url) { return getData(type)[new window.URL(url).origin]; },
|
||||||
|
setDefault: function setDefault(type, perm) { data[Permissions.defaultKey(type)] = perm; },
|
||||||
|
default: function _default(type) { return getDefault(type); },
|
||||||
|
entries: function entries(type) { return Object.entries(getData(type)); },
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
key(type) { return Permissions.key(type); };
|
||||||
|
defaultKey(type) { return Permissions.defaultKey(type); };
|
||||||
|
|
||||||
|
static key(type) { return `perms:${type}`; };
|
||||||
|
static defaultKey(type) { return `perms:${type}Default`; };
|
||||||
|
};
|
||||||
|
|
||||||
|
Object.assign(Permissions.prototype, Permission);
|
||||||
|
|
||||||
|
//</>
|
Loading…
Reference in New Issue
Block a user