Kirka.IO Enhanced

Wallhack, adblock and more for Kirka.IO.

// ==UserScript==
// @name Kirka.IO Enhanced
// @namespace -
// @version 1.5.0
// @description Wallhack, adblock and more for Kirka.IO.
// @author NotYou
// @match *://kirka.io/*
// @run-at document-end
// @license GPL-3.0-or-later
// @grant GM.info
// @icon 
// ==/UserScript==

(function() {
    const TITLE = (GM && GM.info && GM.info.script && GM.info.script.name) || 'Kirka.IO Enhanced'

    let wallhackEnabled = false

    const KEYS = {
        CHAT_VISIBILITY_TOGGLE: 'V',
        WALLHACK_TOGGLE: 'F'
    }

    let players = []

    class Visibility {
        static visible(el) {
            el.style.opacity = '1'
            el.style.pointerEvents = 'auto'
        }

        static invisible(el) {
            el.style.opacity = '0'
            el.style.pointerEvents = 'none'
        }

        static toggle(el) {
            if(el.style.opacity === '0') {
                return this.visible(el)
            }

            this.invisible(el)
        }
    }

    class Wallhack {
        static visible(e) {
            e.alphaTest = .99
            e.fog = false
            e.depthTest = false
        }

        static invisible(e) {
            e.alphaTest = 1
            e.fog = true
            e.depthTest = true
        }
    }

    class CSS {
        static init() {
            const _CSS = `
            .notification {
              position: absolute;
              background: var(--secondary-5);
              border: 4px solid rgb(62, 77, 124);
              border-bottom: 4px solid var(--secondary-6);
              border-top: 4px solid rgb(77, 92, 139);
              width: 250px;
              height: 100px;
              right: 15px;
              bottom: 15px;
              z-index: 100;
              color: rgb(255, 255, 255);
              transition: .3s;
              opacity: 0.85;
              pointer-events: none;
            }

            .notification-title {
              font-size: x-large;
              text-align: center;
              margin: 2px;
            }

            .notification-body {
              margin: 3px;
              font-size: medium;
            }

            .highlight-disabled::after {
              content: 'disabled';
              color: rgb(210, 50, 50);
            }

            .highlight-enabled::after {
              content: 'enabled';
              color: rgb(50, 210, 50);
            }

            .bloody-screen {
              pointer-events: none !important;
              width: 100vw;
              height: 100vh;
              position: absolute;
              left: 0;
              top: 0;
              box-shadow: rgba(255, 0, 0, .8) 0 0 150px inset;
              transition: .3s opacity;
            }`

            addStyle(_CSS)
        }
    }

    class AdBlock {
        static init() {
            let adBlockCss

            let ads = ['ad-left', 'ad-right', 'ad-bottom', 'ad-change-weapon']
            let adsSelectors = ''

            ads.forEach(e => {
                let adSelector = ''
                let prefixes = ['.', '#']

                prefixes.forEach(r => {
                    adSelector += r + e + ','
                })

                adsSelectors += adSelector
            })

            adBlockCss = adsSelectors.slice(0, -1) + '{ display: none !important }'

            addStyle(adBlockCss)

            window.show_rewarded = _
            window.show_preroll = _
        }
    }

    class Keybindings {
        static init() {
            let keybidings = [
                // Toggle Chat Visibility
                {
                    key: KEYS.CHAT_VISIBILITY_TOGGLE,
                    fn() {
                        let chat = document.querySelector('.chat')

                        if(chat) {
                            chat.style.transition = '.3s opacity'
                            Visibility.toggle(chat)
                        }
                    }
                },

                // Toggle Wallhack
                {
                    key: KEYS.WALLHACK_TOGGLE,
                    fn() {
                        wallhackEnabled = !wallhackEnabled

                        if(wallhackEnabled) {
                            players.forEach(visibleEach)
                        } else {
                            players.forEach(invisibleEach)
                        }

                        function visibleEach(e) {
                            Wallhack.visible(e)
                        }

                        function invisibleEach(e) {
                            Wallhack.invisible(e)
                        }

                        notify('Wallhack', 'Wallhack is ', wallhackEnabled)
                    }
                },
            ]

            keybidings.forEach(e => {
                window.addEventListener('keydown', r => {
                    const focusedElem = document.querySelector(':focus') || document.activeElement

                    if(focusedElem.matches('.chat .input')) {
                        return
                    }

                    if(r.code === 'Key' + e.key.toUpperCase()) {
                        r.preventDefault()
                        e.fn()
                    }
                })
            })
        }
    }

    class BloodyScreen {
        static init() {
            const bloodyScreen = document.createElement('div')
            bloodyScreen.className = 'bloody-screen'
            Visibility.invisible(bloodyScreen)
            document.body.appendChild(bloodyScreen)

            const obs = new MutationObserver(() => {
                const progress = document.querySelector('.hp-progress')


                if(progress) {
                    const hp = currentHealth()

                    if(hp < 35 && hp > 0) {
                        Visibility.visible(bloodyScreen)
                    } else {
                        Visibility.invisible(bloodyScreen)
                    }
                }
            })

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

            function currentHealth() {
                return +document.querySelector('.hp-progress').style.width.slice(0, -1)
            }
        }
    }

    class _Wallhack {
        static init() {
            Object.defineProperty(Object.prototype, 'material', {
                set(value) {
                    this._material = value

                    if (this._material && this._material.name && this._material.name.indexOf('player') !== -1) {
                        players.push(value)

                        if(wallhackEnabled) {
                            Wallhack.visible(value)
                        }
                    }
                },

                get() {
                    return this._material
                }
            })
        }
    }

    class Main {
        static init() {
            const MODULES = [
                CSS,
                AdBlock,
                Keybindings,
                BloodyScreen,
                _Wallhack,
            ]

            initModules(MODULES)
        }
    }

    Main.init()

    function initModules(modules) {
        for (let i = 0; i < modules.length; i++) {
            const MODULE = modules[i]

            initModule(MODULE)
        }
    }

    function initModule(module) {
        try {
            module.init()
        } catch(e) {
            console.error(TITLE, module.name + ' module, has error:', e)
        }
    }

    function notify(title, body, highlight) {
        const notifClass = 'notification'
        const notif = document.createElement('div')
        const notifTitle = document.createElement('h3')
        const notifBody = document.createElement('p')

        notifTitle.className = notifClass + '-title'
        notifBody.className = notifClass + '-body'
        notif.className = notifClass

        notifTitle.textContent = title ?? 'Kirka.IO Enchanced'
        notifBody.innerHTML = (body ?? '') + (`<span class="highlight-${highlight ? 'enabled' : highlight === false ? 'disabled' : ''}"></span>`)

        notif.style.right = '100vw'

        notif.appendChild(notifTitle)
        notif.appendChild(notifBody)
        document.body.appendChild(notif)

        setTimeout(() => {
            notif.style.right = ''
        }, 300)

        setTimeout(() => {
            notif.style.right = '100vw'

            setTimeout(() => {
                notif.remove()
            }, 300)
        }, 1300)
    }

    function addStyle(css) {
        const styleNode = document.createElement('style')
        styleNode.appendChild(document.createTextNode(css))
        document.head.appendChild(styleNode)
    }

    function _() {}
})()