TBD: Quick Mention for TorrentBD

Quickly mention/tag users in the shoutbox and the forums.

// ==UserScript==
// @name         TBD: Quick Mention for TorrentBD
// @namespace    https://naeembolchhi.github.io/
// @version      0.9
// @description  Quickly mention/tag users in the shoutbox and the forums.
// @author       NaeemBolchhi
// @license      GPL-3.0-or-later
// @icon         
// @match        https://*.torrentbd.com/
// @match        https://*.torrentbd.net/
// @match        https://*.torrentbd.org/
// @match        https://*.torrentbd.me/
// @match        https://*.torrentbd.com/?spotlight
// @match        https://*.torrentbd.net/?spotlight
// @match        https://*.torrentbd.org/?spotlight
// @match        https://*.torrentbd.me/?spotlight
// @match        https://*.torrentbd.com/forums.php*
// @match        https://*.torrentbd.net/forums.php*
// @match        https://*.torrentbd.org/forums.php*
// @match        https://*.torrentbd.me/forums.php*
// @run-at       document-idle
// @grant        none
// ==/UserScript==

// Convert 6 digit HEX to RGB color
function getRGB(color) {
  color = color.replace('#','');
  return `rgb(${parseInt(color.replace(/....$/,''), 16)}, ${parseInt(color.replace(/..$/,'').replace(/^../,''), 16)}, ${parseInt(color.replace(/^..../,''), 16)})`;
}

// Determine active theme to set fill color
let fillColor;
if (document.body.className.match(/light/)) {
    fillColor = 'rgb(58, 58, 64)';
    if (localStorage.theme == 'light' && localStorage.themeLight) {
        fillColor = getRGB(JSON.parse(localStorage.themeLight).accent1);
    }
} else {
    fillColor = 'rgb(184, 198, 204)';
    if (localStorage.theme == 'dark' && localStorage.themeDark) {
        fillColor = getRGB(JSON.parse(localStorage.themeDark).accent1);
    }
}

// Stylesheet for quickmention in the shoutbox and quicktag in the forums
const quickCSS = `
  .at-them {
    cursor: pointer;
    color: transparent !important;
    background-image: url("data:image/svg+xml;utf8,<svg xmlns='http://www.w3.org/2000/svg' xmlns:xlink='http://www.w3.org/1999/xlink' style='fill: ${fillColor};' viewBox='0 0 915 300' xml:space='preserve'><g><path d='M200.4,204.3c-10.1,9.8-21.5,16.2-34.8,19.1c-40.5,9-81.6-18.2-89.4-59.1c-7.9-41.4,20.2-81.7,61.4-88.2   c21.2-3.4,40.3,1.9,57.2,15c2.6,2,5.1,2.9,8.4,2.8c7-0.2,14-0.1,21.4-0.1c0,2.2,0,3.9,0,5.6c0,22.8-0.1,45.7,0,68.5   c0.1,14,10.1,21.9,23.6,19c7.5-1.6,13-7.5,13.6-16.3c1.9-28.7-1.4-56.3-17.6-81.3c-25.2-38.8-72-59-116.2-49.7   c-46.3,9.7-81.2,45.7-89,91.9C28.2,195.2,73.8,255.1,138,261.9c22.9,2.4,44.4-2.2,64.8-12.8c0.9-0.5,1.8-0.9,3-1.5   c6.2,10.6,12.3,21,18.6,31.7c-10.7,6.5-21.8,11.2-33.4,14.6C106.2,319.1,16.9,263.1,2.1,175.5C-10.9,98.1,37,24.4,113,4.7   C196.7-17,282.8,39,297.1,124.8c3,18,4.2,36.2,0.8,54.2c-4.1,22.1-17.4,37.4-39.1,43.7c-21.5,6.3-40.4,0.5-56.1-15.5   C201.9,206.4,201.3,205.4,200.4,204.3z M187.1,150.1c0-20.4-16.9-37.4-37.3-37.6c-20.4-0.1-37.4,16.7-37.6,37.3   c-0.2,20.5,17,37.9,37.5,37.8C170.2,187.5,187.1,170.5,187.1,150.1z'/></g><g><path d='M914.8,165.4H743.5c2.5,15.1,9.1,27.1,19.8,36c10.7,8.9,24.4,13.4,41.1,13.4c19.9,0,37-7,51.4-20.9l44.9,21.1   c-11.2,15.9-24.6,27.6-40.2,35.3c-15.6,7.6-34.2,11.4-55.7,11.4c-33.3,0-60.5-10.5-81.4-31.5c-21-21-31.4-47.4-31.4-79   c0-32.4,10.4-59.3,31.3-80.8c20.9-21.4,47.1-32.1,78.6-32.1c33.5,0,60.7,10.7,81.6,32.1c21,21.4,31.4,49.7,31.4,84.9L914.8,165.4z    M861.5,123.4C858,111.6,851,102,840.6,94.5c-10.4-7.4-22.4-11.1-36.1-11.1c-14.9,0-27.9,4.2-39.1,12.5c-7,5.2-13.6,14.4-19.6,27.5   H861.5z'/><path d='M334.6,43.7h53.3v24.4c9.1-10,19.3-17.5,30.4-22.5c11.1-4.9,23.3-7.4,36.5-7.4c13.3,0,25.3,3.3,36,9.8   c10.7,6.5,19.3,16,25.8,28.5c8.5-12.5,18.9-22,31.2-28.5c12.3-6.5,25.8-9.8,40.4-9.8c15.1,0,28.4,3.5,39.9,10.5   c11.5,7,19.7,16.2,24.7,27.5s7.5,29.8,7.5,55.3v124.6h-53.7V148.4c0-24.1-3-40.4-9-48.9c-6-8.5-15-12.8-27-12.8   c-9.1,0-17.3,2.6-24.5,7.8c-7.2,5.2-12.6,12.4-16.1,21.6c-3.5,9.2-5.3,23.9-5.3,44.2v95.9h-53.7V153.3c0-19-1.4-32.8-4.2-41.3   c-2.8-8.5-7-14.9-12.6-19c-5.6-4.2-12.4-6.3-20.4-6.3c-8.9,0-16.9,2.6-24.2,7.9c-7.2,5.3-12.7,12.7-16.2,22.2   c-3.6,9.5-5.4,24.5-5.4,44.9v94.5h-53.3V43.7z'/></g></svg>");
    background-size: 85% auto;
    background-position: center right;
    background-repeat: no-repeat;
    transition: background-size .05s;
  }
  .at-them:active {
    background-size: 78% auto;
  }
  .quicktag a {
    height: 2rem;
    width: 2rem;
    cursor: pointer;
    transition: background 0s linear;
  }
  .quicktag.tagged a {
    background: var(--link-color);
    border-radius: 8px;
  }
  .quicktag.tagged a i {
    color: var(--main-bg);
  }
  ul.options.right {
    display: flex;
    align-items: center;
    justify-content: center;
    gap: 21.3px;
    margin-right: 18px;
    -webkit-touch-callout: none;
    -webkit-user-select: none;
    -khtml-user-select: none;
    -moz-user-select: none;
    -ms-user-select: none;
    user-select: none;
  }
  ul.options.right li {
    display: flex;
    padding-right: 0;
    padding-bottom: 4px;
  }
  ul.options.right li a {
    display: flex;
    align-items: center;
    justify-content: center;
  }
`;

// Adding styles for the hover effect
function addStyle() {
    if (!document.head) {window.location.reload();}
    let style = document.createElement('style');
    style.setAttribute('type', 'text/css');
    style.id = 'quickMention';
    style.innerHTML = quickCSS;
    document.head.appendChild(style);
}

// Shoutbox @username shortcut
function shoutBox() {
    document.addEventListener('click', function(event) { try {
        if (!event.target.classList.contains('shout-time')) {return;}

        let userName,
            shoutText = document.getElementById('shout_text');
        try {
            userName = event.target.parentNode.querySelector('.shout-user .tbdrank').innerText;
        } catch(e) {return;}

        shoutText.focus();
        shoutText.value = shoutText.value + '@' + userName + ' ';
        shoutText.value = shoutText.value.replace(/\ \ /g,' ').replace(/\ \ /g,' ');
    } catch(e) {}});

    document.addEventListener('mouseover', function(event) { try {
        if (!event.target.classList.contains('shout-time') || !event.target.parentNode.querySelector('.shout-user .tbdrank')) {return;}
        event.target.classList.add('at-them');
    } catch(e) {}});

    document.addEventListener('mouseout', function(event) { try {
        if (!event.target.classList.contains('shout-time') || !event.target.parentNode.querySelector('.shout-user .tbdrank')) {return;}
        event.target.classList.remove('at-them');
    } catch(e) {}});
}

// Add a new tag button to posts that don't have an edit button
function postTagger() {
    let allPosts = document.querySelectorAll('#middle-block div[id^=post]');

    for (let x = 0; x < allPosts.length; x++) {
        if (!allPosts[x].querySelector('a[href*="editpost"]')) {
            // NO EDIT BUTTON FOUND
            let tagContainer = allPosts[x].querySelector('ul.options.right');
            let tagElement = document.createElement('li');
            tagElement.classList.add('quicktag');
            tagElement.innerHTML = '<a class="tooltipped" data-position="bottom" data-relay="20" data-tooltip="Tag this User"><i class="material-icons sm">alternate_email</i></a>';
            tagContainer.insertBefore(tagElement, tagContainer.children[0]);
        };
    }
}

// Push a toast with needed message when tagging or untagging
function tagToast(key) {
    let toastJS = encodeURIComponent(`Materialize.toast('Tagging this user.', 2000, 'tag-toast')`);
    if (key === false) {
        toastJS = encodeURIComponent(`Materialize.toast('Not tagging this user.', 2000, 'untag-toast')`);
    }

    location.assign('javascript:' + toastJS);
}

// When tag button is clicked
function tagUser(key) {
    if (key.classList.contains('tagged')) {
        // untagging
        key.classList.remove('tagged');
        tagToast(false);
    } else {
        // tagging
        key.classList.add('tagged');
        tagToast();
    }
}

// Generate current taglist
function tagList() {
    let allPosts = document.querySelectorAll('#middle-block div[id^=post]'),
        userList = '';

    for (let x = 0; x < allPosts.length; x++) {
        if (allPosts[x].querySelector('.quicktag.tagged')) {
            userList = userList + ' ' + allPosts[x].querySelector('a[href*="account-details"]').innerText;
        }
    }

    return userList.replace(/^\s/,'').replace(/\s/g, ', ').replace(/\,\s\,\s/g,', ').replace(/\,\s$/,'').replace(/\,$/,'').replace(/\s$/,'');
}

// Tag and untag users in the forums
function forumTags() {
    document.addEventListener('click', function(event) {
        let target = event.target;

        // Clicking the new tag button
        if (target.classList.contains('quicktag')) {
            tagUser(target);
        }
        if (target.parentNode.classList.contains('quicktag')) {
            tagUser(target.parentNode);
        }
        if (target.parentNode.parentNode.classList.contains('quicktag')) {
            tagUser(target.parentNode.parentNode);
        }

        // Clicking the reply button (new post)
        if (target.id === 'reply-trigger' || target.parentNode.id === 'reply-trigger') {
            let inputField = document.getElementById('resource');
            inputField.value = tagList();
            if (inputField.value !== '') {
                inputField.parentNode.classList.remove('hidden');
            } else {
                inputField.parentNode.classList.add('hidden');
            }
        }

        // Clicking the taglist (adding tags to existing post)
        if ((target.getAttribute('href') === '#tagmodal' && target.getAttribute('onclick').match(/injectpostid/i)) || (target.parentNode.getAttribute('href') === '#tagmodal' && target.parentNode.getAttribute('onclick').match(/injectpostid/i))) {
            let inputField = document.getElementById('tagstext');
            inputField.value = tagList();
        }
    });
}

// Initiate parts of the script as needed
addStyle();
if (window.location.pathname === '/' || window.location.pathname === '/?spotlight') {
    shoutBox();
}
if (window.location.pathname === '/forums.php') {
    postTagger();
    forumTags();
}