mirror of
https://github.com/kkapsner/CanvasBlocker
synced 2025-01-20 10:28:31 +01:00
Added setting "ignoreFrequentColors"
This commit is contained in:
parent
9da251b14d
commit
1733be23f2
@ -184,6 +184,15 @@
|
|||||||
"description": ""
|
"description": ""
|
||||||
},
|
},
|
||||||
|
|
||||||
|
"ignoreFrequentColors_title": {
|
||||||
|
"message": "Ignoriere die häufigsten Farben",
|
||||||
|
"description": ""
|
||||||
|
},
|
||||||
|
"ignoreFrequentColors_description": {
|
||||||
|
"message": "Anzahl der Farben, die pro Canvas nicht vorgetäuscht werden sollen. Dies ist ein Parameter, der die Detektion des Addons erschweren soll.\nACHTUNG: Dies kann die Geschwindigkeit des Addons beeinträchtigen, da für jedes Bild die Farbstatistik berechnet werden muss. Außerdem kann es die Sicherheit des Addons verringern, wenn der Wert zu hoch gestellt wird. Deswegen wird stark empfohlen, diesen Wert nicht über 3 zu setzen.",
|
||||||
|
"description": ""
|
||||||
|
},
|
||||||
|
|
||||||
"disableNotifications": {
|
"disableNotifications": {
|
||||||
"message": "Benachrichtigungen deaktivieren",
|
"message": "Benachrichtigungen deaktivieren",
|
||||||
"description": ""
|
"description": ""
|
||||||
|
@ -184,6 +184,15 @@
|
|||||||
"description": ""
|
"description": ""
|
||||||
},
|
},
|
||||||
|
|
||||||
|
"ignoreFrequentColors_title": {
|
||||||
|
"message": "Ignore the most frequent colors",
|
||||||
|
"description": ""
|
||||||
|
},
|
||||||
|
"ignoreFrequentColors_description": {
|
||||||
|
"message": "Number of colors that should not be faked per canvas. This is a parameter to prevent detection.\nCAUTION: This can reduce the performance of the addon because the color statistic has to be calculated for every image. Additional this can lower the safety of the addon, therefore it is highly recommended not to set this value above 3.",
|
||||||
|
"description": ""
|
||||||
|
},
|
||||||
|
|
||||||
"disableNotifications": {
|
"disableNotifications": {
|
||||||
"message": "disable notifications",
|
"message": "disable notifications",
|
||||||
"description": ""
|
"description": ""
|
||||||
|
80
lib/colorStatistics.js
Normal file
80
lib/colorStatistics.js
Normal file
@ -0,0 +1,80 @@
|
|||||||
|
/* jslint moz: true */
|
||||||
|
/* 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";
|
||||||
|
|
||||||
|
|
||||||
|
var scope;
|
||||||
|
if ((typeof exports) !== "undefined"){
|
||||||
|
scope = exports;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
window.scope.colorStatistics = {};
|
||||||
|
scope = window.scope.colorStatistics;
|
||||||
|
}
|
||||||
|
|
||||||
|
class Statistic{
|
||||||
|
constructor(){
|
||||||
|
this.colors = Object.create(null);
|
||||||
|
this.numberOfColors = 0;
|
||||||
|
|
||||||
|
this.minBoundary = {count: Number.NEGATIVE_INFINITY};
|
||||||
|
this.maxBoundary = {count: Number.POSITIVE_INFINITY, previousColor: this.minBoundary};
|
||||||
|
this.minBoundary.nextColor = this.maxBoundary;
|
||||||
|
}
|
||||||
|
addColor(r, g, b, a){
|
||||||
|
var index = String.fromCharCode(r, g, b, a);
|
||||||
|
var color = this.colors[index];
|
||||||
|
if (!color){
|
||||||
|
color = {index, color: [r, g, b, a], count: 0, previousColor: this.minBoundary, nextColor: this.minBoundary.nextColor};
|
||||||
|
this.numberOfColors += 1;
|
||||||
|
this.minBoundary.nextColor = color;
|
||||||
|
color.nextColor.previousColor = color;
|
||||||
|
this.colors[index] = color;
|
||||||
|
}
|
||||||
|
color.count += 1;
|
||||||
|
if (color.count > color.nextColor.count){
|
||||||
|
// swap colors to remain in right order
|
||||||
|
// a -> b -> c -> d becomes a -> c -> b -> d
|
||||||
|
var a = color.previousColor;
|
||||||
|
var b = color;
|
||||||
|
var c = color.nextColor;
|
||||||
|
var d = color.nextColor.nextColor;
|
||||||
|
|
||||||
|
a.nextColor = c;
|
||||||
|
c.previousColor = a;
|
||||||
|
|
||||||
|
c.nextColor = b;
|
||||||
|
b.previousColor = c;
|
||||||
|
|
||||||
|
b.nextColor = d;
|
||||||
|
d.previousColor = b;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
getMaxColors(n){
|
||||||
|
var n = Math.min(n, this.numberOfColors);
|
||||||
|
var colors = Object.create(null);
|
||||||
|
var current = this.maxBoundary;
|
||||||
|
for (;n && current;n -= 1){
|
||||||
|
current = current.previousColor;
|
||||||
|
colors[current.index] = current;
|
||||||
|
}
|
||||||
|
return colors;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
scope.compute = function computeColorStatistics(rawData){
|
||||||
|
var statistic = new Statistic();
|
||||||
|
for (var i = 0, l = rawData.length; i < l; i += 4){
|
||||||
|
statistic.addColor(
|
||||||
|
rawData[i + 0],
|
||||||
|
rawData[i + 1],
|
||||||
|
rawData[i + 2],
|
||||||
|
rawData[i + 3]
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return statistic;
|
||||||
|
}
|
||||||
|
}());
|
@ -8,6 +8,7 @@ var settings = {
|
|||||||
minFakeSize: 1,
|
minFakeSize: 1,
|
||||||
maxFakeSize: 0,
|
maxFakeSize: 0,
|
||||||
rng: "constant",
|
rng: "constant",
|
||||||
|
ignoreFrequentColors: 0,
|
||||||
persistentRndStorage: "",
|
persistentRndStorage: "",
|
||||||
storePersistentRnd: false,
|
storePersistentRnd: false,
|
||||||
askOnlyOnce: true,
|
askOnlyOnce: true,
|
||||||
|
@ -14,6 +14,8 @@
|
|||||||
scope = window.scope.modifiedAPI;
|
scope = window.scope.modifiedAPI;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const colorStatistics = require("./colorStatistics");
|
||||||
|
|
||||||
// let Cu = require("chrome").Cu;
|
// let Cu = require("chrome").Cu;
|
||||||
|
|
||||||
var randomSupply = null;
|
var randomSupply = null;
|
||||||
@ -52,7 +54,7 @@
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
function getFakeCanvas(window, original){
|
function getFakeCanvas(window, original, prefs){
|
||||||
try {
|
try {
|
||||||
// original may not be a canvas -> we must not leak an error
|
// original may not be a canvas -> we must not leak an error
|
||||||
var context = getContext(window, original);
|
var context = getContext(window, original);
|
||||||
@ -61,6 +63,11 @@
|
|||||||
var l = desc.length;
|
var l = desc.length;
|
||||||
|
|
||||||
var ignoredColors = {};
|
var ignoredColors = {};
|
||||||
|
if (prefs("ignoreFrequentColors")){
|
||||||
|
var statistic = colorStatistics.compute(source);
|
||||||
|
ignoredColors = statistic.getMaxColors(prefs("ignoreFrequentColors"));
|
||||||
|
}
|
||||||
|
|
||||||
var rng = randomSupply.getPixelRng(l, window, ignoredColors);
|
var rng = randomSupply.getPixelRng(l, window, ignoredColors);
|
||||||
|
|
||||||
for (var i = 0; i < l; i += 4){
|
for (var i = 0; i < l; i += 4){
|
||||||
@ -175,7 +182,7 @@
|
|||||||
return function toDataURL(){
|
return function toDataURL(){
|
||||||
if (canvasSizeShouldBeFaked(this, prefs)){
|
if (canvasSizeShouldBeFaked(this, prefs)){
|
||||||
notify.call(this, "fakedReadout");
|
notify.call(this, "fakedReadout");
|
||||||
return original.apply(getFakeCanvas(window, this), window.Array.from(arguments));
|
return original.apply(getFakeCanvas(window, this, prefs), window.Array.from(arguments));
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
return original.apply(this, window.Array.from(arguments));
|
return original.apply(this, window.Array.from(arguments));
|
||||||
@ -201,7 +208,7 @@
|
|||||||
return function toBlob(callback){
|
return function toBlob(callback){
|
||||||
if (canvasSizeShouldBeFaked(this, prefs)){
|
if (canvasSizeShouldBeFaked(this, prefs)){
|
||||||
notify.call(this, "fakedReadout");
|
notify.call(this, "fakedReadout");
|
||||||
return original.apply(getFakeCanvas(window, this), window.Array.from(arguments));
|
return original.apply(getFakeCanvas(window, this, prefs), window.Array.from(arguments));
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
return original.apply(this, window.Array.from(arguments));
|
return original.apply(this, window.Array.from(arguments));
|
||||||
@ -228,7 +235,7 @@
|
|||||||
return function mozGetAsFile(callback){
|
return function mozGetAsFile(callback){
|
||||||
if (canvasSizeShouldBeFaked(this, prefs)){
|
if (canvasSizeShouldBeFaked(this, prefs)){
|
||||||
notify.call(this, "fakedReadout");
|
notify.call(this, "fakedReadout");
|
||||||
return original.apply(getFakeCanvas(window, this), window.Array.from(arguments));
|
return original.apply(getFakeCanvas(window, this, prefs), window.Array.from(arguments));
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
return original.apply(this, window.Array.from(arguments));
|
return original.apply(this, window.Array.from(arguments));
|
||||||
@ -257,11 +264,11 @@
|
|||||||
var fakeCanvas;
|
var fakeCanvas;
|
||||||
var context = this;
|
var context = this;
|
||||||
if (this && this.canvas) {
|
if (this && this.canvas) {
|
||||||
fakeCanvas = getFakeCanvas(window, this.canvas);
|
fakeCanvas = getFakeCanvas(window, this.canvas, prefs);
|
||||||
}
|
}
|
||||||
if (fakeCanvas && fakeCanvas !== this.canvas){
|
if (fakeCanvas && fakeCanvas !== this.canvas){
|
||||||
context = window.HTMLCanvasElement.prototype.getContext.call(
|
context = window.HTMLCanvasElement.prototype.getContext.call(
|
||||||
getFakeCanvas(window, this.canvas),
|
fakeCanvas,
|
||||||
"2d"
|
"2d"
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@ -349,7 +356,12 @@
|
|||||||
var xPixels = pixels;
|
var xPixels = pixels;
|
||||||
var ret = original.apply(this, window.Array.from(arguments));
|
var ret = original.apply(this, window.Array.from(arguments));
|
||||||
var l = xPixels.length;
|
var l = xPixels.length;
|
||||||
var rng = randomSupply.getPixelRng(l, window, {});
|
var ignoredColors = {};
|
||||||
|
if (prefs("ignoreFrequentColors")){
|
||||||
|
var statistic = colorStatistics.compute(pixels);
|
||||||
|
ignoredColors = statistic.getMaxColors(prefs("ignoreFrequentColors"));
|
||||||
|
}
|
||||||
|
var rng = randomSupply.getPixelRng(l, window, ignoredColors);
|
||||||
|
|
||||||
for (var i = 0; i < l; i += 4){
|
for (var i = 0; i < l; i += 4){
|
||||||
var [r, g, b, a] = rng(
|
var [r, g, b, a] = rng(
|
||||||
|
@ -25,6 +25,7 @@
|
|||||||
|
|
||||||
"lib/logging.js",
|
"lib/logging.js",
|
||||||
|
|
||||||
|
"lib/colorStatistics.js",
|
||||||
"lib/modifiedAPI.js",
|
"lib/modifiedAPI.js",
|
||||||
"lib/randomSupplies.js",
|
"lib/randomSupplies.js",
|
||||||
"lib/intercept.js",
|
"lib/intercept.js",
|
||||||
|
@ -130,6 +130,15 @@ document.body.appendChild(table);
|
|||||||
"rng": ["persistent"]
|
"rng": ["persistent"]
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"name": "ignoreFrequentColors",
|
||||||
|
"title": "Ignore most frequent colors",
|
||||||
|
"type": "integer",
|
||||||
|
"value": 0,
|
||||||
|
"displayDependencies": {
|
||||||
|
"blockMode": ["fakeReadout"]
|
||||||
|
}
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"name": "askOnlyOnce",
|
"name": "askOnlyOnce",
|
||||||
"title": "Ask only once",
|
"title": "Ask only once",
|
||||||
|
@ -6,6 +6,7 @@ Version 0.4.0:
|
|||||||
- switched to webExtension
|
- switched to webExtension
|
||||||
- notifications are now done via page action
|
- notifications are now done via page action
|
||||||
- minimal and maximal fake size are now respected in all fakeable functions
|
- minimal and maximal fake size are now respected in all fakeable functions
|
||||||
|
- fake readout now fakes one pixel at once and no longer one single channel
|
||||||
|
|
||||||
new features:
|
new features:
|
||||||
- information of all fake events in one tab are visible
|
- information of all fake events in one tab are visible
|
||||||
@ -13,6 +14,8 @@ Version 0.4.0:
|
|||||||
- new preferences:
|
- new preferences:
|
||||||
* minimal fake size
|
* minimal fake size
|
||||||
* setting to enable the inspection of the content of the faked canvas
|
* setting to enable the inspection of the content of the faked canvas
|
||||||
|
* new random number generator "constant"
|
||||||
|
* setting to not fake the most frequent colors in a canvas
|
||||||
|
|
||||||
fixes:
|
fixes:
|
||||||
- ask mode did not work for input types
|
- ask mode did not work for input types
|
||||||
|
Loading…
x
Reference in New Issue
Block a user