-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathBackground tab setTimeout muffler.user.js
150 lines (128 loc) · 4.91 KB
/
Background tab setTimeout muffler.user.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
// ==UserScript==
// @name Background tab setTimeout muffler
// @namespace BSP
// @description Defers setTimeout calls if no user input has been received for 15 seconds. Useful for stopping background tabs from continuing to use CPU cycles.
// @include http://*
// @include https://*
// @exclude http://*.google.tld/*
// @exclude https://*.google.tld/*
// @exclude http://irc.lc/*
// @exclude https://irc.lc/*
// @exclude http://www.newsblur.com/*
// @exclude https://www.newsblur.com/*
// @exclude http://steamcommunity.com/id*
// @exclude https://www.duolingo.com/*
// @version 1
// @run-at document-start
// ==/UserScript==
var setInterval_old = unsafeWindow.setInterval;
var setTimeout_old = unsafeWindow.setTimeout;
var clearInterval_old = unsafeWindow.clearInterval;
var clearTimeout_old = unsafeWindow.clearTimeout;
var startDate = new Date;
//QueuedTimer format: {ID: [startTime, repeat, func, delay, args]}
var queuedTimers = null;
//ActiveTimer format: {ID: [repeat, func, delay, args, browserTimerID]}
var activeTimers = {};
var nextID = 1000000;
var lastUserInteraction = Date.now();
function debugLog() {
false && console && console.log && console.log(Array.slice(arguments, 0));
}
function muffleTimers() {
return (Date.now() - lastUserInteraction) > 15000;
}
function addTimer(ID, repeat, func, delay, args, isNew) {
if(typeof func != "function" && typeof func != "string") {
debugLog("addTimer", "invalid func", arguments);
throw Error("setInterval/setTimeout: invalid func");
}
delay = +(delay || 0); //coerce to number
if(isNaN(delay)) { //check for NaN (results from non-number strings/objects, etc.)
debugLog("addTimer", "invalid delay", arguments);
throw Error("setInterval/setTimeout: invalid delay");
}
//if(Object.keys(queuedTimers || {}).length + Object.keys(activeTimers).length > 500 && new Date - startDate > 15000) {
// debugLog("addTimer", "More than 100 timers!? Fuck that!", arguments);
// throw Error("STFU PLZ");
//}
if(muffleTimers()) {
if(!queuedTimers) queuedTimers = {};
isNew && debugLog("addTimer", "Deferred", [ID, repeat, delay, func]);
queuedTimers[ID] = [Date.now(), repeat, func, delay, args];
} else {
var browserTimerID = setTimeout_old(execTimer, delay, ID, repeat, func, delay, args);
isNew && debugLog("addTimer", "Scheduled", [ID, repeat, delay, func]);
activeTimers[ID] = [repeat, func, delay, args, browserTimerID];
}
return ID;
}
function execTimer(ID, repeat, func, delay, args) {
delete activeTimers[ID];
debugLog("execTimer", "Executing", ID, repeat);
//Re-add to the list so it can be cleared while in progress
if(repeat) {
addTimer(ID, repeat, func, delay, args, false);
}
try {
if(typeof func == "string") {
(function() { unsafeWindow.eval(func); }).apply(unsafeWindow);
} else {
func.apply(unsafeWindow, args);
}
} catch(ex) { /* Silence it like JS normally does */ debugLog("execTimer", "Error", arguments, ex); }
}
function onUserInteraction(event) {
lastUserInteraction = Date.now();
//If any timers are queued, restart them
if(queuedTimers) {
debugLog("onUserInteraction", "Processing deferred timers", arguments, queuedTimers);
try {
for(var ID in queuedTimers) {
//QueuedTimer format: {ID: [startTime, repeat, func, delay, args]}
var timer = queuedTimers[ID];
var delay = Math.max(0, timer[0] + timer[3] - Date.now());
var browserTimerID = setTimeout_old(execTimer, delay, ID, timer[1], timer[2], timer[3], timer[4]);
//ActiveTimer format: {ID: [repeat, func, delay, args, browserTimerID]}
activeTimers[ID] = [timer[1], timer[2], timer[3], timer[4], browserTimerID];
delete queuedTimers[ID];
}
} catch(ex) {
debugLog("onUserInteraction", "Error", ex);
}
queuedTimers = null;
}
removeEventListener("mousemove", onUserInteraction);
removeEventListener("keydown", onUserInteraction);
setTimeout_old(function() {
addEventListener("mousemove", onUserInteraction);
addEventListener("keydown", onUserInteraction);
}, 5000);
}
addEventListener("mousemove", onUserInteraction);
addEventListener("keydown", onUserInteraction);
unsafeWindow.setInterval = function setInterval() {
var func = arguments[0];
var delay = arguments[1];
var args = Array.slice(arguments, 2);
return addTimer(nextID++, true, func, delay, args, true);
};
unsafeWindow.setTimeout = function setTimeout() {
var func = arguments[0];
var delay = arguments[1];
var args = Array.slice(arguments, 2);
return addTimer(nextID++, false, func, delay, args, true);
};
function clearTimer(ID) {
if(queuedTimers && typeof(queuedTimers[ID]) != "undefined") {
debugLog("clearTimer", "Clearing deferred timer", arguments);
delete queuedTimers[ID];
}
if(typeof(activeTimers[ID]) != "undefined") {
debugLog("clearTimer", "Clearing active timer", arguments);
clearTimeout(activeTimers[ID][4]);
delete activeTimers[ID];
}
};
unsafeWindow.clearTimeout = clearTimer;
unsafeWindow.clearInterval = clearTimer;