mirror of
https://github.com/kkapsner/CanvasBlocker
synced 2025-07-04 04:26:35 +02:00
Added URL specific settings
For blockMode and showNotifications. Fixes #148.
This commit is contained in:
parent
78a0ccc243
commit
01780da9f5
10 changed files with 618 additions and 78 deletions
|
@ -19,7 +19,8 @@
|
|||
const logging = require("./logging");
|
||||
|
||||
scope.check = function check({url, errorStack}){
|
||||
var match = checkBoth(errorStack, url, settings.blockMode).match(
|
||||
url = new URL(url || "about:blank");
|
||||
var match = checkBoth(errorStack, url, settings.get("blockMode", url)).match(
|
||||
/^(block|allow|fake|ask)(|Readout|Everything|Context|Input|Internal)$/
|
||||
);
|
||||
if (match){
|
||||
|
@ -50,7 +51,6 @@
|
|||
|
||||
function checkURL(url, blockMode){
|
||||
logging.message("check url %s for block mode %s", url, blockMode);
|
||||
url = new URL(url || "about:blank");
|
||||
switch (url.protocol){
|
||||
case "about:":
|
||||
if (url.href === "about:blank"){
|
||||
|
|
|
@ -78,8 +78,8 @@
|
|||
port.postMessage({"canvasBlocker-notify": data});
|
||||
}
|
||||
|
||||
function prefs(name){
|
||||
return settings[name];
|
||||
function prefs(...args){
|
||||
return settings.get(...args);
|
||||
}
|
||||
|
||||
|
||||
|
|
|
@ -58,7 +58,7 @@
|
|||
port.onMessage.addListener(function(data){
|
||||
if (data.hasOwnProperty("canvasBlocker-notify")){
|
||||
if (
|
||||
settings.showNotifications &&
|
||||
settings.get("showNotifications", url) &&
|
||||
!lists.get("ignore").match(url)
|
||||
){
|
||||
browser.pageAction.show(port.sender.tab.id);
|
||||
|
|
|
@ -10,6 +10,25 @@
|
|||
defaultValue: 1,
|
||||
options: [0, 1, 25, 50, 75, 100]
|
||||
},
|
||||
{
|
||||
name: "urlSettings",
|
||||
defaultValue: [],
|
||||
urlContainer: true,
|
||||
entries: [
|
||||
{name: "url", defaultValue: ""}
|
||||
]
|
||||
},
|
||||
{
|
||||
name: "urls",
|
||||
defaultValue: [],
|
||||
dynamic: true,
|
||||
dependencies: ["urlSettings"],
|
||||
getter: function(settings){
|
||||
return settings.urlSettings.map(function(urlSetting){
|
||||
return urlSetting.url;
|
||||
});
|
||||
}
|
||||
},
|
||||
{
|
||||
name: "whiteList",
|
||||
defaultValue: ""
|
||||
|
@ -21,6 +40,7 @@
|
|||
{
|
||||
name: "blockMode",
|
||||
defaultValue: "fakeReadout",
|
||||
urlSpecific: true,
|
||||
options: [
|
||||
"blockReadout", "fakeReadout", "fakeInput", "askReadout", null,
|
||||
"blockEverything", "block", "ask", "allow", "allowEverything"
|
||||
|
@ -84,7 +104,8 @@
|
|||
},
|
||||
{
|
||||
name: "showNotifications",
|
||||
defaultValue: true
|
||||
defaultValue: true,
|
||||
urlSpecific: true
|
||||
},
|
||||
{
|
||||
name: "storeImageForInspection",
|
||||
|
|
468
lib/settings.js
468
lib/settings.js
|
@ -31,64 +31,240 @@
|
|||
});
|
||||
}());
|
||||
const settingDefinitions = require("./settingDefinitions.js");
|
||||
const definitionsByName = {};
|
||||
const defaultSymbol = "";
|
||||
|
||||
const eventHandler = {any: []};
|
||||
const eventHandler = {any: {}};
|
||||
eventHandler.any[defaultSymbol] = [];
|
||||
eventHandler.all = eventHandler.any;
|
||||
const settings = {};
|
||||
let urlContainer;
|
||||
|
||||
settingDefinitions.forEach(function(settingDefinition){
|
||||
var name = settingDefinition.name;
|
||||
settings[name] = settingDefinition.defaultValue;
|
||||
eventHandler[name] = [];
|
||||
|
||||
settingDefinition.on = function on(callback){
|
||||
scope.on(name, callback);
|
||||
};
|
||||
settingDefinition.invalid = function invalid(newValue){
|
||||
if (settingDefinition.fixed){
|
||||
return "fixed";
|
||||
}
|
||||
else if ((typeof newValue) !== (typeof settingDefinition.defaultValue)){
|
||||
function isDefinitionInvalid(settingDefinition, newValue){
|
||||
if (newValue === undefined && settingDefinition.optional){
|
||||
return false;
|
||||
}
|
||||
else if (settingDefinition.fixed){
|
||||
return "fixed";
|
||||
}
|
||||
else if ((typeof newValue) !== (typeof settingDefinition.defaultValue)){
|
||||
return "wrongType";
|
||||
}
|
||||
else if (Array.isArray(settingDefinition.defaultValue)){
|
||||
if (!Array.isArray(newValue)){
|
||||
return "wrongType";
|
||||
}
|
||||
else if (
|
||||
settingDefinition.options &&
|
||||
!settingDefinition.options.includes(newValue)
|
||||
){
|
||||
return "noOption";
|
||||
var entriesInvalid = newValue.reduce(function(v, entry){
|
||||
v = v || settingDefinition.entries.reduce(function(v, entryDefinition){
|
||||
return v || isDefinitionInvalid(entryDefinition, entry[entryDefinition.name]);
|
||||
}, false);
|
||||
if (!v){
|
||||
if (Object.keys(entry).some(function(key){
|
||||
return !settingDefinition.entries.some(function(entryDefinition){
|
||||
return key === entryDefinition.name;
|
||||
});
|
||||
})){
|
||||
return "noOption";
|
||||
}
|
||||
}
|
||||
return v;
|
||||
}, false);
|
||||
if (entriesInvalid){
|
||||
return entriesInvalid;
|
||||
}
|
||||
return false;
|
||||
};
|
||||
settingDefinition.get = function getValue(){
|
||||
return settings[name];
|
||||
};
|
||||
settingDefinition.set = function setValue(newValue){
|
||||
logging.verbose("New value for %s:", name, newValue);
|
||||
var invalid = settingDefinition.invalid(newValue);
|
||||
if (invalid){
|
||||
if (invalid === "fixed"){
|
||||
logging.warning("Trying to set the fixed setting", name, ":", newValue);
|
||||
}
|
||||
else if (
|
||||
settingDefinition.options &&
|
||||
!settingDefinition.options.includes(newValue)
|
||||
){
|
||||
return "noOption";
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
function createGetter(settingDefinition){
|
||||
if (settingDefinition.dynamic){
|
||||
return function getValue(){
|
||||
return settingDefinition.getter(scope);
|
||||
};
|
||||
}
|
||||
else if (settingDefinition.urlSpecific){
|
||||
return function getValue(url){
|
||||
if (url){
|
||||
var matching = urlContainer.get().filter(function(urlSetting){
|
||||
return urlSetting.hasOwnProperty(settingDefinition.name);
|
||||
}).filter(function(urlSetting){
|
||||
return urlSetting.match(url);
|
||||
});
|
||||
if (matching.length){
|
||||
return matching[0][settingDefinition.name];
|
||||
}
|
||||
}
|
||||
else if (invalid === "wrongType"){
|
||||
logging.warning("Wrong type provided for setting", name, ":", newValue);
|
||||
return settings[settingDefinition.name];
|
||||
};
|
||||
}
|
||||
else {
|
||||
return function getValue(){
|
||||
return settings[settingDefinition.name];
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
function createSetter(settingDefinition){
|
||||
if (settingDefinition.dynamic){
|
||||
return function setValue(newValue){
|
||||
settingDefinition.setter(scope);
|
||||
};
|
||||
}
|
||||
else {
|
||||
const name = settingDefinition.name;
|
||||
const isValid = function isValid(newValue){
|
||||
var invalid = settingDefinition.invalid(newValue);
|
||||
if (invalid){
|
||||
if (invalid === "fixed"){
|
||||
logging.warning("Trying to set the fixed setting", name, ":", newValue);
|
||||
}
|
||||
else if (invalid === "wrongType"){
|
||||
logging.warning("Wrong type provided for setting", name, ":", newValue);
|
||||
}
|
||||
else if (invalid === "noOption"){
|
||||
logging.warning("Provided value outside specified options for ", name, ":", newValue);
|
||||
}
|
||||
else {
|
||||
logging.warning("Unknown invalid state:", invalid);
|
||||
}
|
||||
return false;
|
||||
}
|
||||
else if (invalid === "noOption"){
|
||||
logging.warning("Provided value outside specified options for ", name, ":", newValue);
|
||||
}
|
||||
else {
|
||||
logging.warning("Unknown invalid state:", invalid);
|
||||
}
|
||||
}
|
||||
else {
|
||||
return true;
|
||||
};
|
||||
const storeValue = function storeValue(newValue){
|
||||
settings[name] = newValue;
|
||||
if (!settingDefinition.transient){
|
||||
var storeObject = {};
|
||||
storeObject[name] = newValue;
|
||||
browser.storage.local.set(storeObject);
|
||||
}
|
||||
|
||||
};
|
||||
|
||||
if (settingDefinition.urlSpecific){
|
||||
return function setValue(newValue, url){
|
||||
logging.verbose("New value for %s:", name, newValue);
|
||||
if (isValid(newValue)){
|
||||
if (url){
|
||||
var urlContainerValue = urlContainer.get();
|
||||
var matching = urlContainerValue.filter(function(urlSetting){
|
||||
return urlSetting.match(url);
|
||||
});
|
||||
if (!matching.length){
|
||||
let newEntry = {url};
|
||||
newEntry[settingDefinition.name] = newValue;
|
||||
urlContainerValue.push(newEntry);
|
||||
matching = [newEntry];
|
||||
}
|
||||
matching[0][settingDefinition.name] = newValue;
|
||||
urlContainer.set(urlContainerValue);
|
||||
}
|
||||
else {
|
||||
storeValue(newValue);
|
||||
}
|
||||
}
|
||||
};
|
||||
}
|
||||
else {
|
||||
return function setValue(newValue){
|
||||
logging.verbose("New value for %s:", name, newValue);
|
||||
if (isValid(newValue)){
|
||||
storeValue(newValue);
|
||||
}
|
||||
};
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
function createResetter(settingDefinition){
|
||||
if (settingDefinition.dynamic){
|
||||
return function(){};
|
||||
}
|
||||
else {
|
||||
const name = settingDefinition.name;
|
||||
let reset = function(){
|
||||
settings[name] = settingDefinition.defaultValue;
|
||||
browser.storage.local.remove(name);
|
||||
};
|
||||
if (settingDefinition.urlSpecific){
|
||||
return function(url){
|
||||
if (url){
|
||||
var urlContainerValue = urlContainer.get();
|
||||
var matching = urlContainerValue.filter(function(urlSetting){
|
||||
return urlSetting.match(url);
|
||||
});
|
||||
if (matching.length){
|
||||
delete matching[0][name];
|
||||
if (Object.keys(matching[0]).every(function(key){return key === "url";})){
|
||||
urlContainerValue = urlContainerValue.filter(function(urlSetting){
|
||||
return urlSetting !== matching[0];
|
||||
});
|
||||
}
|
||||
urlContainer.set(urlContainerValue);
|
||||
}
|
||||
}
|
||||
else {
|
||||
reset();
|
||||
}
|
||||
};
|
||||
}
|
||||
else {
|
||||
return reset;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
settingDefinitions.forEach(function(settingDefinition){
|
||||
if (settingDefinition.urlContainer){
|
||||
urlContainer = settingDefinition;
|
||||
settingDefinition.refresh = function(){
|
||||
settingDefinition.set(settingDefinition.get());
|
||||
};
|
||||
}
|
||||
|
||||
var name = settingDefinition.name;
|
||||
definitionsByName[name] = settingDefinition;
|
||||
settings[name] = settingDefinition.defaultValue;
|
||||
eventHandler[name] = {};
|
||||
|
||||
settingDefinition.on = function on(callback, url){
|
||||
if (!settingDefinition.dynamic){
|
||||
scope.on(name, callback, url);
|
||||
}
|
||||
if (settingDefinition.dependencies){
|
||||
settingDefinition.dependencies.forEach(function(dependency){
|
||||
scope.on(dependency, function(){
|
||||
callback({name, newValue: settingDefinition.get()});
|
||||
}, url);
|
||||
});
|
||||
}
|
||||
};
|
||||
settingDefinition.invalid = function invalid(newValue){
|
||||
return isDefinitionInvalid(settingDefinition, newValue);
|
||||
};
|
||||
settingDefinition.get = createGetter(settingDefinition);
|
||||
|
||||
settingDefinition.set = createSetter(settingDefinition);
|
||||
|
||||
settingDefinition.reset = createResetter(settingDefinition);
|
||||
|
||||
if (settingDefinition.urlSpecific){
|
||||
if (!urlContainer){
|
||||
logging.error("Unable to use url specific settings without url-container");
|
||||
}
|
||||
else {
|
||||
settingDefinition.urlContainer = urlContainer;
|
||||
let entry = Object.create(settingDefinition);
|
||||
entry.optional = true;
|
||||
urlContainer.entries.push(entry);
|
||||
}
|
||||
}
|
||||
|
||||
Object.defineProperty(
|
||||
scope,
|
||||
name,
|
||||
|
@ -101,31 +277,57 @@
|
|||
});
|
||||
|
||||
scope.getDefinition = function(name){
|
||||
var foundDefinitions = settingDefinitions.filter(function(settingDefinition){
|
||||
return name === settingDefinition.name;
|
||||
});
|
||||
if (foundDefinitions.length){
|
||||
return Object.create(foundDefinitions[0]);
|
||||
var foundDefinition = definitionsByName[name];
|
||||
if (foundDefinition){
|
||||
return Object.create(foundDefinition);
|
||||
}
|
||||
else {
|
||||
return undefined;
|
||||
}
|
||||
};
|
||||
|
||||
scope.set = function(name, ...args){
|
||||
var foundDefinition = definitionsByName[name];
|
||||
if (foundDefinition){
|
||||
return foundDefinition.set(...args);
|
||||
}
|
||||
else {
|
||||
logging.error("Try to set unkown setting:", name);
|
||||
}
|
||||
};
|
||||
scope.get = function(name, ...args){
|
||||
var foundDefinition = definitionsByName[name];
|
||||
if (foundDefinition){
|
||||
return foundDefinition.get(...args);
|
||||
}
|
||||
else {
|
||||
logging.error("Try to get unkown setting:", name);
|
||||
}
|
||||
};
|
||||
|
||||
scope.forEach = function forEachSetting(...args){
|
||||
settingDefinitions.map(function(settingDefinition){
|
||||
settingDefinitions.filter(function(settingDefinition){
|
||||
return !settingDefinition.dynamic;
|
||||
}).map(function(settingDefinition){
|
||||
return Object.create(settingDefinition);
|
||||
}).forEach(...args);
|
||||
};
|
||||
scope.on = function onSettingsChange(name, callback){
|
||||
|
||||
scope.on = function onSettingsChange(name, callback, url){
|
||||
if (Array.isArray(name)){
|
||||
name.forEach(function(name){
|
||||
onSettingsChange(name, callback);
|
||||
onSettingsChange(name, callback, url);
|
||||
});
|
||||
}
|
||||
else {
|
||||
if (eventHandler.hasOwnProperty(name)){
|
||||
eventHandler[name].push(callback);
|
||||
if (!url){
|
||||
url = defaultSymbol;
|
||||
}
|
||||
if (!eventHandler[name].hasOwnProperty(url)){
|
||||
eventHandler[name][url] = [];
|
||||
}
|
||||
eventHandler[name][url].push(callback);
|
||||
}
|
||||
else {
|
||||
logging.warning("Unable to register event handler for unknown setting", name);
|
||||
|
@ -133,28 +335,136 @@
|
|||
}
|
||||
};
|
||||
|
||||
const resetSymbol = Symbol("reset");
|
||||
function changeValue(name, newValue){
|
||||
var settingDefinition = scope.getDefinition(name);
|
||||
var oldValue = settings[name];
|
||||
if (newValue === resetSymbol){
|
||||
newValue = settingDefinition.defaultValue;
|
||||
}
|
||||
settings[name] = newValue;
|
||||
(eventHandler[name] || []).forEach(function(callback){
|
||||
((eventHandler[name] || {})[defaultSymbol] || []).forEach(function(callback){
|
||||
callback({name, newValue, oldValue});
|
||||
});
|
||||
|
||||
if (settingDefinition.urlSpecific){
|
||||
urlContainer.get().forEach(function(entry){
|
||||
if (!entry.hasOwnProperty(name)){
|
||||
((eventHandler[name] || {})[entry.url] || []).forEach(function(callback){
|
||||
callback({name, newValue, oldValue, url: entry.url});
|
||||
});
|
||||
}
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
logging.verbose("registering storage onchange listener");
|
||||
browser.storage.onChanged.addListener(function(changes, area){
|
||||
if (area === "local"){
|
||||
logging.notice("settings changed", changes);
|
||||
var delayedChange = [];
|
||||
Object.entries(changes).forEach(function(entry){
|
||||
const [name, change] = entry;
|
||||
changeValue(name, change.newValue);
|
||||
if (urlContainer && name === urlContainer.name){
|
||||
// changes in the url container have to trigger after the other changes
|
||||
delayedChange.push(entry);
|
||||
}
|
||||
else {
|
||||
if (change.hasOwnProperty("newValue")){
|
||||
changeValue(name, change.newValue);
|
||||
}
|
||||
else {
|
||||
changeValue(name, resetSymbol);
|
||||
}
|
||||
}
|
||||
});
|
||||
eventHandler.any.forEach(function(callback){
|
||||
delayedChange.forEach(function(entry){
|
||||
const [name, change] = entry;
|
||||
if (change.hasOwnProperty("newValue")){
|
||||
changeValue(name, change.newValue);
|
||||
}
|
||||
else {
|
||||
changeValue(name, resetSymbol);
|
||||
}
|
||||
});
|
||||
eventHandler.any[""].forEach(function(callback){
|
||||
callback();
|
||||
});
|
||||
}
|
||||
});
|
||||
|
||||
if (urlContainer){
|
||||
urlContainer.on(function({newValue, oldValue}){
|
||||
newValue.forEach(function(urlSetting){
|
||||
var regExp;
|
||||
var domain = !!urlSetting.url.match(/^[\w.]+$/);
|
||||
if (domain){
|
||||
regExp = new RegExp(
|
||||
"(?:^|\\.)" + urlSetting.url.replace(/([\\+*?[^\]$(){}=!|.])/g, "\\$1") + "\\.?$",
|
||||
"i"
|
||||
);
|
||||
}
|
||||
else {
|
||||
regExp = new RegExp(urlSetting.url, "i");
|
||||
}
|
||||
const match = function(url){
|
||||
if (!url){
|
||||
return false;
|
||||
}
|
||||
else if (
|
||||
url instanceof String ||
|
||||
(typeof url) === "string"
|
||||
){
|
||||
return url === urlSetting.url;
|
||||
}
|
||||
else if (domain){
|
||||
return (url.hostname || "").match(regExp);
|
||||
}
|
||||
else {
|
||||
return url.href.match(regExp);
|
||||
}
|
||||
};
|
||||
Object.defineProperty(
|
||||
urlSetting,
|
||||
"match",
|
||||
{
|
||||
enumerable: false,
|
||||
writable: true,
|
||||
configurable: true,
|
||||
value: match
|
||||
}
|
||||
);
|
||||
});
|
||||
|
||||
var newUrls = newValue.map(function(entry){return entry.url;});
|
||||
var oldUrls = oldValue.map(function(entry){return entry.url;});
|
||||
var matching = {};
|
||||
newUrls.forEach(function(url, i){
|
||||
matching[url] = {new: i, old: oldUrls.indexOf(url)};
|
||||
});
|
||||
oldUrls.forEach(function(url, i){
|
||||
if (!matching[url]){
|
||||
matching[url] = {new: -1, old: i};
|
||||
}
|
||||
});
|
||||
Object.keys(matching).forEach(function(url){
|
||||
var oldEntry = oldValue[matching[url].old] || {};
|
||||
var newEntry = newValue[matching[url].new] || {};
|
||||
urlContainer.entries.forEach(function(settingDefinition){
|
||||
var name = settingDefinition.name;
|
||||
var oldValue = oldEntry[name];
|
||||
var newValue = newEntry[name];
|
||||
|
||||
if (oldValue !== newValue){
|
||||
((eventHandler[name] || {})[url] || []).forEach(function(callback){
|
||||
callback({name, newValue, oldValue, url});
|
||||
});
|
||||
}
|
||||
});
|
||||
});
|
||||
});
|
||||
}
|
||||
|
||||
const settingsMigration = {
|
||||
validVersions: [undefined, 0.1, 0.2],
|
||||
transitions: {
|
||||
|
@ -170,6 +480,47 @@
|
|||
if (oldStorage.hasOwnProperty("askOnlyOnce")){
|
||||
newStorage.askOnlyOnce = oldStorage.askOnlyOnce? "individual": "no";
|
||||
}
|
||||
return newStorage;
|
||||
},
|
||||
0.2: function(oldStorage){
|
||||
var newStorage = {
|
||||
storageVersion: 0.3,
|
||||
urlSettings: (
|
||||
oldStorage.urlSettings &&
|
||||
Array.isArray(oldStorage.urlSettings)
|
||||
)? oldStorage.urlSettings: []
|
||||
};
|
||||
|
||||
var urlSettings = {};
|
||||
|
||||
(oldStorage.blackList || "").split(",").forEach(function(url){
|
||||
var entry = urlSettings[url];
|
||||
if (!entry){
|
||||
entry = {url, blockMode: "block"};
|
||||
urlSettings[url] = entry;
|
||||
newStorage.urlSettings.push(entry);
|
||||
}
|
||||
});
|
||||
(oldStorage.whiteList || "").split(",").forEach(function(url){
|
||||
var entry = urlSettings[url];
|
||||
if (!entry){
|
||||
entry = {url, blockMode: "allow"};
|
||||
urlSettings[url] = entry;
|
||||
newStorage.urlSettings.push(entry);
|
||||
}
|
||||
});
|
||||
(oldStorage.ignoreList || "").split(",").forEach(function(url){
|
||||
var entry = urlSettings[url];
|
||||
if (!entry){
|
||||
entry = {url, showNotifications: false};
|
||||
urlSettings[url] = entry;
|
||||
newStorage.urlSettings.push(entry);
|
||||
}
|
||||
else {
|
||||
entry.showNotifications = false;
|
||||
}
|
||||
});
|
||||
|
||||
return newStorage;
|
||||
}
|
||||
}
|
||||
|
@ -210,7 +561,18 @@
|
|||
logging.notice("Changed settings:", toChange);
|
||||
browser.storage.local.set(toChange);
|
||||
}
|
||||
var delayedChange = [];
|
||||
Object.entries(storage).forEach(function(entry){
|
||||
const [name, value] = entry;
|
||||
if (urlContainer && name === urlContainer.name){
|
||||
// changes in the url container have to trigger after the other changes
|
||||
delayedChange.push(entry);
|
||||
}
|
||||
else {
|
||||
changeValue(name, value);
|
||||
}
|
||||
});
|
||||
delayedChange.forEach(function(entry){
|
||||
const [name, value] = entry;
|
||||
changeValue(name, value);
|
||||
});
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue