Chain Warn v2

description

Você precisará instalar uma extensão como Tampermonkey, Greasemonkey ou Violentmonkey para instalar este script.

Você precisará instalar uma extensão como Tampermonkey ou Violentmonkey para instalar este script.

Você precisará instalar uma extensão como Tampermonkey ou Violentmonkey para instalar este script.

Você precisará instalar uma extensão como Tampermonkey ou Userscripts para instalar este script.

Você precisará instalar uma extensão como o Tampermonkey para instalar este script.

Você precisará instalar um gerenciador de scripts de usuário para instalar este script.

(Eu já tenho um gerenciador de scripts de usuário, me deixe instalá-lo!)

Você precisará instalar uma extensão como o Stylus para instalar este estilo.

Você precisará instalar uma extensão como o Stylus para instalar este estilo.

Você precisará instalar uma extensão como o Stylus para instalar este estilo.

Você precisará instalar um gerenciador de estilos de usuário para instalar este estilo.

Você precisará instalar um gerenciador de estilos de usuário para instalar este estilo.

Você precisará instalar um gerenciador de estilos de usuário para instalar este estilo.

(Eu já possuo um gerenciador de estilos de usuário, me deixar fazer a instalação!)

// ==UserScript==
// @name         Chain Warn v2
// @namespace    namespace
// @version      0.9
// @description  description
// @author       tos
// @match        *.torn.com/*
// @grant        GM_addStyle
// @license      MIT
// ==/UserScript==

const alert_time = 100 //seconds
const min_chain = 100
const alert_sound = true

GM_addStyle(`
.cw_overlay {
  position: fixed;
  pointer-events: none;
  height: 100%;
  width: 100%;
  z-index: 20000;
  will-change: background-color;
}

.cw_idle::after {
  background-color: #FF03;
  bottom: 0;
  content: "";
  left: -10px;
  position: absolute;
  right: -10px;
  top: 0;
}

.cw_warn {
  animation-name: cw_warn;
  animation-duration: 1.5s;
  animation-iteration-count: infinite;
}

@keyframes cw_warn {
  0% {
    background-color: #FF000000;
  }
  50% {
    background-color: #FF000055;
  }
  100 {
    background-color: #FF000000;
  }
}
`)

let warn_overlay = null
function chain_value () {
  return parseInt(document.querySelector('[class^=chain-bar] [class^=bar-value]')?.innerText.replace(',', '').split('/')[0])
}

// playTone(gain, frequency, duration)
let a = new AudioContext()
function playTone(w,x,y){
  //console.log("Gain:"+w, "Hz:"+x, "ms:"+y)
  osc = a.createOscillator()
  gain_node = a.createGain()
  osc.connect(gain_node)
  osc.frequency.value = x
  osc.type = "square"
  gain_node.connect(a.destination)
  gain_node.gain.value = w * 0.01
  osc.start(a.currentTime)
  osc.stop(a.currentTime + y *0.001)
}
function alert() {
  playTone(10,233,100)
  playTone(3,603,200)
}
let alarm = null

const chainbar_observer = new MutationObserver((mutations) => {
  for (const mutation of mutations) {
    try {
      if (mutation.target.parentElement.className.includes('bar-timeleft')) {
        //const a = performance.now()
        const t = mutation.target.data.split(':')
        const s = parseInt(t[0]) * 60 + parseInt(t[1])
        if (0 < s && s < alert_time && t.length < 3 && min_chain <= chain_value()) {
          warn_overlay.classList.add('cw_warn')
          if (!alarm && alert_sound) alarm = setInterval(alert, 1500)
        }
        else {
          warn_overlay.classList.remove('cw_warn')
          clearInterval(alarm)
          alarm = null
        }
        //const b = performance.now()
        //console.log(b-a)
      }
    } catch(err) { console.error(err)}
  }
})


function waitForSelector(selector) {
  return new Promise((resolve, reject) => {
    if (document.querySelector(selector)) {
      return resolve(document.querySelector(selector))
    }

    const observer = new MutationObserver(mutations => {
      if (document.querySelector(selector)) {
        resolve(document.querySelector(selector))
        observer.disconnect()
      }
    })

    observer.observe(document.body, {
      childList: true,
      subtree: true
    })
  })
}


waitForSelector('A[class^=chain-bar]').then((sidebar) => {
  chainbar = {}
  chainbar.A = document.querySelector('[class^=chain-bar]')
  chainbar.barstats = document.querySelector('[class^=chain-bar] [class^=bar-stats]')
  if (chainbar.A && chainbar.barstats) {
    document.querySelector('body').insertAdjacentHTML('afterbegin', `<div class="cw_overlay"></div>`)
    warn_overlay = document.querySelector('.cw_overlay')
    chainbar.A.classList.add('cw_idle')
    chainbar_observer.observe(chainbar.barstats, { subtree: true, characterData: true })
  }
}).catch(console.error)