CanvasBlocker/options/options.js

468 lines
14 KiB
JavaScript
Raw Normal View History

/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
2017-10-03 15:35:31 +02:00
(function(){
"use strict";
2019-04-09 08:29:52 +02:00
const extension = require("../lib/extension");
2019-04-08 00:02:29 +02:00
const logging = require("../lib/logging");
2017-11-07 00:36:44 +01:00
logging.setPrefix("options page");
const optionsGui = require("./optionsGui");
2019-04-08 00:02:29 +02:00
const settings = require("../lib/settings");
const settingsDisplay = require("../lib/settingsDisplay");
const search = require("../lib/search");
const settingStrings = require("../lib/settingStrings");
const searchParameters = new URLSearchParams(window.location.search);
2019-04-08 00:02:29 +02:00
const settingsMigration = require("../lib/settingsMigration");
require("./theme").init("options");
2017-11-07 00:36:44 +01:00
var callbacks = {
2019-02-27 23:49:00 +01:00
openNavigatorSettings: function(){
logging.verbose("open navigator settings");
window.open("navigator.html", "_blank");
},
2017-11-07 00:36:44 +01:00
showReleaseNotes: function(){
logging.verbose("open release notes");
window.open("../releaseNotes.txt", "_blank");
},
clearPersistentRnd: function(){
logging.message("clear persistent rnd storage");
logging.notice("empty storage");
settings.persistentRndStorage = "";
logging.notice("send message to main script");
extension.message.send({"canvasBlocker-clear-domain-rnd": true});
2017-11-14 01:07:27 +01:00
},
inspectSettings: function(){
logging.verbose("open settings inspection");
2017-11-14 01:07:27 +01:00
window.open("export.html", "_blank");
},
2018-10-09 08:14:50 +02:00
openSettingSanitation: function(){
logging.verbose("open settings sanitation");
window.open("sanitize.html", "_blank");
},
saveSettings: function(){
logging.verbose("save settings");
const data = {};
settings.forEach(function(def){
data[def.name] = def.get();
});
const blob = new Blob([JSON.stringify(data, null, "\t")], {type: "application/json"});
const link = document.createElement("a");
link.href = window.URL.createObjectURL(blob);
link.target = "_blank";
link.download = "CanvasBlocker-settings.json";
document.body.appendChild(link);
link.click();
document.body.removeChild(link);
},
2019-05-04 01:12:45 +02:00
inspectWhitelist: function(){
logging.verbose("open whitelist inspection");
window.open("whitelist.html", "_blank");
},
loadSettings: function(){
logging.verbose("load settings");
new Promise(function(resolve, reject){
const input = document.createElement("input");
input.type = "file";
input.addEventListener("change", function(){
if (this.files.length){
var file = this.files[0];
var reader = new FileReader();
reader.onload = function(result){
resolve(this.result);
};
reader.onerror = function(err){
reject(err);
};
reader.readAsText(this.files[0]);
}
});
input.click();
}).then(function(text){
return JSON.parse(text);
}).then(function(json){
while (settingsMigration.transitions.hasOwnProperty(json.storageVersion)){
let oldVersion = json.storageVersion;
json = settingsMigration.transitions[json.storageVersion](json);
if (oldVersion === json.storageVersion){
break;
}
}
const keys = Object.keys(json);
keys.forEach(function(key){
const setting = settings.getDefinition(key);
if (!settings){
throw new Error("Unknown setting " + key + ".");
}
if (!setting.fixed && setting.invalid(json[key])){
throw new Error("Invalid value " + json[key] + " for " + key + ".");
}
});
keys.forEach(function(key){
settings[key] = json[key];
});
}).catch(function(err){
alert(err);
});
},
resetSettings: function(){
2019-04-09 08:29:52 +02:00
if (window.confirm(extension.getTranslation("resetSettings_confirm"))){
browser.storage.local.clear();
}
2017-06-25 22:11:28 +02:00
}
2017-11-07 00:36:44 +01:00
};
2018-08-18 22:35:52 +02:00
new Promise(function(resolve){
const port = browser.runtime.connect();
port.onMessage.addListener(function(data){
if (data.hasOwnProperty("tabId")){
logging.notice("my tab id is", data.tabId);
port.disconnect();
resolve(data.tabId);
}
});
}).then(function(tabId){
return browser.tabs.get(tabId);
}).then(function(tab){
2019-04-09 08:29:52 +02:00
document.querySelector("head title").textContent = extension.getTranslation("options_title");
let head = document.createElement("header");
2018-08-18 22:35:52 +02:00
document.body.insertBefore(head, document.body.firstChild);
2018-08-18 22:35:52 +02:00
if (tab.url === window.location.href){
let heading = document.createElement("h1");
2019-04-09 08:29:52 +02:00
heading.textContent = extension.getTranslation("options");
2018-08-18 22:35:52 +02:00
head.appendChild(heading);
let introduction = document.createElement("div");
2019-04-09 08:29:52 +02:00
introduction.textContent = extension.getTranslation("optionsIntroduction");
2018-08-18 22:35:52 +02:00
head.appendChild(introduction);
if (searchParameters.has("notice")){
let noticeName = searchParameters.get("notice") + "Notice";
2019-04-09 08:29:52 +02:00
let notice = extension.getTranslation(noticeName);
2018-08-18 22:35:52 +02:00
if (notice){
let bookmarkingNotice = document.createElement("div");
bookmarkingNotice.className = noticeName + " bookmarkNotice";
bookmarkingNotice.textContent = notice;
const dontShowAgain = document.createElement("label");
dontShowAgain.className = "dontShowOptionsOnUpdate";
const dontShowAgainInput = document.createElement("input");
dontShowAgainInput.type = "checkbox";
settings.onloaded(function(){
dontShowAgainInput.checked = settings.dontShowOptionsOnUpdate;
});
dontShowAgainInput.addEventListener("change", function(){
settings.dontShowOptionsOnUpdate = this.checked;
});
dontShowAgain.appendChild(dontShowAgainInput);
dontShowAgain.appendChild(
document.createTextNode(
" " + extension.getTranslation("dontShowOptionsOnUpdate")
)
);
bookmarkingNotice.appendChild(dontShowAgain);
2018-08-18 22:35:52 +02:00
head.appendChild(bookmarkingNotice);
const newUrl = new URL(window.location.href);
newUrl.search = "";
window.history.pushState({}, "", newUrl.href);
2018-08-18 22:35:52 +02:00
}
}
2018-08-18 22:35:52 +02:00
document.body.classList.add("standalone");
}
else {
const linkDiv = document.createElement("div");
linkDiv.className = "optionsLink";
const link = document.createElement("a");
link.href = window.location.href;
link.target = "_blank";
2019-04-09 08:29:52 +02:00
link.textContent = extension.getTranslation("openInTab");
linkDiv.appendChild(link);
head.appendChild(linkDiv);
}
2018-08-18 22:35:52 +02:00
});
2017-11-07 00:36:44 +01:00
var table = document.createElement("table");
2017-11-11 23:34:59 +01:00
table.className = "settings " + (settings.displayDescriptions? "display": "hide") + "Descriptions";
settings.on("displayDescriptions", function(){
table.className = "settings " + (settings.displayDescriptions? "display": "hide") + "Descriptions";
});
2017-11-07 00:36:44 +01:00
document.body.appendChild(table);
2017-12-24 21:52:44 +01:00
2018-07-02 00:29:41 +02:00
const displayHidden = settings.getDefinition(settingsDisplay.displayHidden);
const searchInput = search.init();
table.appendChild(
optionsGui.createThead(
displayHidden,
searchInput
)
);
const searchOnly = searchParameters.has("searchOnly");
if (searchOnly){
document.body.classList.add("searching");
}
if (searchParameters.has("search")){
searchInput.value = searchParameters.get("search");
}
search.on(function({search, results, lastResults}){
lastResults.forEach(function(node){
node.classList.remove("found");
});
if (search || searchOnly){
document.body.classList.add("searching");
results.forEach(function(node){
node.classList.add("found");
});
}
else {
document.body.classList.remove("searching");
}
});
2018-07-02 00:29:41 +02:00
2017-12-24 21:52:44 +01:00
let lastSection = null;
let addSection = function addSection(name){
let body = document.createElement("tbody");
if (name){
let row = document.createElement("tr");
row.className = "section";
let cell = document.createElement("td");
2018-07-02 00:29:41 +02:00
cell.colSpan = 3;
2017-12-24 21:52:44 +01:00
row.appendChild(cell);
let heading = document.createElement("h2");
2019-04-09 08:29:52 +02:00
heading.textContent = extension.getTranslation("section_" + name);
2017-12-24 21:52:44 +01:00
cell.appendChild(heading);
body.appendChild(row);
}
table.appendChild(body);
let rows = [];
let section = {
addRow: function(row){
rows.push(row);
body.appendChild(row);
},
updateDisplay: function(){
const searchMode = document.body.classList.contains("searching");
2018-01-08 17:08:36 +01:00
var anyVisible = false;
rows.forEach(function(row){
var isHidden = row.classList.contains("hidden");
if (!isHidden){
if (searchMode){
if (!row.classList.contains("found")){
return;
}
}
2018-01-08 17:08:36 +01:00
if (anyVisible){
row.classList.remove("firstVisible");
}
else {
anyVisible = true;
row.classList.add("firstVisible");
}
}
});
body.classList[anyVisible? "remove": "add"]("hidden");
2017-10-03 15:35:31 +02:00
}
2017-12-24 21:52:44 +01:00
};
search.on(function(){section.updateDisplay();});
2017-12-24 21:52:44 +01:00
lastSection = section;
};
addSection();
const beforeChangeEventListeners = {};
const {hide: hideContainer, expand: expandContainer} = settings.getContainers();
2017-12-24 21:52:44 +01:00
settingsDisplay.forEach(function(display){
if (typeof display === "string"){
addSection(display);
2017-11-07 00:36:44 +01:00
}
2017-12-24 21:52:44 +01:00
else {
var setting = settings.getDefinition(display.name);
if (!setting){
if (display.inputs){
setting = {
name: display.name,
inputs: display.inputs.map(settings.getDefinition)
};
}
else if (display.actions){
setting = {
name: display.name,
actions: display.actions.map(function(action){
return {
name: action,
action: callbacks[action]
};
}).filter(function(action){
return action.action;
})
};
2017-12-24 21:52:44 +01:00
}
else if (callbacks[display.name]){
setting = {
name: display.name,
action: callbacks[display.name]
};
}
}
if (setting){
2018-09-12 09:21:52 +02:00
setting.display = display;
let originalSet = setting.set;
if (originalSet){
const eventListeners = [];
beforeChangeEventListeners[setting.name] = eventListeners;
setting.set = function(...args){
if (eventListeners.every(function(listener){
return listener.call(setting, ...args);
})){
return originalSet.apply(this, args);
}
else {
return false;
}
2019-01-24 15:43:20 +01:00
};
}
2018-07-02 00:29:41 +02:00
let hideChangeListeners = [];
setting.setHide = function setHide(value){
if (hideContainer){
hideContainer.setHideByName(display.name, value);
if (computeDependencies){
computeDependencies();
}
}
};
setting.onHideChange = function(listener){
hideChangeListeners.push(listener);
};
setting.getHide = function getHide(){
if (hideContainer){
return hideContainer.getHideByName(display.name);
}
else {
return false;
}
};
if (hideContainer){
hideContainer.onHideChange(display.name, function(value){
if (computeDependencies){
computeDependencies();
}
hideChangeListeners.forEach(function(listener){
listener(value);
});
});
}
let expandChangeListeners = [];
setting.setExpand = function setExpand(value){
if (expandContainer){
expandContainer.setExpandByName(display.name, value);
}
};
setting.onExpandChange = function(listener){
expandChangeListeners.push(listener);
};
setting.getExpand = function getExpand(){
if (expandContainer){
return expandContainer.getExpandByName(display.name);
}
else {
return false;
}
};
if (expandContainer){
expandContainer.onExpandChange(display.name, function(value){
expandChangeListeners.forEach(function(listener){
listener(value);
});
});
}
2017-12-24 21:52:44 +01:00
var row = optionsGui.createSettingRow(setting);
settingStrings.getStrings(setting).forEach(function(string){
search.register(string, row);
});
2017-12-24 21:52:44 +01:00
let section = lastSection;
section.addRow(row);
2018-07-02 00:29:41 +02:00
if (!display.displayDependencies){
display.displayDependencies = {};
2017-12-24 21:52:44 +01:00
}
2018-07-02 00:29:41 +02:00
var displayDependencies = display.displayDependencies;
displayDependencies = Array.isArray(displayDependencies)?
displayDependencies:
[displayDependencies];
var computeDependencies = function(){
logging.verbose("evaluate display dependencies for", setting);
row.classList[(
(
displayHidden.get() ||
!setting.getHide()
) &&
displayDependencies.some(function(displayDependency){
return Object.keys(displayDependency).every(function(key){
return displayDependency[key].indexOf(settings[key]) !== -1;
});
})
)? "remove": "add"]("hidden");
section.updateDisplay();
};
computeDependencies();
displayDependencies.forEach(function(displayDependency){
Object.keys(displayDependency).forEach(function(name){
settings.on(name, computeDependencies);
});
});
displayHidden.on(computeDependencies);
2017-10-03 15:35:31 +02:00
}
2017-11-07 00:36:44 +01:00
}
2017-06-25 22:11:28 +02:00
});
const version = document.createElement("div");
version.className = "version";
fetch(browser.extension.getURL("manifest.json")).then(function(response){
return response.json();
}).then(function(manifest){
version.textContent = "Version " + manifest.version;
});
document.body.appendChild(version);
settings.onloaded(function(){
const reCaptchaEntry = "^https://www\\.google\\.com/recaptcha/api2/(?:b?frame|anchor).*$";
const {url: urlContainer} = settings.getContainers();
settings.on("protectWindow", function({newValue}){
if (newValue){
const urlValue = urlContainer.get();
const matching = urlValue.filter(function(entry){
return entry.url === reCaptchaEntry;
});
if (
newValue &&
(
matching.length === 0 ||
matching[0].protectWindow
) &&
2019-04-09 08:29:52 +02:00
window.confirm(extension.getTranslation("protectWindow_askReCaptchaException"))
){
settings.set("protectWindow", false, reCaptchaEntry);
}
}
});
beforeChangeEventListeners.sharePersistentRndBetweenDomains.push(function(value){
if (value){
2019-04-09 08:29:52 +02:00
if (!confirm(extension.getTranslation("sharePersistentRndBetweenDomains_confirmMessage"))){
return false;
}
}
return true;
});
});
searchInput.search();
2017-10-03 19:51:03 +02:00
}());