1
0
mirror of https://github.com/kkapsner/CanvasBlocker synced 2025-01-11 06:05:10 +01:00
CanvasBlocker/options/optionsGui.js

487 lines
14 KiB
JavaScript
Raw Normal View History

2017-11-07 00:36:44 +01:00
/* 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/. */
(function(){
"use strict";
2019-11-28 01:26:35 +01:00
let scope;
2017-11-07 00:36:44 +01:00
if ((typeof exports) !== "undefined"){
scope = exports;
}
else {
2019-03-12 22:24:23 +01:00
scope = require.register("./optionsGui", {});
2017-11-07 00:36:44 +01:00
}
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
function createDescription(setting){
2019-11-28 01:26:35 +01:00
const c = document.createElement("div");
2017-11-07 00:36:44 +01:00
c.className = "content";
2019-11-28 01:26:35 +01:00
const title = document.createElement("span");
2017-11-07 00:36:44 +01:00
title.className = "title";
2019-04-09 08:29:52 +02:00
title.textContent = extension.getTranslation(setting.name + "_title");
2017-11-07 00:36:44 +01:00
c.appendChild(title);
2019-11-28 01:26:35 +01:00
let descriptionText = extension.getTranslation(setting.name + "_description");
if (setting.urlSpecific){
2019-04-09 08:29:52 +02:00
const urlSpecificDescription = extension.getTranslation(setting.name + "_urlSpecific");
if (urlSpecificDescription){
descriptionText += (descriptionText? "\n\n": "") + urlSpecificDescription;
}
}
2017-11-11 23:34:59 +01:00
if (descriptionText){
2019-11-28 01:26:35 +01:00
const info = document.createElement("div");
2017-11-11 23:34:59 +01:00
info.className = "info";
c.appendChild(info);
2017-11-07 00:36:44 +01:00
2019-11-28 01:26:35 +01:00
const description = document.createElement("div");
2017-11-11 23:34:59 +01:00
description.className = "description";
description.textContent = descriptionText;
info.appendChild(description);
}
2017-11-07 00:36:44 +01:00
return c;
}
function createSelect(setting){
2019-11-28 01:26:35 +01:00
const select = document.createElement("select");
2017-11-07 00:36:44 +01:00
select.dataset.type = typeof setting.defaultValue;
setting.options.forEach(function(value){
2019-11-28 01:26:35 +01:00
const option = document.createElement("option");
2017-11-07 00:36:44 +01:00
if (typeof value === typeof setting.defaultValue){
option.value = value;
option.text = extension.getTranslation(setting.name + "_options." + value) || value;
2017-11-07 00:36:44 +01:00
if (setting.defaultValue === value){
option.selected = true;
option.selectedText = option.text;
option.notSelectedText = option.text + extension.getTranslation("labelForDefaultOption");
2017-11-07 00:36:44 +01:00
}
}
else {
option.disabled = true;
option.text = "\u2500".repeat(20);
}
select.appendChild(option);
});
select.update = function(){
Array.from(select.options).forEach(function(option){
if (option.notSelectedText){
option.text = option.notSelectedText;
}
});
const selectedOption = select.options[select.selectedIndex];
if (selectedOption.selectedText){
selectedOption.text = selectedOption.selectedText;
}
};
2017-11-07 00:36:44 +01:00
return select;
}
2019-11-28 01:26:35 +01:00
const inputTypes = {
2024-04-08 00:12:46 +02:00
all: {
updateCallback: function(input, value, defaultValue){
if (input.update){
input.update();
}
2024-04-08 00:12:46 +02:00
input.classList[value === defaultValue? "remove": "add"]("changed");
}
},
2017-11-07 00:36:44 +01:00
number: {
input: function(value){
const input = document.createElement("input");
2017-11-07 00:36:44 +01:00
input.type = "number";
input.value = value;
return input;
},
2024-04-08 00:12:46 +02:00
updateCallback: function(input, value, defaultValue){
input.value = value;
2024-04-08 00:12:46 +02:00
inputTypes.all.updateCallback(input, value, defaultValue);
2017-11-07 00:36:44 +01:00
return input.value;
},
getValue: function(input){
2017-11-07 00:36:44 +01:00
return parseFloat(input.value);
}
},
string: {
input: function(value, setting){
let input;
if (setting && setting.display && setting.display.multiline){
input = document.createElement("textarea");
input.rows = 1;
}
else {
input = document.createElement("input");
input.type = "text";
}
2017-11-07 00:36:44 +01:00
input.value = value;
return input;
},
2024-04-08 00:12:46 +02:00
updateCallback: function(input, value, defaultValue){
input.value = value;
2024-04-08 00:12:46 +02:00
inputTypes.all.updateCallback(input, value, defaultValue);
2017-11-07 00:36:44 +01:00
return input.value;
},
getValue: function(input){
2017-11-07 00:36:44 +01:00
return input.value;
}
},
boolean: {
input: function(value){
const input = document.createElement("input");
2017-11-07 00:36:44 +01:00
input.type = "checkbox";
input.checked = value;
input.style.display = "inline-block";
2017-11-07 00:36:44 +01:00
return input;
},
2024-04-08 00:12:46 +02:00
updateCallback: function(input, value, defaultValue){
input.checked = value;
2024-04-08 00:12:46 +02:00
inputTypes.all.updateCallback(input, value, defaultValue);
2017-11-07 00:36:44 +01:00
return input.checked;
},
getValue: function(input){
2017-11-07 00:36:44 +01:00
return input.checked;
}
},
object: false
2017-11-07 00:36:44 +01:00
};
2019-11-28 01:26:35 +01:00
function createKeyInput(setting, url){
const input = document.createElement("table");
let inSection = false;
setting.keys.forEach(function(key){
if (setting.display.displayedSection){
if (typeof key === "object"){
if (key.level === 1){
inSection = key.name === setting.display.displayedSection;
return;
}
}
if (!inSection){
return;
}
}
let row = document.createElement("tr");
if (typeof key === "object"){
let cell = document.createElement("td");
cell.colSpan = 2;
let h = document.createElement("h" + (2 + (key.level || 1)));
h.textContent = key.message? extension.getTranslation(key.message): key.name;
cell.appendChild(h);
row.appendChild(cell);
input.appendChild(row);
return;
}
let nameCell = document.createElement("td");
nameCell.textContent = setting.display.replaceKeyPattern?
key.replace(setting.display.replaceKeyPattern, ""):
key;
row.appendChild(nameCell);
let keyType = inputTypes[typeof setting.defaultKeyValue];
let keyInput = keyType.input(setting.defaultKeyValue);
let inputCell = document.createElement("td");
inputCell.appendChild(keyInput);
row.appendChild(inputCell);
setting.on(function(){
const container = setting.get(url);
keyType.updateCallback(
keyInput,
container && container.hasOwnProperty(key)?
container[key]:
setting.defaultKeyValue,
2024-04-08 00:12:46 +02:00
setting.defaultKeyValue
2019-11-28 01:26:35 +01:00
);
});
keyInput.addEventListener("change", function(){
const value = keyType.getValue(keyInput);
let container = setting.get(url);
if (!container){
container = setting.defaultValue;
}
container[key] = value;
if (setting.set(container, url)){
logging.message("changed setting", setting.name, "(", key, "):", value);
}
else {
container = setting.get(url);
keyType.updateCallback(
keyInput,
container && container.hasOwnProperty(key)?
container[key]:
setting.defaultKeyValue,
2024-04-08 00:12:46 +02:00
setting.defaultKeyValue
2019-11-28 01:26:35 +01:00
);
logging.message("setting", setting.name, "(", key, ") was not changed");
}
});
input.appendChild(row);
});
return input;
}
function getPopulateUrlTable(setting, type, body){
return function populateUrlTable({newValue}){
body.innerHTML = "";
newValue.forEach(function(entry){
let row = document.createElement("tr");
let urlCell = document.createElement("td");
urlCell.classList.add("url");
urlCell.addEventListener("click", function(){
const input = document.createElement("input");
input.classList.add("urlInput");
input.style.width = urlCell.clientWidth + "px";
input.style.height = urlCell.clientHeight + "px";
urlCell.innerHTML = "";
urlCell.appendChild(input);
input.title = extension.getTranslation("inputURL");
input.value = entry.url;
input.focus();
input.addEventListener("blur", function(){
const url = input.value.trim();
if (url){
entry.url = url;
setting.urlContainer.refresh();
}
urlCell.removeChild(input);
urlCell.textContent = entry.url;
});
input.addEventListener("click", function(event){
event.stopPropagation();
});
2019-11-28 01:26:35 +01:00
});
urlCell.textContent = entry.url;
row.appendChild(urlCell);
let input = createInput(setting, entry.url);
2024-04-08 00:12:46 +02:00
type.updateCallback(input, setting.get(entry.url), setting.defaultValue);
2019-11-28 01:26:35 +01:00
if (!entry.hasOwnProperty(setting.name)){
input.classList.add("notSpecifiedForUrl");
}
let inputCell = document.createElement("td");
inputCell.appendChild(input);
row.appendChild(inputCell);
let clearCell = document.createElement("td");
let clearButton = document.createElement("button");
clearButton.className = "reset";
clearButton.textContent = "\xD7";
clearButton.addEventListener("click", function(){
setting.reset(entry.url);
});
clearCell.appendChild(clearButton);
row.appendChild(clearCell);
body.appendChild(row);
});
};
}
function createUrlSpecificInput(setting, input, type){
const container = document.createElement("div");
container.className = "urlValues " + (setting.getExpand()? "expanded": "collapsed");
container.appendChild(input);
const collapser = document.createElement("button");
collapser.classList.add("collapser");
container.appendChild(collapser);
collapser.addEventListener("click", function(){
setting.setExpand(!setting.getExpand());
});
setting.onExpandChange(function(value){
container.classList[value? "remove": "add"]("collapsed");
container.classList[value? "add": "remove"]("expanded");
});
let urlTable = document.createElement("table");
let caption = document.createElement("caption");
caption.textContent = extension.getTranslation(setting.urlContainer.name + "_title");
urlTable.appendChild(caption);
let body = document.createElement("tbody");
urlTable.appendChild(body);
let foot = document.createElement("tfoot");
let footRow = document.createElement("tr");
let footCell = document.createElement("td");
footCell.colSpan = 3;
let newInput = document.createElement("input");
newInput.className = "inputURL";
newInput.title = extension.getTranslation("inputURL");
const addURLSetting = function(){
const url = newInput.value.trim();
if (url){
setting.set(setting.get(url), url);
newInput.value = "";
newInput.focus();
}
};
newInput.addEventListener("keypress", function(event){
if ([10, 13].indexOf(event.keyCode) !== -1){
addURLSetting();
}
});
footCell.appendChild(newInput);
let footPlus = document.createElement("button");
footPlus.classList.add("add");
footPlus.textContent = "+";
footPlus.addEventListener("click", addURLSetting);
footCell.appendChild(footPlus);
footRow.appendChild(footCell);
foot.appendChild(footRow);
urlTable.appendChild(foot);
container.appendChild(urlTable);
2017-11-07 00:36:44 +01:00
2019-11-28 01:26:35 +01:00
setting.urlContainer.on(getPopulateUrlTable(setting, type, body));
return container;
}
function createInput(setting, url = ""){
2019-11-28 01:26:35 +01:00
const type = inputTypes[typeof setting.defaultValue];
let input;
2017-11-07 00:36:44 +01:00
if (setting.options){
input = createSelect(setting);
}
else {
if (type){
input = type.input(setting.defaultValue, setting);
2017-11-07 00:36:44 +01:00
}
}
if (type){
2024-04-08 00:12:46 +02:00
setting.on(function(){
type.updateCallback(input, setting.get(url), setting.defaultValue);
}, url);
2017-11-07 00:36:44 +01:00
input.addEventListener("change", function(){
2019-11-28 01:26:35 +01:00
const value = type.getValue(input);
if (setting.set(value, url)){
2024-04-08 00:12:46 +02:00
type.updateCallback(input, value, setting.defaultValue);
logging.message("changed setting", setting.name, ":", value);
}
else {
2024-04-08 00:12:46 +02:00
type.updateCallback(input, setting.get(url), setting.defaultValue);
logging.message("setting", setting.name, "was not changed");
}
2017-11-07 00:36:44 +01:00
});
}
else if (setting.keys){
2019-11-28 01:26:35 +01:00
input = createKeyInput(setting, url);
}
if (setting.urlSpecific && url === ""){
2019-11-28 01:26:35 +01:00
return createUrlSpecificInput(setting, input, type);
}
return input || document.createElement("span");
2017-11-07 00:36:44 +01:00
}
function createButton(setting){
2019-11-28 01:26:35 +01:00
const button = document.createElement("button");
2019-04-09 08:29:52 +02:00
button.textContent = extension.getTranslation(setting.name + "_label");
2017-11-07 00:36:44 +01:00
button.addEventListener("click", setting.action);
return button;
}
function createInteraction(setting){
2019-11-28 01:26:35 +01:00
const c = document.createElement("div");
2017-11-07 00:36:44 +01:00
c.className = "content";
2019-11-28 01:26:35 +01:00
let interaction;
2017-11-07 00:36:44 +01:00
if (setting.action){
interaction = createButton(setting);
}
else if (setting.actions){
interaction = document.createElement("span");
setting.actions.forEach(function(action){
2019-11-28 01:26:35 +01:00
const button = createButton(action);
interaction.appendChild(button);
});
}
else if (setting.inputs){
interaction = document.createElement("span");
setting.inputs.forEach(function(inputSetting){
2019-11-28 01:26:35 +01:00
const input = createInput(inputSetting);
input.classList.add("multiple" + setting.inputs.length);
interaction.appendChild(input);
});
}
2017-11-07 00:36:44 +01:00
else {
interaction = createInput(setting);
}
interaction.classList.add("setting");
2017-11-07 00:36:44 +01:00
interaction.dataset.storageName = setting.name;
interaction.dataset.storageType = typeof setting.defaultValue;
c.appendChild(interaction);
return c;
}
2018-07-02 00:29:41 +02:00
function createHide(setting){
2019-11-28 01:26:35 +01:00
const label = document.createElement("label");
2018-07-02 00:29:41 +02:00
label.className = "content hideContent";
2019-04-09 08:29:52 +02:00
label.title = extension.getTranslation("hideSetting");
2019-11-28 01:26:35 +01:00
const input = document.createElement("input");
2018-07-02 00:29:41 +02:00
input.type = "checkbox";
input.className = "hide";
input.checked = setting.getHide();
input.addEventListener("change", function(){
setting.setHide(this.checked);
});
setting.onHideChange(function(value){
input.checked = value;
});
label.appendChild(input);
2019-11-28 01:26:35 +01:00
const display = document.createElement("span");
2018-07-02 00:29:41 +02:00
display.className = "display";
label.appendChild(display);
return label;
}
2017-11-07 00:36:44 +01:00
function createSettingRow(setting){
2019-11-28 01:26:35 +01:00
const tr = document.createElement("tr");
2017-11-07 00:36:44 +01:00
tr.className = "settingRow";
2018-07-02 00:29:41 +02:00
2019-11-28 01:26:35 +01:00
const hide = document.createElement("td");
2018-07-02 00:29:41 +02:00
hide.className = "hideColumn";
hide.appendChild(createHide(setting));
tr.appendChild(hide);
2017-11-07 00:36:44 +01:00
2019-11-28 01:26:35 +01:00
const left = document.createElement("td");
2017-11-07 00:36:44 +01:00
left.appendChild(createDescription(setting));
tr.appendChild(left);
2019-11-28 01:26:35 +01:00
const right = document.createElement("td");
2017-11-07 00:36:44 +01:00
right.appendChild(createInteraction(setting));
tr.appendChild(right);
return tr;
}
scope.createSettingRow = createSettingRow;
2018-07-02 00:29:41 +02:00
function createThead(displayHidden, restContent){
2018-07-02 00:29:41 +02:00
const tHead = document.createElement("thead");
const searchRow = document.createElement("tr");
2018-07-02 00:29:41 +02:00
const hideHeadCell = document.createElement("td");
hideHeadCell.className = "hideColumn";
searchRow.appendChild(hideHeadCell);
2018-07-02 00:29:41 +02:00
const restHeadCell = document.createElement("td");
restHeadCell.colSpan = 2;
if (restContent){
restHeadCell.appendChild(restContent);
}
searchRow.appendChild(restHeadCell);
tHead.appendChild(searchRow);
const displayHiddenRow = document.createElement("tr");
displayHiddenRow.className = "settingRow displayHiddenRow";
displayHiddenRow.appendChild(hideHeadCell.cloneNode());
const displayHiddenDescription = document.createElement("td");
displayHiddenDescription.appendChild(createDescription(displayHidden));
displayHiddenRow.appendChild(displayHiddenDescription);
2019-11-28 01:26:35 +01:00
const displayHiddenInteraction = document.createElement("td");
displayHiddenInteraction.appendChild(createInteraction(displayHidden));
displayHiddenRow.appendChild(displayHiddenInteraction);
tHead.appendChild(displayHiddenRow);
2018-07-02 00:29:41 +02:00
return tHead;
}
scope.createThead = createThead;
2017-11-07 00:36:44 +01:00
}());