// ==UserScript==
// @name Chess.com Favicon Alerts
// @description Add number of games waiting to favicon
// @version 0.8
// @author Jim Farrand
// @author Peter Wooley (Original GMail Favicon script)
// @license MIT
// @namespace http://xyxyx.org/
// @include https://www.chess.com/*
// @include http://www.chess.com/*
// @grant GM_getValue
// @grant GM_setValue
// @grant GM_registerMenuCommand
// ==/UserScript==
if(typeof GM_getValue === "undefined") {
function GM_getValue(name, fallback) {
return fallback;
}
}
var titleNotificationConfigKey = 'titleNotificationEnabled';
var debuggingConfigKey = 'debuggingEnabled';
var autoReloadConfigKey = 'autoReloadEnabled';
var lastCountKey = 'lastCount';
var lastCountUpdateTimeKey = 'lastCountUpdateTime';
var lastCountChangeTimeKey = 'lastCountChangeTime';
var flashIconForNewKey = 'flashIconEnabled';
// Register GM Commands and Methods
if(typeof GM_registerMenuCommand !== "undefined") {
var setTitleNotification = function(state) {
console.log("Setting title notifications: " + state);
GM_setValue(titleNotificationConfigKey, state);
};
var setDebugging = function(state) {
console.log("Setting debugging: " + state);
GM_setValue(debuggingConfigKey, state);
};
var setAutoReload = function(state) {
console.log("Setting auto-reload: " + state);
GM_setValue(autoReloadConfigKey, state);
}
var setFlashIcon = function(state) {
console.log("Setting flash icon: " + state);
GM_setValue(flashIconForNewKey, state);
}
GM_registerMenuCommand( "Chess.com Favicon Alerts > Set Title Notifications On",
function() { setTitleNotification(true) });
GM_registerMenuCommand( "Chess.com Favicon Alerts > Set Title Notifications Off",
function() { setTitleNotification(false) });
GM_registerMenuCommand( "Chess.com Favicon Alerts > Set Debugging On",
function() { setDebugging(true) });
GM_registerMenuCommand( "Chess.com Favicon Alerts > Set Debugging Off",
function() { setDebugging(false) });
GM_registerMenuCommand( "Chess.com Favicon Alerts > Set Auto Reload On",
function() { setAutoReload(true) });
GM_registerMenuCommand( "Chess.com Favicon Alerts > Set Auto Reload Off",
function() { setAutoReload(false) });
GM_registerMenuCommand( "Chess.com Favicon Alerts > Set Flash Icon After Change On",
function() { setFlashIcon(true) });
GM_registerMenuCommand( "Chess.com Favicon Alerts > Set Flash Icon After Change Off",
function() { setFlashIcon(false) });
}
if(!window.frameElement) {
new ChessDotComFavIconAlerts();
}
function ChessDotComFavIconAlerts() {
var self = this;
// PRIVATE VARIABLES AND METHODS
// The URL attached to the little hand icon, with the link containing the number of games
var gotoReadyGameURL = "http://www.chess.com/echess/goto_ready_game";
// Min time to wait after suspected suspend before refreshing
var reloadRandomizationMin = 20 * 1000;
// Random time to wait after suspected suspend before refreshing, in addition to reloadAfterSuspendMinimum
var reloadRandomizationMax = 40 * 1000;
var searchElement;
var iconCanvas;
var isDebugging = function() {
return false || GM_getValue(debuggingConfigKey, false);
}
var getLastCount = function() {
return GM_getValue(lastCountKey);
}
var getLastCountUpdateTime = function() {
return GM_getValue(lastCountUpdateTimeKey);
}
var getLastCountChangeTime = function() {
return GM_getValue(lastCountChangeTimeKey);
}
var setLastCount = function(value) {
GM_setValue(lastCountKey, value);
}
var setLastCountUpdateTime = function(value) {
GM_setValue(lastCountUpdateTimeKey, value);
}
var setLastCountChangeTime = function(value) {
GM_setValue(lastCountChangeTimeKey, value);
}
var isAutoReloadEnabled = function() {
return false || GM_getValue(autoReloadConfigKey, false);
}
var isFlashIconEnabled = function() {
return false || GM_getValue(flashIconForNewKey, false);
}
var isTitleUpdatedEnabled = function() {
return false || GM_getValue(titleNotificationConfigKey, false);
}
var head = window.document.getElementsByTagName('head')[0];
// Element that contains the count
var findSearchElement = function() {
var searchElement = document.getElementById("topright");;
if (isDebugging()) { console.log("findSearchElement: " + searchElement); }
return searchElement;
}
var setIcon = function(icon) {
var links = head.getElementsByTagName("link");
for (var i = 0; i < links.length; i++) {
if ((links[i].rel == "shortcut icon" || links[i].rel=="icon") && links[i].href != icon) {
head.removeChild(links[i]);
} else if(links[i].href == icon) {
return;
}
}
var newIcon = document.createElement("link");
newIcon.type = "image/png";
newIcon.rel = "shortcut icon";
newIcon.href = icon;
head.appendChild(newIcon);
setTimeout(function() {
if (isDebugging()) { console.log("Timeout function"); }
var shim = document.createElement('iframe');
shim.width = shim.height = 0;
document.body.appendChild(shim);
shim.src = "icon";
document.body.removeChild(shim);
if (isDebugging()) { console.log("Timeout function done"); }
}, 1000);
}
var getIconCanvas = function() {
if(!iconCanvas) {
iconCanvas = document.createElement('canvas');
iconCanvas.height = iconCanvas.width = 16;
var ctx = iconCanvas.getContext('2d');
for (var y = 0; y < iconCanvas.width; y++) {
for (var x = 0; x < iconCanvas.height; x++) {
if (self.pixelMaps.icons.unread[y][x]) {
ctx.fillStyle = self.pixelMaps.icons.unread[y][x];
ctx.fillRect(x, y, 1, 1);
}
}
}
}
return iconCanvas;
}
var showCount = function() {
// We could decide here whether to show the count or the other icon
return true;
}
// TODO: This could be made abstract so that that this class can be more easily reused
var getCount = function() {
// Return the number of things
if(searchElement) {
var center;
var topRightChildren = searchElement.childNodes;
for (var i = 0; i < topRightChildren.length; i++) {
var topRightChild = topRightChildren.item(i);
if (topRightChild.tagName == "LI" && topRightChild.hasAttribute("class") && topRightChild.getAttribute("class") == "center") {
var centerChildren = topRightChild.childNodes;
for (var i = 0; i < centerChildren.length; i++) {
var centerChild = centerChildren.item(i);
if (centerChild.tagName == "A" && centerChild.hasAttribute("href") && centerChild.getAttribute("href") == gotoReadyGameURL) {
var result = centerChild.textContent;
if (isDebugging()) { console.log("getCount: " + result); }
return result;
}
}
}
}
if (isDebugging()) { console.log("getCount: 0"); }
return 0;
}
}
this.construct = function() {
if (isDebugging()) { console.log("Creating ChessDotComFavIconAlerts"); }
// PUBLIC VARIABLES AND METHODS
this.backgroundFillColour = "#ff0000";
this.backgroundBorderColour = "#990000";
this.digitColour = "#ffffff";
// How long must have passed without user input in this window before we reload the page?
this.inactivityTimeout = 15 * 60 * 1000; // 15 minutes
// How old must the data be before we reload the page?
this.dataTimeout = 3 * 60 * 1000; // 3 minutes
// Note that we might have received data from another tab/window, which is why there are seperate data/inactivity timeouts
// How long to flash the icon for after it changes
this.flashPeriod = 15 * 1000;
// TODO: More things could be private
this.icons = {
// TODO: These are the same, and incorrectly named.
read: '',
unread: '',
};
this.pixelMaps = {
icons: {
// TODO: Transparency
'unread':
[["#ffffff","#ffffff","#ffffff","#ffffff","#ffffff","#ffffff","#afc59b","#aac193","#6d9645","#c6d4bc","#ffffff","#ffffff","#ffffff","#ffffff","#ffffff","#ffffff"],["#ffffff","#ffffff","#ffffff","#ffffff","#ffffff","#e1e8dc","#7ea159","#eef3e9","#67923a","#407119","#f5f8f7","#ffffff","#ffffff","#ffffff","#ffffff","#ffffff"],["#ffffff","#ffffff","#ffffff","#ffffff","#ffffff","#b4c8a4","#59882a","#5c8a2e","#69933d","#507d29","#c7d4c1","#ffffff","#ffffff","#ffffff","#ffffff","#ffffff"],["#ffffff","#ffffff","#ffffff","#ffffff","#ffffff","#f1f4f0","#598729","#69933e","#6c963e","#406f23","#ffffff","#ffffff","#ffffff","#ffffff","#ffffff","#ffffff"],["#ffffff","#ffffff","#ffffff","#ffffff","#ffffff","#bcceac","#6a9342","#68933c","#608b39","#5b8149","#c1cfb9","#ffffff","#ffffff","#ffffff","#ffffff","#ffffff"],["#ffffff","#ffffff","#ffffff","#ffffff","#ffffff","#7b9f5b","#5f8b35","#68933c","#638e39","#5b8247","#84a176","#ffffff","#ffffff","#ffffff","#ffffff","#ffffff"],["#ffffff","#ffffff","#ffffff","#ffffff","#ffffff","#ffffff","#dce5d7","#5f8d2e","#3c6a23","#f6f8f5","#ffffff","#ffffff","#ffffff","#ffffff","#ffffff","#ffffff"],["#ffffff","#ffffff","#ffffff","#ffffff","#ffffff","#ffffff","#c2d2b5","#618e30","#3b6924","#d8e1d3","#ffffff","#ffffff","#ffffff","#ffffff","#ffffff","#ffffff"],["#ffffff","#ffffff","#ffffff","#ffffff","#ffffff","#ffffff","#b8cba7","#608d31","#3b6826","#c2d1bb","#ffffff","#ffffff","#ffffff","#ffffff","#ffffff","#ffffff"],["#ffffff","#ffffff","#ffffff","#ffffff","#ffffff","#ffffff","#a9c192","#5d8932","#3f6c2a","#a2b897","#ffffff","#ffffff","#ffffff","#ffffff","#ffffff","#ffffff"],["#ffffff","#ffffff","#ffffff","#ffffff","#ffffff","#fefeff","#608c35","#5e8939","#477232","#567e41","#ffffff","#ffffff","#ffffff","#ffffff","#ffffff","#ffffff"],["#ffffff","#ffffff","#ffffff","#f7f9f7","#7da05c","#548324","#69943c","#5b853a","#4b7536","#497332","#38671f","#819f72","#ffffff","#ffffff","#ffffff","#ffffff"],["#ffffff","#ffffff","#ffffff","#b0c59e","#5a882a","#69933d","#6c963e","#557f39","#4b7536","#4d7737","#4c7636","#36651d","#beceb7","#ffffff","#ffffff","#ffffff"],["#ffffff","#ffffff","#ffffff","#bcceae","#5f8c31","#69933e","#66903d","#4a7436","#4c7636","#4d7737","#4c7636","#416e2a","#cdd8c7","#ffffff","#ffffff","#ffffff"],["#ffffff","#ffffff","#ffffff","#40750c","#618d33","#68933b","#588238","#4b7536","#4c7636","#4b7535","#487331","#406d28","#2d5f14","#ffffff","#ffffff","#ffffff"],["#ffffff","#ffffff","#ffffff","#f3f6f2","#a8bf90","#799e58","#5b8346","#4b7535","#4c7636","#587f43","#759564","#a8bc9d","#ffffff","#ffffff","#ffffff","#ffffff"]]
},
numbers: [
[
[0,1,1,0],
[1,0,0,1],
[1,0,0,1],
[1,0,0,1],
[0,1,1,0]
],
[
[0,1,0],
[1,1,0],
[0,1,0],
[0,1,0],
[1,1,1]
],
[
[1,1,1,0],
[0,0,0,1],
[0,1,1,0],
[1,0,0,0],
[1,1,1,1]
],
[
[1,1,1,0],
[0,0,0,1],
[0,1,1,0],
[0,0,0,1],
[1,1,1,0]
],
[
[0,0,1,0],
[0,1,1,0],
[1,0,1,0],
[1,1,1,1],
[0,0,1,0]
],
[
[1,1,1,1],
[1,0,0,0],
[1,1,1,0],
[0,0,0,1],
[1,1,1,0]
],
[
[0,1,1,0],
[1,0,0,0],
[1,1,1,0],
[1,0,0,1],
[0,1,1,0]
],
[
[1,1,1,1],
[0,0,0,1],
[0,0,1,0],
[0,1,0,0],
[0,1,0,0]
],
[
[0,1,1,0],
[1,0,0,1],
[0,1,1,0],
[1,0,0,1],
[0,1,1,0]
],
[
[0,1,1,0],
[1,0,0,1],
[0,1,1,1],
[0,0,0,1],
[0,1,1,0]
],
]
};
this.timer = setInterval(this.poll, 100);
this.poll();
return true;
}
// This breaks unless the parameter is a string
this.drawNumberedIcon = function(number) {
if (! (number instanceof String) ) {
number = number.toString();
}
if(!self.textedCanvas) {
self.textedCanvas = [];
}
if(!self.textedCanvas[number]) {
if (isDebugging()) { console.log("drawNumberedIcon(" + number + ")"); }
var iconCanvas = getIconCanvas();
var textedCanvas = document.createElement('canvas');
textedCanvas.height = textedCanvas.width = iconCanvas.width;
var ctx = textedCanvas.getContext('2d');
ctx.drawImage(iconCanvas, 0, 0);
ctx.fillStyle = this.backgroundFillColour;
ctx.strokeStyle = this.backgroundBorderColour;
ctx.strokeWidth = 1;
var count = number.length;
var bgHeight = self.pixelMaps.numbers[0].length;
var bgWidth = 0;
var padding = count > 2 ? 0 : 1;
for(var index = 0; index < count; index++) {
bgWidth += self.pixelMaps.numbers[number[index]][0].length;
if(index < count-1) {
bgWidth += padding;
}
}
bgWidth = bgWidth > textedCanvas.width-4 ? textedCanvas.width-4 : bgWidth;
ctx.fillRect(textedCanvas.width-bgWidth-4,2,bgWidth+4,bgHeight+4);
var digit;
var digitsWidth = bgWidth;
for(var index = 0; index < count; index++) {
digit = number[index];
if (self.pixelMaps.numbers[digit]) {
var map = self.pixelMaps.numbers[digit];
var height = map.length;
var width = map[0].length;
ctx.fillStyle = this.digitColour;
for (var y = 0; y < height; y++) {
for (var x = 0; x < width; x++) {
if(map[y][x]) {
ctx.fillRect(14- digitsWidth + x, y+4, 1, 1);
}
}
}
digitsWidth -= width + padding;
}
}
ctx.strokeRect(textedCanvas.width-bgWidth-3.5,2.5,bgWidth+3,bgHeight+3);
self.textedCanvas[number] = textedCanvas;
if (isDebugging()) { console.log("drawNumberedIcon: Done making icon"); }
}
return self.textedCanvas[number];
}
var resetTimer = function(init) {
var time = new Date().getTime();
self.lastActivity = time;
}
this.poll = function() {
var lastCount = getLastCount();
var time = new Date().getTime();
var count;
if (self.foundCountAlready) {
count = lastCount;
if (isAutoReloadEnabled()) {
// TODO: Maybe we shouldn't do this on explorer, analysis board, and a few other places
var lastCountUpdateTime = getLastCountUpdateTime();
var refreshTime = lastCountUpdateTime + self.dataTimeout;
if (self.lastActivity) {
var inactivityRefreshTime = self.lastActivity + self.inactivityTimeout;
if (inactivityRefreshTime > refreshTime) {
refreshTime = inactivityRefreshTime;
}
}
if (self.noReloadBefore) {
if (self.noReloadBefore > refreshTime) {
refreshTime = self.noReloadBefore;
} else {
// Some activity happened since this was set, so clear it and pick a new one next time
self.noReloadBefore = undefined;
}
}
var time = new Date().getTime();
if (isDebugging()) {
var d = new Date(refreshTime);
var formattedTime = d.getUTCHours() + ":" + (d.getUTCMinutes() < 10 ? "0" : "") + d.getUTCMinutes();
if (self.noReloadBefore) {
formattedTime += ":" + (d.getUTCSeconds() < 10 ? "0" : "") + d.getUTCSeconds()
} else {
formattedTime += " (ish)";
}
if (!self.lastDebugTime || self.lastDebugTime != formattedTime) {
self.lastDebugTime = formattedTime;
console.log("poll: Will reload page at " + formattedTime);
}
}
if (time > refreshTime) {
if (isDebugging()) { console.log("poll: Page reload timeout passed after " + ((self.pageLoadTime - time)/1000) + "sec"); }
if ( self.noReloadBefore ) {
// We already did a random period, and passed it, so we can reload now
self.noReloadBefore = undefined; // Probably unneeded, we'll lose this after the reload
location.reload();
} else {
// If we massively overshot the refresh time, it's possible that this machine was suspended
// (which is why we didn't get woken up)
// That can be a problem - often the CPU becomes active several seconds before the network, and so if we
// immediately reload, we will get an error
// Also, we don't want tabs all piling up and reloading at the same moment
// We therefore wait some extra random time before refreshing
self.noReloadBefore = time + reloadRandomizationMin + Math.ceil((reloadRandomizationMax-reloadRandomizationMin)*Math.random());
}
}
}
} else {
searchElement = findSearchElement();
if(!searchElement) {
if (isDebugging()) { console.log("poll: Search element not found, using last value"); }
count = lastCount;
} else {
if (isDebugging()) { console.log("poll: Found search element"); }
var lastCountUpdateTime = getLastCountUpdateTime();
if (lastCountUpdateTime && lastCountUpdateTime > time) {
if (isDebugging()) { console.log("poll: Stored count more recent"); }
// Some other page got a more up to date value
count = lastCount;
} else {
count = getCount();
if (count !== lastCount) {
if (isDebugging()) { console.log("Count updated to: " + count); }
setLastCount(count);
setLastCountChangeTime(time);
}
setLastCountUpdateTime(time);
}
self.foundCountAlready = true;
}
}
var displayCountIcon;
if (count == 0 || !showCount()) {
displayCountIcon = false;
} else {
var lastCountChangeTime = getLastCountChangeTime();
if (isFlashIconEnabled() && (time - lastCountChangeTime) < self.flashPeriod && (!self.lastActivity || self.lastActivity < lastCountTime)) {
displayCountIcon = (0 == (Math.floor(time / 1000) % 2));
} else {
displayCountIcon = true;
}
}
if(displayCountIcon) {
setIcon(self.drawNumberedIcon(count).toDataURL('image/png'));
} else {
setIcon(self.icons.read);
}
if (isTitleUpdatedEnabled()) {
if (count === 0) {
document.title = self.pageTitle;
} else {
document.title = "(" + count + ") " + self.pageTitle;
}
}
}
this.toString = function() { return '[object ChessDotComFavIconAlerts]'; }
this.pageLoadTime = new Date().getTime();
this.pageTitle = document.title;
window.addEventListener('mousemove', resetTimer);
window.addEventListener('click', resetTimer);
window.addEventListener('onkeypress', resetTimer);
if (isDebugging()) { console.log("Done creating ChessDotComFavIconAlerts"); } ;
return this.construct();
}