// ==UserScript==
// @name Asana Notifications favicon
// @grant none
// @namespace https://greasyfork.org/en/scripts/6118-asana-notifications-favicon
// @version 0.4
// @description Shows a red badge on the favicon if the inbox has unread messages
// @match https://app.asana.com/*
// @copyright 2014+, Jordi Gimenez Gamez
// ==/UserScript==
(function() {
// changes the favicon. avoids setting the same favicon twice.
function updateFavicon(unread) {
var readIcon = "https://d1gwm4cf8hecp4.cloudfront.net/images/favicon.ico";
var unreadIcon = "";
if(this.lastState != unread) { // ensures we're not changing the icon all the time
var faviconLink = document.evaluate("//link[@rel='shortcut icon']").iterateNext();
faviconLink.href = unread ? unreadIcon : readIcon;
this.lastState = unread;
}
}
// checks read count and updates favicon if necessary
function updateRead() {
var unread = document.getElementsByClassName('inbox-count').length > 0;
updateFavicon(unread);
}
// observes any changes in the body and checks for changes in read count
function install() {
// this is costly. any change in ANY view will call the callback, but I can't find a better way without polling
if(this.observer === undefined) {
this.observer = new MutationObserver(function(mutations) {
updateRead();
// middle ground: disable observation for some time after an event
observer.disconnect();
setTimeout(install, 10000 /* 10 sec */);
});
}
this.observer.observe(document.body, { childList: true, subtree: true });
updateRead(); // make sure it's up to date before first event
}
// checks that the required Inbox Counter hack is enabled
function checkFeatures() {
function alertCSS(message) {
// original CSS here: http://jsfiddle.net/joshnh/FxfHc/
var css = document.createElement("style");
css.type = "text/css";
css.innerHTML = "#cssalert{position:relative;z-index:9999;}#cssalert:hover:after{background:hsla(0,0%,0%,.8);border-radius:3px;color:#f6f6f6;content:'Click to dismiss';font:700 12px/30px sans-serif;height:30px;left:50%;margin-left:-60px;position:absolute;text-align:center;top:50px;width:120px}#cssalert:hover:before{border-bottom:10px solid hsla(0,0%,0%,.8);border-left:10px solid transparent;border-right:10px solid transparent;content:'';height:0;left:50%;margin-left:-10px;position:absolute;top:40px;width:0}#cssalert:target{display:none}.cssalert{background-color:#c4453c;background-image:-webkit-linear-gradient(135deg,transparent,transparent 25%,hsla(0,0%,0%,.05) 25%,hsla(0,0%,0%,.05) 50%,transparent 50%,transparent 75%,hsla(0,0%,0%,.05) 75%,hsla(0,0%,0%,.05));background-image:-moz-linear-gradient(135deg,transparent,transparent 25%,hsla(0,0%,0%,.1) 25%,hsla(0,0%,0%,.1) 50%,transparent 50%,transparent 75%,hsla(0,0%,0%,.1) 75%,hsla(0,0%,0%,.1));background-image:-ms-linear-gradient(135deg,transparent,transparent 25%,hsla(0,0%,0%,.1) 25%,hsla(0,0%,0%,.1) 50%,transparent 50%,transparent 75%,hsla(0,0%,0%,.1) 75%,hsla(0,0%,0%,.1));background-image:-o-linear-gradient(135deg,transparent,transparent 25%,hsla(0,0%,0%,.1) 25%,hsla(0,0%,0%,.1) 50%,transparent 50%,transparent 75%,hsla(0,0%,0%,.1) 75%,hsla(0,0%,0%,.1));background-image:linear-gradient(135deg,transparent,transparent 25%,hsla(0,0%,0%,.1) 25%,hsla(0,0%,0%,.1) 50%,transparent 50%,transparent 75%,hsla(0,0%,0%,.1) 75%,hsla(0,0%,0%,.1));background-size:20px 20px;box-shadow:0 5px 0 hsla(0,0%,0%,.1);color:#f6f6f6;display:block;font:700 16px/40px sans-serif;height:40px;position:absolute;text-align:center;text-decoration:none;top:-45px;width:100%;-webkit-animation:cssalert 1s ease forwards;-moz-animation:cssalert 1s ease forwards;-ms-animation:cssalert 1s ease forwards;-o-animation:cssalert 1s ease forwards;animation:cssalert 1s ease forwards}@-webkit-keyframes cssalert{0%{opacity:0}50%{opacity:1}100%{top:0}}@-moz-keyframes cssalert{0%{opacity:0}50%{opacity:1}100%{top:0}}@-ms-keyframes cssalert{0%{opacity:0}50%{opacity:1}100%{top:0}}@-o-keyframes cssalert{0%{opacity:0}50%{opacity:1}100%{top:0}}@keyframes cssalert{0%{opacity:0}50%{opacity:1}100%{top:0}}";
document.body.appendChild(css);
var div = document.createElement("div");
div.id = "cssalert";
var a = document.createElement("a");
a.className = "cssalert";
a.href = "#cssalert";
a.text = message;
div.appendChild(a);
document.body.insertBefore(div, document.body.firstChild); // at the top
}
if(!ExperimentalFeature.enabled({name:"inbox_new_count"})) {
alertCSS('In order for Asana Notifications favicon to work, you need to enable Inbox Counter in Account Settings > Hacks');
}
}
// let asana settle down
setTimeout(function() {
checkFeatures();
install();
}, 10000 /* 10 sec */);
}());