// ==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 // @downloadURL https://update.greasyfork.icu/scripts/257/Chesscom%20Favicon%20Alerts.user.js // @updateURL https://update.greasyfork.icu/scripts/257/Chesscom%20Favicon%20Alerts.meta.js // ==/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(); }