mirror of
https://github.com/kkapsner/CanvasBlocker
synced 2025-01-11 14:14:46 +01:00
640bd36b86
Fixes #589 and fixes #590
215 lines
5.9 KiB
JavaScript
215 lines
5.9 KiB
JavaScript
/* 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";
|
|
let scope;
|
|
if ((typeof exports) !== "undefined"){
|
|
scope = exports;
|
|
}
|
|
else {
|
|
scope = require.register("./askForPermission", {});
|
|
}
|
|
|
|
const {parseErrorStack} = require("./callingStack");
|
|
|
|
// Check canvas appearance
|
|
function canvasAppearance(window, api, context){
|
|
let oldBorder = false;
|
|
let canvas = false;
|
|
let inDOM = null;
|
|
if (api === "canvas" && context){
|
|
let nodeName;
|
|
try {
|
|
nodeName = context.nodeName;
|
|
}
|
|
catch (error){
|
|
nodeName = "";
|
|
}
|
|
if (nodeName === "CANVAS"){
|
|
canvas = context;
|
|
}
|
|
else if (
|
|
context instanceof window.CanvasRenderingContext2D ||
|
|
context instanceof window.WebGLRenderingContext
|
|
){
|
|
canvas = context.canvas;
|
|
}
|
|
}
|
|
if (canvas){
|
|
oldBorder = canvas.style.border;
|
|
canvas.style.border = "2px solid red";
|
|
inDOM = canvas.ownerDocument.contains(canvas);
|
|
}
|
|
return {
|
|
canvas: canvas,
|
|
askCategory: canvas? (inDOM? "visible": "invisible"): (api === "canvas"? "nocanvas": api),
|
|
get text(){
|
|
const text = canvas? (this.visible? "visible": "invisible"): (api === "canvas"? "nocanvas": api);
|
|
Object.defineProperty(this, "text", {value: text});
|
|
return text;
|
|
},
|
|
inDom: inDOM,
|
|
get visible(){
|
|
let visible = inDOM;
|
|
if (inDOM){
|
|
canvas.scrollIntoView();
|
|
const rect = canvas.getBoundingClientRect();
|
|
const foundEl = window.document.elementFromPoint(
|
|
rect.left + rect.width / 2,
|
|
rect.top + rect.height / 2
|
|
);
|
|
visible = (foundEl === canvas);
|
|
}
|
|
Object.defineProperty(this, "visible", {value: visible});
|
|
return visible;
|
|
},
|
|
reset: function(){
|
|
if (canvas){
|
|
canvas.style.border = oldBorder;
|
|
}
|
|
}
|
|
};
|
|
}
|
|
|
|
const modes = new WeakMap();
|
|
function getAskMode(window, type, _){
|
|
let mode = modes.get(window);
|
|
if (mode){
|
|
return mode[type];
|
|
}
|
|
else {
|
|
mode = {
|
|
context: {
|
|
askText: {
|
|
visible: _("askForVisiblePermission"),
|
|
invisible: _("askForInvisiblePermission"),
|
|
nocanvas: _("askForPermission"),
|
|
audio: _("askForAudioPermission"),
|
|
history: _("askForHistoryPermission"),
|
|
window: _("askForWindowPermission"),
|
|
domRect: _("askForDOMRectPermission"),
|
|
svg: _("askForSVGPermission"),
|
|
},
|
|
askStatus: {
|
|
alreadyAsked: {},
|
|
answer: {}
|
|
}
|
|
},
|
|
input: {
|
|
askText: {
|
|
visible: _("askForVisibleInputPermission"),
|
|
invisible: _("askForInvisibleInputPermission"),
|
|
nocanvas: _("askForInputPermission"),
|
|
audio: _("askForAudioInputPermission"),
|
|
history: _("askForHistoryInputPermission"),
|
|
window: _("askForWindowInputPermission"),
|
|
domRect: _("askForDOMRectInputPermission"),
|
|
svg: _("askForSVGInputPermission"),
|
|
},
|
|
askStatus: {
|
|
alreadyAsked: {},
|
|
answer: {}
|
|
}
|
|
},
|
|
readout: {
|
|
askText: {
|
|
visible: _("askForVisibleReadoutPermission"),
|
|
invisible: _("askForInvisibleReadoutPermission"),
|
|
nocanvas: _("askForReadoutPermission"),
|
|
audio: _("askForAudioReadoutPermission"),
|
|
history: _("askForHistoryReadoutPermission"),
|
|
window: _("askForWindowReadoutPermission"),
|
|
domRect: _("askForDOMRectReadoutPermission"),
|
|
svg: _("askForSVGReadoutPermission"),
|
|
},
|
|
askStatus: {
|
|
alreadyAsked: {},
|
|
answer: {}
|
|
}
|
|
}
|
|
};
|
|
modes.set(window, mode);
|
|
return mode[type];
|
|
}
|
|
}
|
|
|
|
scope.ask = function({window, type, api, canvas, errorStack}, {_, prefs}){
|
|
let answer;
|
|
const askMode = getAskMode(window, type, _);
|
|
const askStatus = askMode.askStatus;
|
|
const appearance = canvasAppearance(window, api, canvas);
|
|
let category = appearance.askCategory;
|
|
if (prefs("askOnlyOnce") !== "no" && askStatus.alreadyAsked[category]){
|
|
// already asked
|
|
appearance.reset();
|
|
return askStatus.answer[category];
|
|
}
|
|
else {
|
|
let imgContainer = null;
|
|
if (type === "readout" && prefs("showCanvasWhileAsking") && canvas){
|
|
try {
|
|
let document = window.top.document;
|
|
imgContainer = document.createElement("div");
|
|
imgContainer.style.cssText = `
|
|
position: fixed;
|
|
top: 0;
|
|
left: 0;
|
|
right: 0;
|
|
bottom: 0;
|
|
background-color: rgba(0, 0, 0, 0.7);
|
|
color: white;
|
|
text-align: center;
|
|
z-index: 100000000000000;
|
|
padding: 1em;`;
|
|
|
|
let heading = document.createElement("h1");
|
|
heading.textContent = "CanvasBlocker";
|
|
imgContainer.appendChild(heading);
|
|
|
|
let text = document.createElement("div");
|
|
text.style.margin = "0.5em auto";
|
|
text.textContent = _("showCanvasWhileAsking_message");
|
|
imgContainer.appendChild(text);
|
|
|
|
let img = document.createElement("img");
|
|
img.style.backgroundColor = "white";
|
|
img.style.border = "2px solid lightgray";
|
|
img.src = HTMLCanvasElement.prototype.toDataURL.call(canvas);
|
|
imgContainer.appendChild(img);
|
|
document.body.appendChild(imgContainer);
|
|
}
|
|
catch (error){
|
|
// unable to read the canvas
|
|
}
|
|
}
|
|
// asking
|
|
let msg = askMode.askText[appearance.text];
|
|
|
|
// visible vs invisible is only calculated here correctly
|
|
category = appearance.text;
|
|
if (prefs("showCallingFile")){
|
|
msg += parseErrorStack(errorStack).toString(_);
|
|
}
|
|
answer = window.top.confirm(msg)? "allow": prefs("askDenyMode");
|
|
if (imgContainer && imgContainer.parentNode){
|
|
imgContainer.parentNode.removeChild(imgContainer);
|
|
}
|
|
|
|
if (prefs("askOnlyOnce") === "combined"){
|
|
["context", "readout", "input"].forEach(function(type){
|
|
const askMode = getAskMode(window, type, _);
|
|
const askStatus = askMode.askStatus;
|
|
askStatus.alreadyAsked[category] = true;
|
|
askStatus.answer[category] = answer;
|
|
});
|
|
}
|
|
else {
|
|
askStatus.alreadyAsked[category] = true;
|
|
askStatus.answer[category] = answer;
|
|
}
|
|
appearance.reset();
|
|
return answer;
|
|
}
|
|
};
|
|
}()); |