CanvasBlocker/lib/modifiedDOMRectAPI.js

280 lines
7.6 KiB
JavaScript
Raw Normal View History

2018-09-04 23:29:58 +02: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";
let scope;
2018-09-04 23:29:58 +02:00
if ((typeof exports) !== "undefined"){
scope = exports;
}
else {
2019-03-12 22:24:23 +01:00
scope = require.register("./modifiedDOMRectAPI", {});
2018-09-04 23:29:58 +02:00
}
2019-12-12 00:09:53 +01:00
const extension = require("./extension");
const {checkerWrapper, setProperties, getStatusByFlag} = require("./modifiedAPIFunctions");
const {byteArrayToString: hash} = require("./hash");
2018-09-04 23:29:58 +02:00
let randomSupply = null;
2018-09-04 23:29:58 +02:00
scope.setRandomSupply = function(supply){
randomSupply = supply;
};
function getHash(domRect){
return hash(new Float64Array([domRect.x, domRect.y, domRect.width, domRect.height]));
}
function getValueHash(value){
return hash(new Float32Array([value]));
}
2018-09-04 23:29:58 +02:00
const registeredRects = new WeakMap();
function registerDOMRect(domRect, notify, window, prefs){
2019-12-12 00:09:53 +01:00
registeredRects.set(extension.getWrapped(domRect), {
2018-09-04 23:29:58 +02:00
notify: function(){
let done = false;
return function(message){
if (!done){
done = true;
notify(message);
}
};
}(),
window,
prefs
2018-09-04 23:29:58 +02:00
});
}
function getDOMRectRegistration(domRect){
2019-12-12 00:09:53 +01:00
return registeredRects.get(extension.getWrapped(domRect));
2018-09-04 23:29:58 +02:00
}
const cache = {};
const valueCache = [{}, {}, {}, {}, {}];
scope.cache = {
valueCache,
X: 0,
Y: 1,
WIDTH: 2,
HEIGHT: 3,
OTHER: 4
};
function getFakeDomRect(window, domRect, prefs, notify){
2018-09-04 23:29:58 +02:00
const hash = getHash(domRect);
let cached = cache[hash];
2018-09-04 23:29:58 +02:00
if (!cached){
notify("fakedDOMRectReadout");
const rng = randomSupply.getRng(4, window);
const getFakeValue = function getFakeValue(value, i){
const valueHash = getValueHash(value);
const cache = valueCache[i];
let cachedValue = cache[valueHash];
if (typeof cachedValue === "number"){
return cachedValue;
}
if ((value * prefs("domRectIntegerFactor", window.location)) % 1 === 0){
cache[valueHash] = value;
return value;
}
else {
const fakedValue = value + 0.01 * (rng(i) / 0xffffffff - 0.5);
const fakedHash = getValueHash(fakedValue);
cache[valueHash] = fakedValue;
cache[fakedHash] = fakedValue;
return fakedValue;
}
};
2018-09-04 23:29:58 +02:00
cached = new domRect.constructor(
getFakeValue(domRect.x, 0),
getFakeValue(domRect.y, 1),
getFakeValue(domRect.width, 2),
getFakeValue(domRect.height, 3)
);
cache[hash] = cached;
cache[getHash(cached)] = cached;
2018-09-04 23:29:58 +02:00
}
return cached;
}
2019-12-10 15:07:22 +01:00
function registerCallback(args, check){
const {prefs, notify, window, original} = check;
const originalValue = args.length?
original.call(this, ...args):
2019-12-10 15:07:22 +01:00
original.call(this);
registerDOMRect(originalValue, notify, window, prefs);
return originalValue;
}
2018-09-04 23:29:58 +02:00
scope.changedFunctions = {
getClientRects: {
object: ["Range", "Element"],
fakeGenerator: function(checker){
return function getClientRects(){
return checkerWrapper(checker, this, arguments, function(args, check){
const {prefs, notify, window, original} = check;
const ret = args.length? original.call(this, ...args): original.call(this);
2018-09-04 23:29:58 +02:00
for (let i = 0; i < ret.length; i += 1){
registerDOMRect(ret[i], notify, window, prefs);
2018-09-04 23:29:58 +02:00
}
return ret;
});
};
}
},
getBoundingClientRect: {
object: ["Range", "Element"],
fakeGenerator: function(checker){
return function getBoundingClientRect(){
2019-12-10 15:07:22 +01:00
return checkerWrapper(checker, this, arguments, registerCallback);
2018-09-04 23:29:58 +02:00
};
}
},
getBounds: {
object: ["DOMQuad"],
fakeGenerator: function(checker){
return function getBounds(){
2019-12-10 15:07:22 +01:00
return checkerWrapper(checker, this, arguments, registerCallback);
2018-09-04 23:29:58 +02:00
};
}
},
getBBox: {
object: ["SVGGraphicsElement"],
fakeGenerator: function(checker){
return function getBBox(){
2019-12-10 15:07:22 +01:00
return checkerWrapper(checker, this, arguments, registerCallback);
2018-09-04 23:29:58 +02:00
};
}
},
getExtentOfChar: {
object: ["SVGTextContentElement"],
fakeGenerator: function(checker){
2019-12-10 15:07:22 +01:00
return function getExtentOfChar(){
return checkerWrapper(checker, this, arguments, registerCallback);
2018-09-04 23:29:58 +02:00
};
}
},
};
function generateChangedDOMRectPropertyGetter(property, readonly = false){
const changedGetter = {
objectGetters: readonly?
[
function(window){return window.DOMRectReadOnly && window.DOMRectReadOnly.prototype;}
]:
[
function(window){return window.DOMRect && window.DOMRect.prototype;},
function(window){return window.DOMRectReadOnly && window.DOMRectReadOnly.prototype;}
],
name: property,
getterGenerator: function(){
2019-12-12 00:09:53 +01:00
const temp = {
get [property](){
const registration = getDOMRectRegistration(this);
if (registration){
return getFakeDomRect(
registration.window,
this,
registration.prefs,
registration.notify
2019-12-12 00:09:53 +01:00
)[property];
}
2019-12-12 00:09:53 +01:00
return this[property];
2018-09-04 23:29:58 +02:00
}
2019-12-12 00:09:53 +01:00
};
return Object.getOwnPropertyDescriptor(temp, property).get;
2018-09-04 23:29:58 +02:00
}
};
if (!readonly){
changedGetter.setterGenerator = function(window, original, prefs){
2019-12-12 00:09:53 +01:00
const temp = {
set [property](newValue){
const registration = getDOMRectRegistration(this);
if (registration){
const fakeDomRect = getFakeDomRect(window, this, prefs, registration.notify);
2019-12-12 00:09:53 +01:00
registeredRects.delete(extension.getWrapped(this));
2019-05-09 23:11:15 +02:00
["x", "y", "width", "height"].forEach((prop) => {
2019-12-12 00:09:53 +01:00
if (prop === property){
this[prop] = newValue;
}
else {
this[prop] = fakeDomRect[prop];
}
});
}
else {
original.call(this, ...arguments);
}
2018-09-04 23:29:58 +02:00
}
2019-12-12 00:09:53 +01:00
};
return Object.getOwnPropertyDescriptor(temp, property).set;
};
}
return changedGetter;
}
scope.changedGetters = [
generateChangedDOMRectPropertyGetter("x", false),
generateChangedDOMRectPropertyGetter("y", false),
generateChangedDOMRectPropertyGetter("width", false),
generateChangedDOMRectPropertyGetter("height", false),
generateChangedDOMRectPropertyGetter("left", true),
generateChangedDOMRectPropertyGetter("right", true),
generateChangedDOMRectPropertyGetter("top", true),
generateChangedDOMRectPropertyGetter("bottom", true),
2018-09-04 23:29:58 +02:00
{
objectGetters: [
function(window){
return window.IntersectionObserverEntry && window.IntersectionObserverEntry.prototype;
}
2018-09-04 23:29:58 +02:00
],
name: "intersectionRect",
getterGenerator: function(checker){
const temp = {
get intersectionRect(){
2019-12-10 15:07:22 +01:00
return checkerWrapper(checker, this, arguments, registerCallback);
2018-09-04 23:29:58 +02:00
}
};
return Object.getOwnPropertyDescriptor(temp, "intersectionRect").get;
}
},
{
objectGetters: [
function(window){
return window.IntersectionObserverEntry && window.IntersectionObserverEntry.prototype;
}
2018-09-04 23:29:58 +02:00
],
name: "boundingClientRect",
getterGenerator: function(checker){
const temp = {
get boundingClientRect(){
2019-12-10 15:07:22 +01:00
return checkerWrapper(checker, this, arguments, registerCallback);
2018-09-04 23:29:58 +02:00
}
};
return Object.getOwnPropertyDescriptor(temp, "boundingClientRect").get;
}
},
{
objectGetters: [
function(window){
return window.IntersectionObserverEntry && window.IntersectionObserverEntry.prototype;
}
2018-09-04 23:29:58 +02:00
],
name: "rootBounds",
getterGenerator: function(checker){
const temp = {
get rootBounds(){
2019-12-10 15:07:22 +01:00
return checkerWrapper(checker, this, arguments, registerCallback);
2018-09-04 23:29:58 +02:00
}
};
return Object.getOwnPropertyDescriptor(temp, "rootBounds").get;
}
}
];
2019-01-24 15:43:20 +01:00
setProperties(scope.changedFunctions, scope.changedGetters, {
type: "readout",
2019-12-10 15:07:22 +01:00
getStatus: getStatusByFlag("protectDOMRect"),
2019-01-24 15:43:20 +01:00
api: "domRect"
2018-09-04 23:29:58 +02:00
});
}());