DanielBlaze / Reddit Base64 Decoder

// ==UserScript==
// @namespace   https://openuserjs.org/users/DanielBlaze
// @name        Reddit Base64 Decoder
// @description Automatically base64 decode links in reddit posts
// @copyright   2018, DanielBlaze (https://openuserjs.org/users/DanielBlaze)
// @updateURL   https://openuserjs.org/meta/DanielBlaze/Reddit_Base64_Decoder.meta.js
// @license     MIT
// @version     1.1.0
// @author      DanielBlaze
// @grant       none
// @match       https://*.reddit.com/r/*
// ==/UserScript==

// ==OpenUserJS==
// @author DanielBlaze
// ==/OpenUserJS==

const DEBUG = false;

/**
 * Create a log group if debug is enabled
 * @param {string} name - the group name
 */
function logGroup(name) {
  if (DEBUG) console.group(name);
}

/**
 * End a log group if debug is enabled
 * @param {string} name - the group name
 */
function logGroupEnd(name) {
  if (DEBUG) console.groupEnd(name);
}

/**
 * Log a message is debug is enabled
 * @param {string} msg
 */
function log(msg) {
  if (DEBUG) console.log(msg);
}

/**
 * Attempts to decode an array of elements. Styling will be lost on any element which is decoded
 * @param {HTMLElement[]} domElements
 */
function decode(domElements) {
  // for all pararaph tags
  domElements.forEach(domElement => {
    // split on the words we find, and make sure we have something before continuing
    const words = domElement.innerText.split(' ');
    if (words.length === 0) return;

    // decode the string
    let changed = false;
    const transformed = words.map(word => {
      try {
        const decoded = atob(word);

        // if it's valid, change it, and underline it. Secure links = underline in blue. Insecure links = warning image + underlined in red
        if (decoded.startsWith('https://')) {
          changed = true;
          return `<a style="border-bottom: 1px dotted #036;" target="_blank" href="${decoded}">${decoded}</a>`;
        } else if (decoded.startsWith('http://')) {
          changed = true;
          return `&nbsp;<img style="vertical-align: middle;" src="" />&nbsp; <a style="border-bottom: 1px dotted #f00;" target="_blank" href="${decoded}">${decoded}</a>`;
        }
      } catch (_) {
      }

      return word;
    });

    // only if we changed something should we update the HTML - otherwise we lose the original formatting
    if (changed) domElement.innerHTML = transformed.join(' ');
  });
}

(function () {
  'use strict';

  // check we have observers available to us
  if (!MutationObserver) return;

  // observer config - only interested in tree modifications
  const config = { childList: true, subtree: true };

  // the callback to decode the string
  const callback = function (mutationsList, observer) {
    logGroup('mutations');
    log(`Observing ${mutationsList.length} mutations`);

    let count = 0;

    mutationsList.forEach(mutation => {
      if (mutation.target.nodeName === 'DIV' && mutation.addedNodes.length > 0) {
        // check we can find the post contents
        const postContents = mutation.target.querySelectorAll('p');
        if (!postContents) return;

        // decode that bad boy
        decode(postContents);
        count++;
      }
    });

    log(`Tried decoding ${count} divs`);
    logGroupEnd('mutations');
  };

  // create an observer instance with the callback
  const observer = new MutationObserver(callback);

  // Start observing the target node for mutations
  observer.observe(document, config);

  // run on page load incase no mutations occur, but only on old reddit
  const postContainer = document.querySelector('.expando');
  if (postContainer) {
    const postContents = postContainer.querySelectorAll('p');
    if (postContents) {
      decode(postContents);
    }
  }
})();