Skip to content

Instantly share code, notes, and snippets.

@jpbochi
Last active January 12, 2022 14:21
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save jpbochi/f651698e371f7deda33d039ddaa2bda5 to your computer and use it in GitHub Desktop.
Save jpbochi/f651698e371f7deda33d039ddaa2bda5 to your computer and use it in GitHub Desktop.
Asana dynamic favicon to show Inbox notifications
// ==UserScript==
// @name Asana Dynamic Favicon
// @namespace https://gist.githubusercontent.com/jpbochi/f651698e371f7deda33d039ddaa2bda5
// @version 0.1.0
// @description This modern Asana Inbox favicon notification displays an orange notification circle (just like Asana does in the app) to the top right of the Asana favicon when your active Organization/Workspace Inbox has any unread items. Enjoy!
// @author JP Bochi
// @match https://app.asana.com/*
// @grant none
// @run-at document-body
// @downloadURL https://gist.githubusercontent.com/jpbochi/f651698e371f7deda33d039ddaa2bda5/raw/asana-favicon-notification.js
// @updateURL https://gist.githubusercontent.com/jpbochi/f651698e371f7deda33d039ddaa2bda5/raw/asana-favicon-notification.js
// ==/UserScript==
// initially forked from https://greasyfork.org/en/scripts/29414-asana-dynamic-favicon by Joe Thomas
(function () {
let lastState = null;
const updateFavicon = (hasNotifications) => {
var readIcon = "";
var unreadIcon = "";
if (lastState !== hasNotifications) {
console.debug('=>> Updating favicon...', { hasNotifications });
var faviconLink = document.evaluate("//link[@rel='shortcut icon']", document, null, XPathResult.ANY_TYPE, null).iterateNext();
faviconLink.href = hasNotifications ? unreadIcon : readIcon;
document.head.appendChild(faviconLink);
lastState = hasNotifications;
}
}
window.updateFavicon = updateFavicon;
const getAddedNodes = (mutations) => (
Array.from(mutations)
.filter(mutation => (mutation.type === 'childList'))
.map(mutation => mutation.addedNodes)
.flatMap(nodes => Array.from(nodes || []))
);
const didAddOrRemoveNodes = (mutations) => (
Array.from(mutations)
.filter(mutation => (mutation.type === 'childList'))
.find(mutation => mutation.addedNodes.length || mutation.removedNodes.length)
);
const queryAll = (nodes, query) => (
[
...nodes.filter(node => node.matches && node.matches(query)),
...nodes.flatMap(node => Array.from(node.querySelectorAll ? node.querySelectorAll(query) : [])),
]
);
const waitForSideBar = () => (new Promise((resolve, reject) => {
const loadedSidebar = document.querySelector('#asana_sidebar');
if (loadedSidebar) return Promise.resolve(loadedSidebar);
const observer = new MutationObserver((mutations) => {
const addedNodes = getAddedNodes(mutations);
const [sidebar] = queryAll(addedNodes, '#asana_sidebar');
if (sidebar) {
observer.disconnect();
console.debug('=>> Found sidebar.', { sidebar });
return resolve(sidebar);
}
const [fullscreen] = queryAll(addedNodes, '#asana_full_page');
if (fullscreen) {
observer.disconnect();
console.debug('=>> Found fullscreen. No sidebar expected.', { fullscreen });
return reject();
}
});
observer.observe(document.body, { subtree: true, childList: true });
}));
const install = () => {
waitForSideBar().then((sidebar) => {
const lookForNotifications = () => {
const hasNotifications = !!sidebar.querySelector('.NotificationsIndicator');
updateFavicon(hasNotifications);
};
// Update icon once before starting the MutationObserver
lookForNotifications();
const observer = new MutationObserver((mutations) => {
console.debug('=>>', { mutations });
if (didAddOrRemoveNodes(mutations)) {
lookForNotifications();
}
});
observer.observe(sidebar, {
subtree: true,
childList: true,
characterData: false,
attributes: false,
});
}, () => {});
};
install();
})();
@jpbochi
Copy link
Author

jpbochi commented Jan 12, 2022

The icons should look like these:

image

image

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment