lib:textjack

3/5/2025, 7:16:43 PM

// ==UserScript==
// @name        lib:textjack
// @version     6
// @match       *://*/*
// @run-at      document-start
// @author      rssaromeo
// @license     GPLv3
// @include     *
// @grant       none
// @icon        
// @description 3/5/2025, 7:16:43 PM
// @namespace https://greasyfork.org/users/1184528
// ==/UserScript==

let obslist = []
var textJackList = []
Object.assign(window, console)
const a = loadlib("allfuncs")

function replaceText(text) {
  var oldtext = text
  for (var cb of textJackList) {
    text = cb(text)
    if (text !== oldtext) {
      // error("changed", text, oldtext)
      return (text = replaceText(text))
    }
  }
  return text
}
// Function to handle text changes
function handleTextChange(mutations) {
  disableObservers()
  mutations.forEach((mutation) => {
    if (mutation.type === "childList") {
      mutation.addedNodes.forEach((node) => {
        if (node.nodeType === Node.TEXT_NODE) {
          // Filter text for text nodes
          var newtext = replaceText(node.textContent)
          if (node.textContent != newtext) {
            // trace("changed")
            node.textContent = newtext
            // log(1, node.textContent, newtext)
          }
        } else if (node.nodeType === Node.ELEMENT_NODE) {
          // debugger
          // If it's an element, check its text content
          node.childNodes.forEach((child) => {
            if (child.nodeType === Node.TEXT_NODE) {
              var newtext = replaceText(child.textContent)
              if (child.textContent != newtext) {
                // trace("changed")
                child.textContent = newtext
                // log(2, child.textContent, newtext)
              }
            }
          })
          // Observe any shadow roots
          if (node.shadowRoot) {
            createObserver(node.shadowRoot)
          }
        }
      })
    } else if (mutation.type === "characterData") {
      // Filter text for character data changes
      var text = mutation.target.textContent
      var newtext = replaceText(text)
      if (newtext !== text) {
        // trace("changed")
        mutation.target.textContent = newtext
      }
    }
  })
  enableObservers()
}

// Create a MutationObserver for the main document
function createObserver(elem) {
  var obs = new MutationObserver((mutations) => {
    handleTextChange(mutations)
    mutations.forEach((mutation) => {
      mutation.addedNodes.forEach((node) => {
        // If the added node is a shadow host, observe its shadow root
        if (node.nodeType === Node.ELEMENT_NODE && node.shadowRoot) {
          createObserver(node.shadowRoot)
        }
      })
    })
  })

  obslist.push([
    obs,
    elem,
    {
      childList: true,
      subtree: true,
      characterData: true,
    },
  ])
}
// Function to enable all observers
function enableObservers() {
  try {
    disableObservers()
  } catch (e) {}
  // warn(obslist, 1)
  obslist.forEach(([observer, target, opts]) =>
    observer.observe(target, opts)
  )
}

// Function to disable all observers
function disableObservers() {
  // warn(obslist, 0)
  obslist.forEach(([observer]) => observer.disconnect())
}

// Optionally, filter existing text content on page load
function filterExistingText(node) {
  disableObservers()
  if (node.nodeType === Node.TEXT_NODE) {
    var text = node.textContent
    var newtext = replaceText(text)
    if (newtext !== text) {
      // trace("changed")
      node.textContent = newtext
    }
  } else if (node.nodeType === Node.ELEMENT_NODE) {
    node.childNodes.forEach(filterExistingText)
    // If the element has a shadow root, filter its content
    if (node.shadowRoot) {
      filterExistingText(node.shadowRoot)
    }
  }
  enableObservers()
}

// Filter existing text in the document
var waitingForBody = false
if (document.body) {
  setup()
} else {
  waitingForBody = true
}

loadlib("libloader").savelib("textjack", function newTextJack(cb) {
  textJackList.push(cb)
  if (document.body) {
    setup()
  } else {
    waitingForBody = true
  }
})
a.bodyload().then(() => {
  if (waitingForBody) {
    setup()
  }
})

function setup() {
  filterExistingText(document.body)
  createObserver(document.body)
  enableObservers()
}