Greasy Fork is available in English.

MierdiTwist

Notifies about new messages in tab's title & icon

// ==UserScript==
// @name         MierdiTwist
// @namespace    http://yelidmod.com/
// @version      0.1
// @description  Notifies about new messages in tab's title & icon
// @author       You
// @match        https://twist.com/a/*
// @grant        none
// ==/UserScript==

(function() {
    'use strict';

    const defaultFavicon = "https://twist.com/a/img/fav/favicon.ico";
    const warningFavicon = "";
    const pendingMessagesWarning = "(!)";

    const showWarning = () => {
        const title = document.querySelector('title').innerText;

        // warning already present or is looking at the tab
        if (title.includes(pendingMessagesWarning) || !document.hidden) {
            return;
        }
        document.querySelector('title').innerText = pendingMessagesWarning + " " + title;
        document.querySelectorAll('[rel="icon"], [rel="shortcut icon"]').forEach((el) => {
           el.href = warningFavicon;
        });
    }

    const hideWarning = () => {
        document.querySelector('title').innerText = document.querySelector('title').innerText.replace(pendingMessagesWarning, "");
        document.querySelectorAll('[rel="icon"], [rel="shortcut icon"]').forEach((el) => {
           el.href = defaultFavicon;
        });
    };

    // whenever user opens twist tab, hide the warnings
    document.addEventListener("visibilitychange", () => {
         if (!document.hidden) {
             hideWarning();
         }
    }, false);

    (function(originalFetch) {
        window.fetch = function(url, options) {
            if (!url.includes("getMessages")) {
                return originalFetch(url, options);
            }

            // apply MITM
            return new Promise((resolve, reject) => {
                originalFetch(url, options).then((response) => {
                    let result = response.clone();

                    resolve(response);

                    result.json().then((json) => {
                        json.data.forEach((event) => {
                            // private message
                            if (event.type == "message_added" && event.to_user == true) {
                                showWarning();
                            }
                        });
                    });
                }).catch(reject);
            });
        };
    })(fetch);
})();