Janka Tietzova

A user script for The West, optimized Dobby2 for butt plugs and cake decorations.

Script này sẽ không được không được cài đặt trực tiếp. Nó là một thư viện cho các script khác để bao gồm các chỉ thị meta // @require https://update.greasyfork.org/scripts/512872/1465973/Janka%20Tietzova.js

// ==UserScript==
// @name            Janka Tietzova
// @namespace       http://tampermonkey.net/
// @version         2.1.1
// @description     A user script for The West, optimized Dobby2 for butt plugs and cake decorations.
// @author          Chvostnatý Gábor
// @include         https://*.the-west.*/game.php*
// @license         MIT
// @icon            
// @grant           none
// @downloadURL https://update.greasyfork.org/scripts/497915/Zdenka%20Studenkova.user.js
// @updateURL https://update.greasyfork.org/scripts/497915/Zdenka%20Studenkova.meta.js
// ==/UserScript==




(function() {
    async function sleep(milliseconds) {  
        await new Promise(r => setTimeout(r, milliseconds))
    }

    function parseConsumableBonuses(bonuses) {
        const { energyText, motivationText, healthText, speedText, luckText, dropText, experienceText, moneyText } = Vajda.searchKeys[Vajda.language]

        const result = {
            energy: 0,
            health: 0,
            motivation: 0,
            hasBuffs: false,
            buffs: {
                character: {
                    luck: 0,
                    money: 0,
                    experience: 0,
                    drop: 0
                },
                travel: 0
            }
        }

        function getValue(bonus, bonusText) {
            return Number(bonus.toLowerCase().replace(bonusText.toLowerCase(), '').replace(' ', '').replace('%', '').replace('+', ''))
        }
        
        for (const bonus of bonuses) {
            if ( bonus.includes(energyText) ) {
                result.energy = getValue(bonus, energyText)
                continue
            }
            
            if ( bonus.includes(motivationText) ) {
                result.motivation = getValue(bonus, motivationText)
                continue
            }

            if ( bonus.includes(healthText) ) { 
                result.health = getValue(bonus, healthText)
                continue
            }

            if ( bonus.includes(speedText) ) {
                result.buffs.travel = getValue(bonus, speedText)
                result.hasBuffs = true
                continue
            }

            if ( bonus.includes(luckText) ) {
                result.buffs.character.luck = getValue(bonus, luckText)
                result.hasBuffs = true
                continue
            }

            if ( bonus.includes(experienceText) ) {
                result.buffs.character.experience = getValue(bonus, experienceText)
                result.hasBuffs = true
                continue
            }

            if ( bonus.includes(moneyText) ) {
                result.buffs.character.money = getValue(bonus, moneyText)
                result.hasBuffs = true
                continue
            }

            if ( bonus.includes(dropText) ) {
                result.buffs.character.drop = getValue(bonus, dropText)
                result.hasBuffs = true
            }
        }

        return result
    }


    class Job {
        constructor(x, y, id, groupId) {
            this.x = x
            this.y = y
            this.id = id
            this.groupId = groupId
            this.isSilver = false
            this.distances = []
            this.experience = 0
            this.money = 0
            this.motivation = 0
            this.stopMotivation = 75
            this.set = -1
            this.bestEquipment_ = undefined
        }

        setIsSilver(val) {
            this.isSilver = val
            return this
        }

        setExperience(val) {
            this.experience = val
            return this
        }

        setMoney(val) {
            this.money = val
            return this
        }

        setMotivation(valOrUpdater) {
            if ( typeof valOrUpdater === 'function' ) {
                this.motivation = valOrUpdater(this.motivation)
            } else {
                this.motivation = valOrUpdater
            } 

            this.motivation = Math.min(100, this.motivation)

            return this
        }

        setStopMotivation(val) {
            this.stopMotivation = val
            return this
        }

        setSet(index) {
            this.set = index
            return this
        }

        getDistances() {
            return this.distances
        }

        async loadMotivation() {
            await Ajax.get('job', 'job', {jobId: this.id, x: this.x, y: this.y}, (r) => {
                this.motivation = r.motivation * 100
            })

            return this.motivation
        }

        /**
         * Calculate job distance to current character position
         */
        calculateDistance() {
            this.distance = GameMap.calcWayTime({x: this.x, y: this.y}, Character.position)
            return this.distance
        }

        /**
         * Calculate job distance to other selected jobs
         */
        calculateJobDistances() {
            this.distances = []
            for ( const job of Vajda.addedJobs ) {
                this.distances.push(GameMap.calcWayTime({x: this.x, y: this.y}, {x: job.x, y: job.y}))
            }
            return this.distances
        }


        saveBestEquipment() {
            const skills = JobsModel.Jobs.find(j => j.id === this.id).get('skills')
            const set = west.item.Calculator.getBestSet(skills, this.id)
            const itemIds = set && set.getItems() || []
            this.bestEquipment_ = itemIds
            return itemIds
        }

        get bestEquipment() {
            return this.bestEquipment_ === undefined ? this.saveBestEquipment() : this.bestEquipment_
        }


        getBestEquipment() {
            if (!Bag.loaded) {
                EventHandler.listen('inventory_loaded', function() {
                    this.getBestEquipment()
                    return EventHandler.ONE_TIME_EVENT
                })
                return
            }
            
            const items = Bag.getItemsByItemIds(this.bestEquipment)
            const result = []

            for (const item of items) {
                const wearItem = Wear.get(item.getType())
                if (!wearItem || (wearItem && (wearItem.getItemBaseId() !== item.getItemBaseId() || wearItem.getItemLevel() < item.getItemLevel()))) {
                    result.push(item)
                }
            }
            return result.map(item => item.obj.getId())
        }
    }

    class Consumable {
        constructor(id, image, name) {
            this.id = id
            this.image = image
            this.name = name
            this.energy = 0
            this.motivation = 0
            this.health = 0
            this.isSelected_ = false
            this.count = 0
        }

        isCakeDecoration() {
            return this.id === 53339000
        }


        removeOne() {
            this.count -= 1

            if ( this.count === 0 ) {
                const consumables = Manager.consumables
                const index = consumables.findIndex(c => c.id === this.id)
                consumables.splice(index, 1)
            }
            return this
        }

        setCount(valOrUpdater) {
            if ( typeof valOrUpdater === 'function' ) {
                this.count = valOrUpdater(this.count)
            } else {
                this.count = valOrUpdater
            }

            this.count = Math.max(0, this.count)
            return this
        }

        setEnergy(val) {
            this.energy = val
            return this
        }

        setMotivation(val) {
            this.motivation = val
            return this
        }

        setHealth(val) {
            this.health = val
            return this
        }

        setIsSelected(isSelected) {
            this.isSelected_ = isSelected
            return this
        }

        hasCooldown() {
            return this.energy > 0 || this.motivation > 0 || this.health > 0
        }

        getBuffHTML() {
            return '<strong>No Buffs</strong>'
        }

        get isSelected() {
            return this.isSelected_
        }
    }

    class Buff extends Consumable {
        constructor(id, image, name, buffs) {
            super(id, image, name, buffs)
            this.buffs_ = buffs
        }

        getBuffImage() {
            return `https://westsk.innogamescdn.com/images/buffs/${this.getBuffType()}.jpg`
        }

        getBuffType() {
            return this.buffs_.travel === 0 ? 'character' : 'travel'
        }

        canUseAsBuff() {
            return CharacterSkills.buffs[this.getBuffType()] === null && !this.hasCooldown() 
        }

        getBuffHTML() {
            const containerStyle = `'
                display: grid;
                grid-template-columns: 30px auto;
            '`

            const imgContainerStyle = `'
                display: flex;
                padding-bottom: 5px;
            '`

            const buffsContainerStyle = `'
                display: grid;
                grid-template-rows: repeat(auto-fill, 15px);
                transform: translateY(5px)
            '`
            const buffs = this.buffs.reduce((acc, b) => acc.concat(`<p>${b}</p>`), '')

            const html = `
                <div style=${containerStyle}>
                    <span style=${imgContainerStyle}>
                        <img style=align-self:end width=25px height=25px src=${this.getBuffImage()} alt='Buff Image'>
                    </span>

                    <span style=${buffsContainerStyle}>
                        ${
                            buffs
                        }
                    </span>
                </div>
            `

            return html
        }

        removeOne() {
            this.count -= 1

            if ( this.count === 0 ) {
                const consumables = Manager.consumables
                const buffs = Manager.buffs
                let index = consumables.findIndex(c => c.id === this.id)
                if ( index !== -1 ) {
                    consumables.splice(index, 1)
                    return this
                }
                index = buffs.findIndex(b => b.id === this.id)
                if ( index !== -1 ) {
                    buffs.splice(index, 1)
                    if ( Manager.selectedBuffs.some(b => b.id === this.id) ) {
                        Manager.selectedBuffs_[this.getBuffType()] = null
                    }
                }
            }
            return this
        }

        get isSelected() {
            return Manager.selectedBuffs.some(b => b.id === this.id) || this.isSelected_
        }


        get buffs() {
            if ( this.getBuffType() === 'travel' ) {
                return [`+${this.buffs_.travel}% speed`]
            }

            const buffs = []
            for ( const buff in this.buffs_.character ) {
                if ( this.buffs_.character[buff] > 0 ) {
                    buffs.push(`+${this.buffs_.character[buff]}% ${buff}`)
                }  
            }
            return buffs
        }
    }


    class ConsumablesManager {
        selectedBuffs_ = {
            travel: null,
            character: null
        }

        constructor() {
            this.consumables_ = []
            this.buffs_ = []
            this.isOptimized_ = false
            this.jobsLeftInRound_ = 0
            this.schedule_ = []
            this.selectedBuffs_ = {
                travel: null,
                character: null
            }
        }

        async loadJobMotivation(updatedJobsMotivation = undefined) {
            let expectedJobCount = 0
            const uniqueJobsCount = Vajda.addedJobs.length

            if ( updatedJobsMotivation !== undefined ) {
                expedtedJobCount = updatedJobsMotivation.reduce((acc, curr) => acc + curr, 0)

                return { expectedJobCount, uniqueJobsCount }
            }

            for ( const job of Vajda.addedJobs ) {
                expectedJobCount += Math.max(await job.loadMotivation() - job.stopMotivation, 0)

                await sleep(500)
            }

            return { expectedJobCount, uniqueJobsCount }
        }

        async createSchedule(updatedJobsMotivation) {
            const bottlePlugs = this.consumables.find(c => c.id === 52871000)

            //in case the game runs in the background and the job to travel to is not canceled we gonna need extra energy point
            //this is unlikely to happen but the energy wont go to waste anyway so why the fuck not
            const { expectedJobCount, uniqueJobsCount } = await this.loadJobMotivation(updatedJobsMotivation)

            this.jobsLeftInRound = expectedJobCount
    
            const availableEnergy = Character.energy
            const energyToFill = expectedJobCount + uniqueJobsCount - availableEnergy
            const refillCount = Math.ceil(
                energyToFill / ( bottlePlugs.energy / 100 * Character.maxEnergy )
            )

            //use plugs when n jobs left in round
            const schedule = []

            for ( let i = 0; i < refillCount; i++ ) {
                //40 15s jobs in consumable cooldown
                schedule.push(40 + i*40)
            }

            this.schedule_ = schedule
        
            return schedule
        }

        isScheduledRefill() {
            return this.isOptimized_ && this.refillsLeft > 0 && this.jobsLeftInRound <= this.schedule.at(-1) 
        }

        async checkSchedule(jobCount) {
            const bottlePlug = this.consumables.find(c => c.id === 52871000)

            for ( let i = 0; i < jobCount; i++ ) {
                this.jobsLeftInRound = p => p - 1
                if ( this.isScheduledRefill() ) {
                    //the use of consumable blocks the start of new jobs for a short amount of time, so this sleep exists to prevent that
                    await sleep(10000)
                    this.useConsumableOrWaitForCooldown(bottlePlug)
                    this.schedule.pop()
                    this.jobsLeftInRound = p => p - (jobCount - i - 1)
                    return
                }
            }

            if ( Character.energy - jobCount <= 10 ) {
                //same here
                await sleep(10000)
                this.useConsumableOrWaitForCooldown(bottlePlug)
                this.schedule.pop()
            }
        }

        canUseConsumable(consumable) {
            if ( consumable instanceof Buff ) {
                return consumable.canUseAsBuff()
            }

            if ( BuffList.cooldowns[consumable.id] !== undefined && BuffList.cooldowns[consumable.id].time > new ServerDate().getTime() ) {
                return false
            }
            return true
        } 

        findProperConsumable(motivationMissing, energyMissing, averageMotivationMissing) {
            const consumablesPool = this.getSelectedConsumables()

            function betterEnergy(item1, item2) {
                let distanceItem1 = Math.abs(energyMissing - item1.energy)
                let distanceItem2 = Math.abs(energyMissing - item2.energy)
                return (distanceItem1 < distanceItem2 ) ? -1 : (distanceItem1 > distanceItem2) ? 1 : 0
            }
            function betterMotivation(item1, item2) {
                let distanceItem1 = Math.abs(averageMotivationMissing - item1.motivation)
                let distanceItem2 = Math.abs(averageMotivationMissing - item2.motivation)
                return (distanceItem2 < distanceItem1) ? item2 : item1
            }
            function findMotivationConsume(consumes) {
                let consumeToChoose = null
                for ( let i = 0; i < consumes.length; i++ ) {
                    if ( consumeToChoose === null && consumes[i].motivation !== 0) {
                        consumeToChoose = consumes[i]
                        continue
                    }
                    
                    if ( consumeToChoose !== null && consumes[i].motivation !== 0) {
                        consumeToChoose = betterMotivation(consumeToChoose, consumables[i])
                    }
                }
                return consumeToChoose
            }
            function findHealthConsume(consumes) {
                for ( let i = 0; i < consumes.length; i++ ) {
                    if ( consumes[i].health !== 0 ) {
                        return consumes[i]
                    }
                }
                return null
            }

            if ( consumablesPool.length  === 0 ) return null
    
            const consumables = consumablesPool.sort(betterEnergy)
            
            if ( Vajda.settings.addEnergy && energyMissing === 100 ) {
                return consumables[0]
            }
    
            if ( Vajda.settings.addMotivation && motivationMissing === Vajda.addedJobs.length ) {
                return findMotivationConsume(consumables) 
            }
    
            if ( Vajda.settings.addHealth && Vajda.isHealthBelowLimit() ) {
                if ( this.isOptimized ) {
                    this.schedule.pop()
                }
                return findHealthConsume(consumables)
            }
        }

        async useConsumable(consumable) {
            const item = Bag.getItemByItemId(consumable.id)
            item.showCooldown()

            if ( Vajda.shouldEquipHealthSet(consumable) )
                await Vajda.equipSet(Vajda.healthSet)

            ItemUse.doIt(consumable.id)
            consumable.removeOne()     
            
            while(true) {
                if ( !this.canUseConsumable(consumable) ) {
                    $('.tw2gui_dialog_framefix').remove()
                    break
                }
                await sleep(1)
            }

            Vajda.currentState = 1
        }

        async useConsumableOrWaitForCooldown(consumableOrId, isSync = false) {
            const consumable = consumableOrId instanceof Consumable ? consumableOrId : this.getSelectedConsumables().find(c => c.id === consumableOrId)
            if ( consumable === undefined ) {
                return false
            }

            Vajda.currentState = 2

            while (true) {
                if ( consumable.hasCooldown() === false || this.canUseConsumable(consumable) ) {
                    break
                }

                if ( !Vajda.isRunning ) {
                    break
                }

                await sleep(1000)
            }

            await this.useConsumable(consumable)
            
            isSync && Vajda.run()
        }

        isConsumableAdded (item) {
            if ( item === undefined )
                return true
            for ( const consumable of this.consumables_ ) {
                if ( consumable.id === item.obj.item_id ) {
                    return true
                }
            }
            for ( const buff of this.buffs_ ) {
                if ( buff.id === item.obj.item_id ) {
                    return true
                }
            }
            return false
        }

        useBuff(type) {
            const buff = this.selectedBuffs_[type]

            if ( buff?.canUseAsBuff() ) {
                this.useConsumable(buff)
            }
        }

        addNewConsumable(item) {
            if ( this.isConsumableAdded(item) ) {
                return
            }
            
            const { energy, motivation, health, hasBuffs, buffs } = parseConsumableBonuses(item.obj.usebonus)

            if ( health === 0 && motivation === 0 && energy === 0 && !hasBuffs )
                return

            if ( hasBuffs ) {
                const buff = new Buff(item.obj.item_id, item.obj.image, item.obj.name, buffs)
                    .setEnergy(energy)
                    .setMotivation(motivation)
                    .setHealth(health)
                    .setCount(item.count)
                
                return buff.hasCooldown() ? this.consumables_.push(buff) : this.buffs_.push(buff)
            }
            
            const consumable = new Consumable(item.obj.item_id, item.obj.image, item.obj.name)
                .setEnergy(energy)
                .setMotivation(motivation)
                .setHealth(health)
                .setCount(item.count)
            this.consumables.push(consumable)
        }

        hasEnoughPlugsAndDecorations() {
            const bottlePlugs = this.consumables.find(c => c.id === 52871000)
            const decorations = this.consumables.find(c => c.id === 53339000)

            if ( !bottlePlugs || !decorations ) {
                new UserMessage("No plugs or decorations were found, defaulting back to selected consumables", UserMessage.TYPE_HINT).show()
                this.isOptimized_ = false
                return false
            }
            return true
        }

        getSelectedConsumables() {
            if ( this.isOptimized_ ) {
                const bottlePlugs = this.consumables.find(c => c.id === 52871000)
                const decorations = this.consumables.find(c => c.id === 53339000)

                return [bottlePlugs, decorations]
            } else {
                return this.consumables.filter(c => c.isSelected === true)
            }
        }

        addSelectedConsumable(val) {
            if ( typeof val === 'number' )
                this.consumables.at(val).isSelected = true

            if ( val instanceof Consumable ) 
                this.consumables.find(c => c.id === val.id).isSelected = true
        }

        get schedule() {
            return this.schedule_
        }

        set schedule(val) {
            this.schedule_ = val
        }

        get consumables() {
            return this.consumables_
        }

        set consumables(val) {
            this.consumables_ = val
        }

        get isOptimized() {
            return this.isOptimized_
        }

        set isOptimized(valOrUpdater) {
            if ( typeof valOrUpdater === 'function' ) {
                this.isOptimized_ = valOrUpdater(this.isOptimized_)
            } else {
                this.isOptimized_ = valOrUpdater
            }
        }

        get jobsLeftInRound() {
            return this.jobsLeftInRound_
        }
        
        set jobsLeftInRound(valOrUpdater) {
            if ( typeof valOrUpdater === 'function' ) {
                this.jobsLeftInRound_ = valOrUpdater(this.jobsLeftInRound_)
            } else {
                this.jobsLeftInRound_ = valOrUpdater
            }
        }

        get refillsLeft() {
            return this.schedule.length
        }

        get buffs() {
            return this.buffs_
        }

        get selectedBuffs() {
            const res = []
            for ( const type in this.selectedBuffs_ ) {
                this.selectedBuffs_[type] && res.push(this.selectedBuffs_[type])
            }

            return res
        }

        set selectedBuffs(buff) {
            const type = buff.getBuffType()
            if ( this.selectedBuffs_[type]?.id === buff.id ) {
                this.selectedBuffs_[type] = null

                return
            }
            this.selectedBuffs_[type] = buff
            new UserMessage(`New ${type} buff selected`, UserMessage.TYPE_SUCCESS).show()
        }
    }


    class ActivityObserver {
        constructor() {
            const fiveMinutes = 5 * 60 * 1000
            this.activityCheckTimeout_ = null
            this.refreshCount = 0
            this._isEnabled = false
            this.timeOut_ = fiveMinutes
            this.selectedConsumables = null
        }

        saveTempCookies() {
            const selectedConsumables = Manager.consumables.reduce((acc, c) => {
                c.isSelected && acc.push(c.id)
                return acc
            }, [])
            const timeOut = this.timeOut_     
            const buffs = {
                travel: Manager.selectedBuffs_.travel?.id || null,
                character: Manager.selectedBuffs_.character?.id || null
            }
            
            const settings = {
                statistics: Vajda.statistics,
                selectedConsumables,
                buffs,
                timeOut,
                direction: Vajda.direction,
                currentJob: Vajda.currentJob,
                refreshCount: this.refreshCount + 1,
                schedule: Manager.schedule,
                jobsLeftInRound: Manager.jobsLeftInRound,
                travelSet: Vajda.travelSet,
                jobSet: Vajda.jobSet,
                healthSet: Vajda.healthSet
            }
            
            const expiracyDateTemporary = new Date()
            const hour = expiracyDateTemporary.getHours()
            expiracyDateTemporary.setHours(2,0,0)

            if ( hour > 2 )
                expiracyDateTemporary.setDate(expiracyDateTemporary.getDate() + 1)

            document.cookie = `vajdasession=${JSON.stringify(settings)};expires=${expiracyDateTemporary.toGMTString()};`
        }

        getTempCookies() {
            const cookies = document.cookie.split("=")
            for ( let i = 0; i < cookies.length; i++ ) {
                if ( cookies[i].includes("vajdasession") ) {
                    const { 
                        timeOut, schedule, selectedConsumables, buffs, jobsLeftInRound, refreshCount, ...vajdaSettings 
                    } = JSON.parse(cookies[i+1].split(";")[0])

                    this.selectedConsumables = selectedConsumables
                    this.buffs = buffs
                    this.refreshCount = refreshCount
                    this.timeOut_ = timeOut
                    this.isEnabled = true
                    Manager.jobsLeftInRound = jobsLeftInRound
                    Manager.schedule = schedule
                    Vajda = {...Vajda, ...vajdaSettings, isRunning: true}
                    document.cookie = 'vajdasession=; expires=Thu, 01 Jan 1970 00:00:00 UTC; path=/;'
                    return true
                }
            }
            
            return false
        }

        restartSession() {
            if ( Vajda.currentState === 2 ) {
                this.start()
                return
            }

            this.saveTempCookies()
            location.reload()
        }

        resumeSession() {
            Vajda.findAllConsumables()
            if ( !this.getTempCookies() ) return
            Vajda.createWindow(false)
            Vajda.run()
        }

        start(forceStart = false) {
            if ( !this.isEnabled && !forceStart ) return

            clearTimeout(this.activityCheckTimeout_)
            this.activityCheckTimeout_ = setTimeout(this.restartSession.bind(this), this.timeOut_)
        }

        stop() {
            this.isEnabled = false
            if ( this.activityCheckTimeout_ !== null )
                clearTimeout(this.activityCheckTimeout_)
            this.activityCheckTimeout_ = null
        }

        getTimeOut(convertToMinutes) {
            if ( convertToMinutes ) {
                return this.timeOut_ / 1000 / 60
            }
            return this.timeOut_
        }

        get isEnabled() {
            return this._isEnabled
        }

        set isEnabled(valOrUpdater) {
            if ( typeof valOrUpdater === 'function' ) {
                this._isEnabled = valOrUpdater(this._isEnabled)
            } else {
                this._isEnabled = valOrUpdater
            }
        }

        set timeOut(val) {
            this.timeOut_ = Math.max(Number(val) * 60 * 1000, 2.5 * 60 * 1000)
        }
    }

    window.Manager = new ConsumablesManager()
    window.Observer = new ActivityObserver()


    Vajda = {
        addedJobs: [],
        allJobs: [],
        addedJobTablePosition: {
            content: "0px",
            scrollbar:"0px"
        },
        consumableSelection: {
            energy: false,
            motivation: false,
            health: false,
            hideBuffs: false
        },
        consumableTablePosition: {
            content: '0px',
            scrollbar: '0px'
        },
        consumableUsed: [],
        currentJob: {
            job: 0,
            direction: true
        },
        currentState: 0,
        healthSet: -1,
        language: "",
        isLoaded: false,
        isRunning: false,
        jobFilter: {
            filterOnlySilver: false,
            filterNoSilver: false,
            filterCenterJobs: false,
            filterJob: ""
        },
        jobSet: -1,
        jobsLoaded: false,
        jobTablePosition: {
            content: "0px",
            scrollbar: "0px"
        },
        lastJobStartTime: undefined,
        states: [
            'Idle',
            'Running',
            'Waiting for a consumable cooldown',
            'Calculating optimal route'
        ],
        statistics: {
            sessionJobsCount: 0,
            sessionXpCount: 0,
            sessionMoneyCount: 0,
            totalJobsCount: 0,
            totalXpCount: 0,
            totalMoneyCount: 0
        },
        selectedSet: 0,
        sets: null,
        settings: {
            addEnergy: false,
            addMotivation: false,
            addHealth: false,
            healthStop: 10,
            setWearDelay: 5,
            jobDelayMin: 0,
            jobDelayMax: 0,
            addDeposit: {
                isEnabled: false,
                limit: NaN
            }
        },
        sortJobTableXp:0,
        sortJobTableDistance:0,
        travelSet: -1,
        regenerationSet: -1,
        window: null,
        searchKeys: {
            "en_DK":{
                energy:"Energy",
                energyText:"Energy increase:",
                motivation:"Work motivation",
                motivationText:"Work motivation increase:",
                health: "Health point bonus",
                healthText:"Health point bonus:"
            },
            "sk_SK": {
                energy:"Energie",
                energyText:"Zvýšenie energie:",
                motivation:"Pracovnej motivácie",
                motivationText:"Zvýšenie pracovnej motivácie:",
                health: "Bonus bodov zdravia",
                healthText:"Bonus bodov zdravia:",
                speedText: "Rýchlosť",
                experience: "skúseností",
                experienceText: "Skúseností z práce, duelov a boja o pevnosť",
                money: "peňazí",
                moneyText: "Peňazí z práce a duelov",
                luck: "šťastie",
                luckText: "Vylepšené šťastie",
                drop: "produktu",
                dropText: "Vylepšenie šance na získanie produktu"
            },
            "cs_CZ":{
                energy:"Energie",
                energyText:"Zvýšení energie:",
                motivation:"Pracovní motivace",
                motivationText:"Zvýšení pracovní motivace:",
                health: "Bonus zdraví",
                healthText:"Bonus zdraví:",
                speedText: "Rychlost",
                experience: "zkušeností",
                experienceText: "zkušeností z prací, duelů a bitev o pevnost",
                money: "prací",
                moneyText: "$ z prací a duelů",
                luck: "štěstí",
                luckText: "Vyšší šance na štěstí",
                drop: "produktu",
                dropText: "Vylepšené šance nálezu produktu"
            },
            "hu_HU":{
                energy:"Energia növekedése:",
                energyText:"Energia növekedése:",
                motivation:"Munka motiváció növelése:",
                motivationText:"Munka motiváció növelése:",
                health: "Életerő bónusz",
                healthText:"Életerő bónusz:"
            },
            "pl_PL":{
                energy:"Wzrost energii:",
                energyText:"Wzrost energii:",
                motivation:"Zwiększenie motywacji do pracy:",
                motivationText:"Zwiększenie motywacji do pracy:",
                health: "Bonus Punktów życia:",
                healthText:"Bonus Punktów życia:"
            },
            "ro_RO":{
                energy:"Energie mărită:",
                energyText:"Energie mărită:",
                motivation:"Creştere a motivaţiei de muncă:",
                motivationText:"Creştere a motivaţiei de muncă:",
                health: "Puncte de viaţă:",
                healthText:"Puncte de viaţă:"
            }
         }
    }


    Vajda.isNumber = function(potentialNumber) {
        return Number.isInteger(parseInt(potentialNumber))
    }

    Vajda.RNG = function(min, max) {
        let minN = Math.min(min, max)
        let maxN = Math.max(min, max)

        let number =  Math.floor((minN + Math.random() * (maxN - minN + 1)))
        return number
    }


    Vajda.isAllowedToDepositMoney = function() {
        const hasMoreMoneyThanLimit = Vajda.settings.addDeposit.limit <= Character.money
        const hasHomeTown = Character.homeTown.town_id !== 0

        if ( hasHomeTown == false ) {
            new UserMessage("You don't have a home town", UserMessage.TYPE_HINT).show()
            return false
        }
        
        return Vajda.settings.addDeposit.isEnabled && hasMoreMoneyThanLimit
    }


    Vajda.getCookies = function() {
        const cookies = document.cookie.split("=")
        for ( let i = 0; i < cookies.length; i++ ) {
            if ( cookies[i].includes("vajdatemporary") ) {
                const obj = cookies[i+1].split(";")
                const tempObject = JSON.parse(obj[0])
                const tmpAddedJobs = tempObject.addedJobs

                for ( const job of tmpAddedJobs ) {
                    const newJob = new Job(job.x, job.y, job.id, job.groupId)
                        newJob
                            .setIsSilver(job.isSilver)
                            .setExperience(job.experience)
                            .setMoney(job.money)
                            .setMotivation(job.motivation)
                            .setStopMotivation(job.stopMotivation)   
                            .setSet(job.set)
                        Vajda.addedJobs.push(newJob)
                    }

                Vajda.travelSet = tempObject.travelSet
                Vajda.jobSet = tempObject.jobSet
                Vajda.healthSet = tempObject.healthSet
                Vajda.currentJob = tempObject.currentJob
                Vajda.setSetForAllJobs()
            }
            if ( cookies[i].includes("vajdapermanent") ) {
                const obj = cookies[i+1].split(";")
                const permanentObject = JSON.parse(obj[0])
                const { isOptimized, isEnabled, timeOut, ...settings } = permanentObject.settings

                Manager.isOptimized = !!isOptimized
                Observer.isEnabled = !!isEnabled
                Observer.timeOut = timeOut || 5
                Vajda.settings = settings
                if ( !settings.addDeposit ) {
                    Vajda.settings.addDeposit = {
                        isEnabled: false,
                        limit: NaN
                    }
                }
                Vajda.statistics.totalJobsCount = Math.floor(permanentObject.totalJobs)
                Vajda.statistics.totalXpCount = permanentObject.totalXp
                Vajda.statistics.totalMoneyCount = permanentObject.totalMoney
            }
        }
    }

    Vajda.setCookies = function() {
        let expiracyDateTemporary = new Date()
        let hour = expiracyDateTemporary.getHours()
        expiracyDateTemporary.setHours(2,0,0)
        if ( hour > 2 )
            expiracyDateTemporary.setDate(expiracyDateTemporary.getDate() + 1)

        const addedJobs = Vajda.addedJobs.map(j => ({
            x: j.x,
            y: j.y,
            id: j.id,
            groupId: j.groupId,
            isSilver: j.isSilver,
            experience: j.experience,
            money: j.money,
            motivation: j.motivation,
            stopMotivation: j.stopMotivation,
            set: j.set
        }))
        
        let temporaryObject = {
            addedJobs,
            travelSet: Vajda.travelSet,
            jobSet: Vajda.jobSet,
            healthSet: Vajda.healthSet,
            currentJob: Vajda.currentJob
        }
        let expiracyDatePernament = new Date()
        expiracyDatePernament.setDate(expiracyDatePernament.getDate() + 360000)
        let pernamentObject = {
            settings: {...Vajda.settings, isOptimized: Manager.isOptimized, isEnabled: Observer.isEnabled, timeOut: Observer.getTimeOut(true)},
            totalJobs: Vajda.statistics.totalJobsCount,
            totalXp: Vajda.statistics.totalXpCount,
            totalMoney: Vajda.statistics.totalMoneyCount
        }
        const jsonTemporary = JSON.stringify(temporaryObject)
        const jsonPernament = JSON.stringify(pernamentObject)
        document.cookie = `vajdatemporary=${jsonTemporary};expires=${expiracyDateTemporary.toGMTString()};`
        document.cookie = `vajdapermanent=${jsonPernament};expires=${expiracyDatePernament.toGMTString()};`
    }

    
    Vajda.shouldEquipHealthSet = function(consumable) {
        if ( !consumable.hasCooldown() ) {
            return false
        }

        if ( Manager.isOptimized ) {
            return false
        }

        return consumable.health > 0 && Vajda.healthSet > -1
    }


    Vajda.parseConsumableBonuses = function(bonuses) {
        let getBonus = (text,type) => {
            switch(type) {
                case 0:
                    text = text.replace(Vajda.searchKeys[Vajda.language].energyText,"")
                    break
                case 1:
                    text = text.replace(Vajda.searchKeys[Vajda.language].motivationText,"")
                    break
                case 2:
                    text = text.replace(Vajda.searchKeys[Vajda.language].healthText,"")
                    break
            }
            text = text.slice(1)
            text = text.replace("%","")
            return parseInt(text)
        }
        let result = Array(3).fill(0)
        for ( let i = 0 ; i < bonuses.length; i++ ) {
            let type = -1
            if ( bonuses[i].includes(Vajda.searchKeys[Vajda.language].energyText) ) {
                type = 0
            }else if ( bonuses[i].includes(Vajda.searchKeys[Vajda.language].motivationText) ) {
                type = 1
            }else if ( bonuses[i].includes(Vajda.searchKeys[Vajda.language].healthText) ) {
                type = 2
            }
            if ( type !=-1 )
            result[type] = getBonus(bonuses[i],type)

        }
        return result
    }

    Vajda.findAllConsumables = function() {
        if ( Vajda.searchKeys[Vajda.language] === undefined ) return
        const energyConsumables = Bag.search(Vajda.searchKeys[Vajda.language].energy)
        for ( const consumable of energyConsumables ) {
            Manager.addNewConsumable(consumable)
        }

        const motivationConsumables = Bag.search(Vajda.searchKeys[Vajda.language].motivation)
        for ( const consumable of motivationConsumables ) {
            Manager.addNewConsumable(consumable)
        }

        const healthConsumables = Bag.search(Vajda.searchKeys[Vajda.language].health)
        for ( const consumable of healthConsumables ) {
            Manager.addNewConsumable(consumable)
        }

        const speedConsumables = Bag.search(Vajda.searchKeys[Vajda.language].speedText)
        for ( const consumable of speedConsumables ) {
            if ( consumable.obj.usetype !== 'none' ) {
                Manager.addNewConsumable(consumable)
            }
        }

        const luckConsumables = Bag.search(Vajda.searchKeys[Vajda.language].luck)
        for ( const consumable of luckConsumables ) {
            if ( consumable.obj.usetype !== 'none' ) {
                Manager.addNewConsumable(consumable)
            }
        }

        const experienceConsumables = Bag.search(Vajda.searchKeys[Vajda.language].experience)
        for ( const consumable of experienceConsumables ) {
            if ( consumable.obj.usetype !== 'none' ) {
                Manager.addNewConsumable(consumable)
            }
        }

        const moneyConsumables = Bag.search(Vajda.searchKeys[Vajda.language].money)
        for ( const consumable of moneyConsumables ) {
            if ( consumable.obj.usetype !== 'none' ) {
                Manager.addNewConsumable(consumable)
            }
        }

        const dropConsumables = Bag.search(Vajda.searchKeys[Vajda.language].drop)
        for ( const consumable of dropConsumables ) {
            if ( consumable.obj.usetype !== 'none' ) {
                Manager.addNewConsumable(consumable)
            }
        }

        Observer.selectedConsumables?.forEach(id => {
            Manager.consumables.find(c => c.id === id).setIsSelected(true)
        })

        for ( const type in Observer.buffs ) {
            Manager.selectedBuffs_[type] = Manager.buffs_.find(b => b.id === Observer.buffs[type]) || null
        }
        Observer.selectedConsumables = null
        Observer.buffs = null
    }

    Vajda.filterConsumables = function(energy, motivation, health, hideBuffs) {
        const result = hideBuffs ? [] : [...Manager.buffs] 

        for ( const consumable of Manager.consumables ) {
            if ( energy && consumable.energy === 0 ) {
                continue
            }

            if ( motivation && consumable.motivation === 0 ) {
                continue
            }

            if ( health && consumable.health === 0 ) {
                continue
            }
            result.push(consumable)
        }

        return result
    }

    Vajda.changeConsumableSelection = function(id, isSelected) {
        Manager.consumables.find(c => c.id === id)?.setIsSelected(isSelected)
    }

    Vajda.changeSelectionAllConsumables = function(selected) {
        for ( const consumable of Manager.consumables ) {
            consumable.setIsSelected(selected)
        }
    }

    Vajda.loadJobs = function() {
        if ( !Vajda.jobsLoaded ) {
            new UserMessage("Loading...", UserMessage.TYPE_HINT).show()
            let index = 0
            let currentLength = 0
            let maxLength = 299
            Ajax.get('map', 'get_minimap', {}, function(r) {
                var tiles = []
                var jobs = []

                for ( let townNumber in r.towns ) {
                    if ( r.towns[townNumber].town_id === Character.homeTown.town_id ) {
                        Vajda.homeTown = r.towns[townNumber]
                        break
                    }
                }


                for ( let jobGroup in r.job_groups ) {
                    const groupId = parseInt(jobGroup)
                    let group = r.job_groups[jobGroup]
                    let jobsGroup = JobList.getJobsByGroupId(groupId)

                    for ( let tilecoord = 0; tilecoord < group.length; tilecoord++ ) {
                        let xCoord = Math.floor(group[tilecoord][0]/GameMap.tileSize)
                        let yCoord = Math.floor(group[tilecoord][1]/GameMap.tileSize)
                        if ( currentLength == 0 ) {
                            tiles[index] = []
                        }

                        tiles[index].push([xCoord,yCoord])
                        currentLength++

                        if ( currentLength === maxLength ) {
                            currentLength = 0
                            index++
                        }

                        for ( let i = 0; i < jobsGroup.length; i++ ) {
                            jobs.push(new Job(group[tilecoord][0],group[tilecoord][1],jobsGroup[i].id, groupId))
                        }
                    }
                }
                let toLoad = tiles.length
                let loaded = 0
                for ( let blocks = 0; blocks < tiles.length; blocks++ ) {
                    GameMap.Data.Loader.load(tiles[blocks], function() {
                        loaded++
                        if ( loaded === toLoad ) {
                            Vajda.jobsLoaded = true
                            Vajda.allJobs = jobs
                            Vajda.findAllConsumables()
                            Vajda.createWindow()
                        }
                    })
                }
            })
        } else {
            Vajda.findAllConsumables()
            Vajda.createWindow()
        }
    }

    Vajda.loadJobData = function(callback) {
        Ajax.get('work','index', {}, function(r) {
            if( r.error ) {
                console.log(r.error)
                return
            }
            JobsModel.initJobs(r.jobs)
            callback()
        })
    }

    Vajda.getJobSet = function(x, y, id) {
        const job = Vajda.findAddedJob(x, y, id)
        if ( job !== null )
            return job.set
    }

    Vajda.setJobSet = function(x,y,id,set) {
        const job = Vajda.findAddedJob(x, y, id)
        if ( job !== null)
            return job.setSet(set)
    }

    Vajda.findAddedJob = function(x, y, id) {
        for ( const job of Vajda.addedJobs ) {
            if ( job.x === x && job.y === y && job.id === id ) {
                return job
            }
        }
        return null
    }

    Vajda.loadSets = async function(callback) {
        Ajax.remoteCallMode('inventory', 'show_equip', {}, function(r) {
            Vajda.sets = r.data
            if ( callback !== undefined ) 
                callback()
        })
    }

    Vajda.loadJobMotivation = function(index, callback) {
        const job = Vajda.addedJobs.at(index)
        Ajax.get('job', 'job', {jobId: job.id, x: job.x, y: job.y}, function(r) {
            if ( callback !== undefined )
                callback(r.motivation*100)
        })
    }

    Vajda.loadLanguage = function() {
        Ajax.remoteCall("settings", "settings", {}, function(resp) {
            Vajda.language = resp.lang.account.key
        })
    }

    Vajda.getConsumableIcon = function(src) {
        return `<div><img src='${src}'></div>`
    }

    Vajda.getItemImage = function(id) {
        return ItemManager.get(id).wear_image
    }

    Vajda.createComboxJobSets = function(x, y, id) {
        let combobox = new west.gui.Combobox()
        Vajda.addComboboxItems(combobox)
        combobox = combobox.select(Vajda.getJobSet(x, y, id))
        combobox.setWidth(60)
        combobox.addListener(function(value) {
            Vajda.setJobSet(x, y, id, value)
            Vajda.selectTab("chosenJobs")
        });
        return combobox.getMainDiv()
    }

    Vajda.addComboboxItems = function(combobox) {
        combobox.addItem(-1, "None")
        for ( let i = 0 ; i < Vajda.sets.length; i++) {
            combobox.addItem(i.toString(), Vajda.sets[i].name)
        }
    }

    Vajda.updateJobDistances = function() {
        for ( let i = 0; i < Vajda.allJobs.length; i++ ) {
            Vajda.allJobs[i].calculateDistance()
        }
    }
    
    Vajda.findJobData = function(job) {
        for ( let i = 0 ;i < JobsModel.Jobs.length; i++ ) {
            if ( JobsModel.Jobs[i].id === job.id ) {
                return JobsModel.Jobs[i]
            }
        }
    }

    Vajda.findJob = function(x, y, id) {
        for ( let i = 0; i < Vajda.allJobs.length; i++ ) {
            if ( Vajda.allJobs[i].id === id && Vajda.allJobs[i].x === x && Vajda.allJobs[i].y === y ) {
                return Vajda.allJobs[i]
            }
        }
    }

    Vajda.addJob = function(x, y, id) {
        if ( !Vajda.checkIfJobAdded(id) ) {
            Vajda.addedJobs.push(Vajda.findJob(x, y, id))
        }
    }

    Vajda.removeJob = function(x, y, id) {
        for ( let i = 0; i < Vajda.addedJobs.length; i++ ) {
            if ( Vajda.addedJobs[i].id === id && Vajda.addedJobs[i].x === x && Vajda.addedJobs[i].y === y) {
                Vajda.addedJobs.splice(i,1)
                Vajda.consolidePosition(i)
                break
            }
        }
    }

    Vajda.parseJobData = function(jobs) {
        for ( let job = 0; job < jobs.length; job++ ) {
            let currentJob = jobs[job]
            let data = Vajda.findJobData(currentJob)
            let xp = data.basis.short.experience
            let money = data.basis.short.money
            currentJob.setMotivation(data.jobmotivation*100)
            if ( currentJob.isSilver ) {
                xp = Math.ceil(1.5*xp)
                money = Math.ceil(1.5*money)
            }
            currentJob.setExperience(xp)
            currentJob.setMoney(money)
        }
    }

    Vajda.getJobName = function(id) {
        return JobList.getJobById(id).name
    }

    Vajda.getJobIcon = function(isSilver, id, x, y) {
        return `
            <div class="job" style="left: 0; top: 0; position: relative;">
                <div onclick="JobWindow.open(${id}, ${x}, ${y})" class="featured ${isSilver && 'silver'}"></div>
                <div class='centermap' onclick='GameMap.center(${x}, ${y})' style="position: absolute; background-image: url('../images/map/icons/instantwork.png'); width: 20px; height: 20px; top: 0; right: 3px; cursor: pointer"></div>
                <img src="../images/jobs/${JobList.getJobById(id).shortname}.png" class="job_icon" alt='job_image'>
            </div>
        `
    }

    Vajda.checkIfJobAdded = function(id) {
        for ( const job of Vajda.addedJobs ) {
            if ( job.id === id ) {
                return true
            }
        }

        return false
    }

    Vajda.isJobSilver = function(x, y, id) {
        const key = `${x}-${y}`
        const jobData = GameMap.JobHandler.Featured[key]
        if ( jobData === undefined || jobData[id] === undefined ) {
            return false
        } else {
            return jobData[id].silver
        }
    }

    Vajda.compareUniqueJobs = function(job, jobs){
        for ( let i = 0 ; i < jobs.length; i++ ) {
            if ( jobs[i].id === job.id ) {
                if ( job.isSilver && !jobs[i].isSilver || (job.isSilver === jobs[i].isSilver && job.distance < jobs[i].distance) ) {
                    const j = jobs.at(i)
                    if ( !Vajda.isJobSilver(j.x, j.y, j.id) )
                        jobs.splice(i,1)
                    jobs.push(job)
                }
                return
            }
        }
        jobs.push(job)
    }

    Vajda.getAllUniqueJobs = function() {
        Vajda.updateJobDistances()
        let jobs = []
        for ( let i = 0 ; i < Vajda.allJobs.length; i++ ) {
            const currentJob = Vajda.allJobs.at(i)
            if ( Vajda.jobFilter.filterJob !== "" ) {
                if ( !Vajda.getJobName(currentJob.id).toLowerCase().includes(Vajda.jobFilter.filterJob)) {
                    continue
                }
            }

            if ( Vajda.checkIfJobAdded(currentJob.id) ) {
                continue
            }
            
            let isSilver = Vajda.isJobSilver(currentJob.x, currentJob.y, currentJob.id)
            currentJob.isSilver = isSilver
            currentJob.calculateDistance()
            
            if ( isSilver && Vajda.jobFilter.filterNoSilver ) {
                continue
            }

            if ( !isSilver && Vajda.jobFilter.filterOnlySilver ) {
                continue
            }

            if ( Vajda.jobFilter.filterCenterJobs && currentJob.id < 131 ) {
                continue
            }

            Vajda.compareUniqueJobs(currentJob, jobs)
        }

        Vajda.parseJobData(jobs)

        let experienceSort = function(a, b) {
            if ( a === null && b === null ) {
                return 0
            }

            if ( a === null && b !== null ) {
                return 1
            }

            if ( a !== null && b === null ) {
                return -1
            }

            let a1 = a.experience
            let b1 = b.experience
            return (a1 > b1) ? -1 :(a1 < b1) ? 1 :0
        }

        let reverseExperienceSort = function(a, b) {
            if ( a === null && b === null ) {
                    return 0
            }
            if ( a === null && b !== null ) {
                return -1
            }

            if ( a !== null && b === null ) {
                return 1
            }

            let a1 = a.experience
            let b1 = b.experience
            return (a1 > b1) ? 1 :(a1 < b1) ? -1 :0
        }

        let distanceSort = function(a, b) {
            if ( a === null && b === null ) {
                return 0
            }

            if ( a === null && b !== null ) {
                return 1
            }

            if ( a !== null && b === null ) {
                return -1
            }

            let a1 = a.distance
            let b1 = b.distance
            return (a1 > b1) ? -1 :(a1 < b1) ? 1 :0
        }

        let reverseDistanceSort = function(a, b) {
            if ( a === null && b === null ) {
                return 0
            }

            if ( a === null && b !== null ) {
                return -1
            }

            if(a !== null && b === null ) {
                return 1
            }

            let a1 = a.distance
            let b1 = b.distance
            return (a1 > b1) ? 1 :(a1 < b1) ? -1 :0
        }

        if ( Vajda.sortJobTableXp === 1 ) {
            jobs.sort(experienceSort)
        }

        if ( Vajda.sortJobTableXp === -1 ) {
            jobs.sort(reverseExperienceSort)
        }

        if ( Vajda.sortJobTableDistance === 1 ) {
            jobs.sort(distanceSort)
        }

        if ( Vajda.sortJobTableDistance === -1 ) {
            jobs.sort(reverseDistanceSort)
        }

        return jobs
    }

    Vajda.parseStopMotivation = function() {
        for ( let i = 0; i < Vajda.addedJobs.length; i++ ) {
            let stopMotivation = $(".vajda-window #x-" + Vajda.addedJobs[i].x + "y-" + Vajda.addedJobs[i].y + "id-" + Vajda.addedJobs[i].id).prop("value")
            if ( Vajda.isNumber(stopMotivation) ) {
                Vajda.addedJobs[i].setStopMotivation(parseInt(stopMotivation))
            } else {
                return false
            }
        }
        return true
    }

    Vajda.setSetForAllJobs = function() {
        for ( let i = 0; i < Vajda.addedJobs.length; i++ ) {
            Vajda.addedJobs[i].setSet(Vajda.jobSet)
        }
    }

    Vajda.consolidePosition = function(removeIndex) {
        if ( removeIndex <= Vajda.currentJob.job && Vajda.currentJob.job > 0 ) {
            Vajda.currentJob.job--
        }
        if ( Vajda.addedJobs.length === 1 ) {
            Vajda.currentJob.direction = true
        }
    }

    Vajda.createDistanceMatrix = function() {
        const matrix = new Array(Vajda.addedJobs.length)

        for ( let i = 0; i < matrix.length; i++ ) {
            matrix[i] = Vajda.addedJobs[i].calculateJobDistances()
        }

        return matrix
    }

    Vajda.countSetBits = function(n) {
        let count = 0
        while (n) {
            n &= n - 1
            count++
        }
        return count
    }

    Vajda.heldKarpSymmetric = function(distances, startJob) {
        const n = distances.length
        const memo = Array(1 << n).fill().map(() => Array(n).fill({ cost: Infinity, path: [] }))
        memo[1 << startJob][startJob] = { cost: 0, path: [startJob] }
    
        for ( let subsetSize = 2; subsetSize <= n; subsetSize++ ) {
            for ( let subset = 0; subset < (1 << n); subset++ ) {
                if ( Vajda.countSetBits(subset) === subsetSize && (subset & (1 << startJob)) ) {
                    for ( let end = 0; end < n; end++ ) {
                        if ( (subset & (1 << end)) !== 0 ) {
                            for ( let prevEnd = 0; prevEnd < n; prevEnd++ ) {
                                if ( prevEnd !== end && (subset & (1 << prevEnd)) !== 0 ) {
                                    const newCost = memo[subset ^ (1 << end)][prevEnd].cost + distances[prevEnd][end]
                                    if (newCost < memo[subset][end].cost) {
                                        memo[subset][end] = { cost: newCost, path: memo[subset ^ (1 << end)][prevEnd].path.concat([end]) }
                                    }
                                }
                            }
                        }
                    }
                }
            }
        }
    
        let minCost = Infinity
        let minPath = []
        for ( let end = 0; end < n; end++ ) {
            if ( end !== startJob && memo[(1 << n) - 1][end].cost < minCost ) {
                minCost = memo[(1 << n) - 1][end].cost
                minPath = memo[(1 << n) - 1][end].path
            }
        }

        return { cost: minCost, path: minPath }
    }


    Vajda.setEntryPoint = function(route) {
        const firstJob = route.at(0)
        const lastJob = route.at(-1)

        if ( firstJob.calculateDistance() > lastJob.calculateDistance() )
            route.reverse()

        //i could in theory make vajda start with the job nearest to current character position but cba
    }

    Vajda.getOptimalRoute = function(distanceMatrix) {
        const jobsCount = distanceMatrix.length

        if ( jobsCount === 1 ) 
            return {
                cost: 0,
                path: [0]
            }

        const routes = []
        for ( let startJob = 0; startJob < jobsCount; startJob++ ) {
            const { cost, path } = Vajda.heldKarpSymmetric(distanceMatrix, startJob)
            routes.push({ cost, path })
        }

        return routes.reduce(function(prev, curr) {
            return prev.cost < curr.cost ? prev : curr
        })
    }

    Vajda.createRoute = function() {
        Vajda.currentJob = {
            job: 0,
            direction: true
        }

        const distanceMatrix = Vajda.createDistanceMatrix()
        const optimalRoute = Vajda.getOptimalRoute(distanceMatrix)

        const addedJobsOrder = []
        for ( const index of optimalRoute.path ) {
            addedJobsOrder.push(Vajda.addedJobs.at(index))
        }

        Vajda.setEntryPoint(addedJobsOrder)

        Vajda.addedJobs = addedJobsOrder
        Vajda.selectTab("chosenJobs")
    }

    Vajda.createSetGui = function() {
        if ( Vajda.sets.length === 0 ) {
            return $(`<span style='font-size: 20px'>No sets available</span>`)
        }
        let htmlSkel = $(`
            <div id='vajda_sets_window' style='display: block; position: relative; width: 650px; height:430px'>
                <div id='vajda_sets_left' style='display: block; position: absolute; width: 250px; height: 430px; top:0px; left:0px'></div>
                <div id='vajda_sets_right' style='display: block; position: absolute; width:300px; height: 410px; top: 0px; left: 325px'></div>
            </div>`)
        let combobox = new west.gui.Combobox("combobox_sets")
        Vajda.addComboboxItems(combobox)
        combobox = combobox.select(Vajda.selectedSet)
        combobox.addListener(function(value) {
            Vajda.selectedSet = value
            Vajda.selectTab("sets")
        })
        let buttonSelectTravelSet = new west.gui.Button("Select travel set", function() {
            Vajda.travelSet = Vajda.selectedSet
            Vajda.selectTab("sets")
        })
        let buttonSelectJobSet = new west.gui.Button("Select job set", function() {
            Vajda.jobSet = Vajda.selectedSet
            Vajda.setSetForAllJobs()
            Vajda.selectTab("sets")
        })
        let buttonSelectHealthSet = new west.gui.Button("Select health set", function() {
            Vajda.healthSet = Vajda.selectedSet
            Vajda.selectTab("sets")
        })
        let buttonSelectRegenSet = new west.gui.Button("Select regeneration set", function() {
            Vajda.regenerationSet = Vajda.selectedSet
            Vajda.selectTab("sets")
        })
        let travelSetText = "None"

        if ( Vajda.travelSet != -1 ) {
            travelSetText = Vajda.sets[Vajda.travelSet].name
        }

        let jobSetText = "None"
        if ( Vajda.jobSet != -1 ) {
            jobSetText = Vajda.sets[Vajda.jobSet].name
        }

        let healthSetText = "None"
        if ( Vajda.healthSet != -1 ) {
            healthSetText = Vajda.sets[Vajda.healthSet].name
        }
        
        let regenSetText = "None"
        if ( Vajda.regenerationSet != -1 ) {
            regenSetText = Vajda.sets[Vajda.regenerationSet].name
        }

        const left = $("<div></div>")
            .append(
                new west.gui.Groupframe()
                .appendToContentPane($("<span>Sets</span><br><br>"))
                .appendToContentPane(combobox.getMainDiv())
                .appendToContentPane($("<br><br><span>Travel set:"+ travelSetText +"</span><br><br>"))
                .appendToContentPane(buttonSelectTravelSet.getMainDiv())
                .appendToContentPane($("<br><br><span>Job set:"+ jobSetText +"</span><br><br>"))
                .appendToContentPane(buttonSelectJobSet.getMainDiv())
                .appendToContentPane($("<br><br><span>Health set:"+ healthSetText +"</span><br><br>"))
                .appendToContentPane(buttonSelectHealthSet.getMainDiv())
                .getMainDiv()
            )
         const right = $("<div style=\'display:block; position:relative; width:300px; height:410px'\></div>")
        //head div
        right.append("<div class=\'wear_head wear_slot'\ style=\'display:block; position:absolute; left:30px; top:1px; width:93px; height:94px; background:url(https://westzz.innogamescdn.com/images/window/wear/bg_sprite.png) 0 0 no-repeat; background-position: -95px 0'\></div>")
        //chest div
        right.append("<div class=\'wear_body wear_slot'\ style=\'display:block; position:absolute; left:30px; top:106px; width:95px; height:138px; background:url(https://westzz.innogamescdn.com/images/window/wear/bg_sprite.png) 0 0 no-repeat; background-position:0 0'\></div>")
        //pants div
        right.append("<div class=\'wear_pants wear_slot'\ style=\'display:block; position:absolute; left:30px; top:258px; width:93px; height:138px; background:url(https://westzz.innogamescdn.com/images/window/wear/bg_sprite.png) 0 0 no-repeat; background-position:0 0'\></div>")
        //neck div
        right.append("<div class=\'wear_neck wear_slot'\ style=\'display:block; position:absolute; left:-47px; top:1px; width:74px; height:74px; background:url(https://westzz.innogamescdn.com/images/window/wear/bg_sprite.png) 0 0 no-repeat; background-position:-189px 0'\></div>")
        //right arm div
        right.append("<div class=\'wear_right_arm wear_slot'\ style=\'display:block; position:absolute; left:-64px; top:79px; width:95px; height:138px; background:url(https://westzz.innogamescdn.com/images/window/wear/bg_sprite.png) 0 0 no-repeat; background-position:0 0'\></div>")
        //animal div
        right.append("<div class=\'wear_animal wear_slot'\ style=\'display:block; position:absolute; left:-64px; top:223px; width:93px; height:94px; background:url(https://westzz.innogamescdn.com/images/window/wear/bg_sprite.png) 0 0 no-repeat; background-position:-95px 0'\></div>")
        //yield div
        right.append("<div class=\'wear_yield wear_slot'\ style=\'display:block; position:absolute; left:-47px; top:321px; width:74px; height:74px; background:url(https://westzz.innogamescdn.com/images/window/wear/bg_sprite.png) 0 0 no-repeat; background-position:-189px 0'\></div>")
        //left arm div
        right.append("<div class=\'wear_left_arm wear_slot'\ style=\'display:block; position:absolute; left:127px; top:52px; width:95px; height:138px; background:url(https://westzz.innogamescdn.com/images/window/wear/bg_sprite.png) 0 0 no-repeat; background-position:0 0'\></div>")
        //belt div
        right.append("<div class=\'wear_belt wear_slot'\ style=\'display:block; position:absolute; left:127px; top:200px; width:93px; height:94px; background:url(https://westzz.innogamescdn.com/images/window/wear/bg_sprite.png) 0 0 no-repeat; background-position:-95px 0'\></div>")
        //boots div
        right.append("<div class=\'wear_foot wear_slot'\ style=\'display:block; position:absolute; left:127px; top:302px; width:93px; height:94px; background:url(https://westzz.innogamescdn.com/images/window/wear/bg_sprite.png) 0 0 no-repeat; background-position:-95px 0'\></div>")
        let keys = ["head","body","pants","neck","right_arm","animal","yield","left_arm","belt","foot"]
        if ( Vajda.selectedSet !== -1 )
            Vajda.insertSetImages(right,keys)
        $("#vajda_sets_left",htmlSkel).append(left)
        $("#vajda_sets_right",htmlSkel).append(right)
        return htmlSkel
    }

    Vajda.getImageSkel = function() {
        return $("<img src=\''\>")
    }

    Vajda.insertSetImages = function(html,keys) {
        for ( let i = 0; i < keys.length; i++ ) {
            if ( Vajda.sets[Vajda.selectedSet][keys[i]] !== null ) {
                $(".wear_"+keys[i], html).append(Vajda.getImageSkel().attr("src", Vajda.getItemImage(Vajda.sets[Vajda.selectedSet][keys[i]])))
            }
        }
        return html
    }

    Vajda.addEventsHeader = function() {
        $(".vajda-window .jobXp").click(function() {
            if ( Vajda.sortJobTableXp === 0 ) {
                Vajda.sortJobTableXp = 1
            } else {
                ( Vajda.sortJobTableXp === 1 ) ? Vajda.sortJobTableXp = -1 : Vajda.sortJobTableXp = 1
            }
            Vajda.sortJobTableDistance = 0
            Vajda.selectTab("jobs")
        })
        $(".vajda-window .jobDistance").click(function() {
            if ( Vajda.sortJobTableDistance === 0 ) {
                Vajda.sortJobTableDistance = 1
            } else {
                ( Vajda.sortJobTableDistance === 1 ) ? Vajda.sortJobTableDistance = -1 : Vajda.sortJobTableDistance = 1
            }
            Vajda.sortJobTableXp = 0
            Vajda.selectTab("jobs")
        })
    }

    Vajda.isInHomeTown = function() {
        const homeTown = Character.homeTown
        return GameMap.calcWayTime(Character.position,{x: homeTown.x, y: homeTown.y}) == 0
    }

    Vajda.addDeposit = async function(townId) {
        const amount = Character.money
        await Ajax.remoteCall("building_bank", "deposit", {
            town_id: townId,
            amount: amount
        }, function(data) {
            if (data.error == false) {
                BankWindow.Balance.Mupdate(data)
                Character.setDeposit(data.deposit)
                Character.setMoney(data.own_money)
                //new UserMessage(s(sextext('Vložil si $ %1 a zaplatil si poplatok $ %2', 'Vložila si $ %1 a zaplatila si poplatok $ %2', Character.charSex), amount, data.fee),UserMessage.TYPE_SUCCESS).show()
            } else
                new UserMessage(data.msg,UserMessage.TYPE_ERROR).show()
        }, BankWindow)
    }
    

    Vajda.goDepositMoney = async function() {
        const townId = Character.homeTown.town_id
        if ( !townId ) return

        await Vajda.equipSet(Vajda.travelSet)
        TaskQueue.add(new TaskWalk(townId, 'town'))

        while(true) {
            if ( Vajda.isInHomeTown() ) {
                break
            }

            if( !Vajda.isRunning ) {
                break
            }

            await sleep(1000)
        }

        await Vajda.addDeposit(townId)
        $('.tw2gui_dialog_framefix').remove()
    }

    Vajda.updateJobsMotivationOnRefill = function(val) {
        return Vajda.addedJobs.map(job => job.setMotivation(p => p + val).motivation - job.stopMotivation)
    }


    Vajda.checkMotivation = async function(index, result, callback) {
        let check = function(index, result) {
            Vajda.loadJobMotivation(index, function(motivation) {
                Vajda.addedJobs.at(index).setMotivation(motivation)
                result.push(motivation)
                if ( index + 1 < Vajda.addedJobs.length ) {
                    check(++index, result)
                } else if( index + 1 === Vajda.addedJobs.length ) {
                    callback(result)
                    return
                }
            })
        }
        check(index, result)
    }

    Vajda.isMotivationAbove = function(result) {
        for ( let i = 0; i < result.length; i++ ) {
            if ( result.at(i) > Vajda.addedJobs.at(i).stopMotivation ) {
                return true
            }
        }
        return false
    }

    Vajda.isStopMotivationZero = function() {
        for ( let i = 0; i < Vajda.addedJobs.length; i++ ) {
            if( Vajda.addedJobs[i].stopMotivation === 0 ) {
                return true
            }
        }
        return false
    }

    Vajda.isHealthBelowLimit = function() {
        if ( Vajda.settings.healthStop >= ((Character.health/Character.maxHealth) * 100) ) {
            return true
        }
        return false
    }

    Vajda.changeJob = function() {
        Vajda.currentJob.direction ? Vajda.currentJob.job++ : Vajda.currentJob.job--;
        if ( Vajda.currentJob.job === Vajda.addedJobs.length ) {
            Vajda.currentJob.job--
            Vajda.currentJob.direction = false
        } else if ( Vajda.currentJob.job < 0 ) {
            Vajda.currentJob.job++
            Vajda.currentJob.direction = true
        }
        Vajda.setCookies()
        Vajda.run()
    }

    Vajda.searchBest = function(skills, jobId, onlyWearable = true) {
        if (!Bag.loaded) {
            EventHandler.listen('inventory_loaded', function() {
                Vajda.searchBest(skills, jobId, onlyWearable)
                return EventHandler.ONE_TIME_EVENT
            })
            return
        }
        let set = west.item.Calculator.getBestSet(skills, jobId), items = set && set.getItems() || [], invItems = Bag.getItemsByItemIds(items), result = [], i, invItem, wearItem
        for (i = 0; i < invItems.length; i++) {
            invItem = invItems[i]
            wearItem = Wear.get(invItem.getType())
            if (!wearItem || (wearItem && (wearItem.getItemBaseId() !== invItem.getItemBaseId() || wearItem.getItemLevel() < invItem.getItemLevel()))) {
                result.push(invItem)
            }
        }
        return result.map(item => item.obj.getId())
    }

    Vajda.getBestGear = function(jobid) {
        let modelId = function(jobid) {
            for ( let i = 0; i < JobsModel.Jobs.length; i++ ) {
                if ( JobsModel.Jobs[i].id === jobid )
                    return i
            }
            return -1
        }
        const result = west.item.Calculator.getBestSet(JobsModel.Jobs[modelId(jobid)].get('skills'), jobid)
        const bestItems = result && result.getItems()
        return bestItems
    }

    Vajda.isWearing = function(itemId) {
        if ( Wear.wear[ItemManager.get(itemId).type] === undefined) return false
        return Wear.wear[ItemManager.get(itemId).type].obj.item_id == itemId
    }

    Vajda.isGearEquiped = async function(items) {
        for ( const itemId of items ) {
            if ( !Vajda.isWearing(itemId) ) {
                return false
            }
        }

        return true
    }

    Vajda.equipBestGear = async function(job) {
        const bestGear = job.getBestEquipment()

        for ( const itemId of bestGear ) {
            Wear.carry(Bag.getItemByItemId(itemId))
        }

        while (true) {
            const isFinished = await Vajda.isGearEquiped(bestGear)
            if ( isFinished ) break
            await sleep(50)
        }
        return Promise.resolve(true)
    }

    Vajda.getSetItemArray = function(set) {
        var items = []
        if ( set.head !== null )
            items.push(set.head)
        if ( set.neck !== null )
            items.push(set.neck)
        if ( set.body !== null )
            items.push(set.body)
        if ( set.right_arm !== null ) 
            items.push(set.right_arm)
        if ( set.left_arm !== null )
            items.push(set.left_arm)
        if ( set.belt !== null )
            items.push(set.belt)
        if ( set.foot !== null )
            items.push(set.foot)
        if ( set.animal !== null ) 
            items.push(set.animal)
        if ( set.yield !== null ) 
            items.push(set.yield)
        if ( set.pants !== null ) 
            items.push(set.pants)
        return items
    }

    Vajda.equipSet = async function(set) {
        if ( set === -1 ) return true
        EquipManager.switchEquip(Vajda.sets[set].equip_manager_id)
        while ( true ) {
            let isFinished = await Vajda.isGearEquiped(Vajda.getSetItemArray(Vajda.sets[set]))
            if ( isFinished ) break
            await sleep(50)
        }
        return Promise.resolve(true)
    }

    Vajda.cancelJobs = function() {
        if ( TaskQueue.queue.length > 0 )
            TaskQueue.cancelAll()
    }


    //https://prnt.sc/KAgbLNqB4zK6
    Vajda.runJob = async function(jobIndex, jobCount) {
        Vajda.statistics.sessionJobsCount += jobCount
        Vajda.statistics.totalJobsCount += jobCount
        const oldXp = Character.experience
        const oldMoney = Character.money
        const job = Vajda.addedJobs.at(jobIndex)
        await Vajda.equipBestGear(job)
        for ( let i = 0; i < jobCount; i++ ) {
            JobWindow.startJob(job.id, job.x, job.y, 15)
        }
        Observer.start()
        await sleep(Vajda.settings.setWearDelay * 1000)
        Vajda.equipSet(job.set)
        Manager.useBuff('character')
        while (true) {
            if ( TaskQueue.queue.length === 0 ) {
                Vajda.updateStatistics(oldXp, oldMoney)
                Vajda.setCookies()
                Vajda.prepareJobRun(jobIndex)
                return
            }
            if ( !Vajda.isRunning || Vajda.isHealthBelowLimit() ) {
                break
            }
            await sleep(1000)
        }
        Vajda.statistics.sessionJobsCount -= TaskQueue.queue.length
        Vajda.statistics.totalJobsCount -= TaskQueue.queue.length
        Vajda.updateStatistics(oldXp, oldMoney)
        Vajda.setCookies()
        Vajda.cancelJobs()

        if ( Vajda.isRunning && Vajda.isHealthBelowLimit() ) {
            await sleep(2000)
            Vajda.run()
        
        }
    }

    Vajda.walkToJob = async function(index) {
        const job = Vajda.addedJobs.at(index)
        const jobGroup = JobList.getJobsByGroupId(job.groupId)

        const jobToWalkTo = jobGroup.map(j => 
            j.id
        ).map( id => {
            const j = JobsModel.getById(id)

            return {
                id: id,
                workpoints: j.workpoints,   //required LP
                jobpoints: j.jobpoints      //character has LP
            }
        }).reduce((prev, curr) => {
            return prev.workpoints < curr.workpoints ? prev : curr
        })

        await Manager.useBuff('travel')

        if ( Vajda.isAllowedToDepositMoney() ) {
            Observer.start()
            await Vajda.goDepositMoney()
        }

        JobWindow.startJob(jobToWalkTo.id, job.x, job.y, 15)
        Observer.start()

        while (true) {
            if ( GameMap.calcWayTime(Character.position, {x: job.x, y: job.y} ) === 0 ) {
                break
            }

            if ( !Vajda.isRunning ) {
                break
            }
            await sleep(1000)
        }
        Vajda.cancelJobs()
        if ( Vajda.isRunning )
            Vajda.prepareJobRun(index)
    }

    Vajda.prepareJobRun = async function(index) {
        const job = Vajda.addedJobs.at(index)
        setTimeout(function() {
            Vajda.loadJobMotivation(index, async function(motivation) {
                if ( Character.energy === 0 || Vajda.isHealthBelowLimit() ) {
                    Vajda.run()
                }   else if ( motivation <= job.stopMotivation && job.stopMotivation > 0 ) {
                    Vajda.checkMotivation(0, [], function(result) {
                        if ( Vajda.isMotivationAbove(result) ) {
                            Vajda.changeJob()
                        } else {
                            Vajda.run()
                        }
                    })
                } else if ( GameMap.calcWayTime(Character.position,{x: job.x, y: job.y}) === 0 ) {
                    let maxJobs = Premium.hasBonus('automation') ? 9 : 4
                    let numberOfJobs
                    if (job.stopMotivation !== 0 ) {
                        numberOfJobs = Math.min(Math.min(motivation - job.stopMotivation, Character.energy), maxJobs)
                    } else {
                        numberOfJobs = Math.min(Character.energy, maxJobs)
                    }
                    Vajda.runJob(index, Math.floor(numberOfJobs))
                    if ( Manager.isOptimized )
                        Manager.checkSchedule(numberOfJobs)
                } else {
                    await Vajda.equipSet(Vajda.travelSet)
                    Vajda.walkToJob(index)
                }
            })
        }, Vajda.RNG(Vajda.settings.jobDelayMin, Vajda.settings.jobDelayMax) * 1000)
    }

    Vajda.canAddMissing = function(result) {
        if ( !Vajda.settings.addMotivation && Vajda.jobsBelowMotivation(result) && !Vajda.isStopMotivationZero() ) {
            alert("Can't continue because of motivation")
            return false
        }

        if ( !Vajda.settings.addEnergy && Character.energy === 0 ) {
            alert("Can't continue because of energy")
            return false
        }

        if ( !Vajda.settings.addHealth && Vajda.isHealthBelowLimit() ) {
            alert("Can't continue because of health")
            return false
        }
        return true
    }

    Vajda.finishRun = function() {
        Vajda.currentState = 0
        Vajda.isRunning = false
        Vajda.selectTab("chosenJobs")
        alert("Finished")
    }

    Vajda.jobsBelowMotivation = function(result) {
        let count = 0
        for ( let i = 0; i < result.length; i++ ) {
            if ( result[i] <= Vajda.addedJobs[i].stopMotivation ) {
                count++
            }
        }
        return count
    }



    Vajda.averageMissingMotivation = function(result) {
        let motivation = 0
        for ( let i = 0; i < result.length; i++ ) {
            motivation += 100-result[i]
        }
        return motivation/result.length
    }


    Vajda.fillUp = async function(result) {
        const energyMissing = 100 - (Character.energy/Character.maxEnergy) * 100
        const motivationMissing = Vajda.jobsBelowMotivation(result)
        const averageMotivationMissing = Vajda.averageMissingMotivation(result)

        const consumableToUse = Manager.findProperConsumable(motivationMissing, energyMissing, averageMotivationMissing)
        

        if ( consumableToUse === null ) return false

        await Manager.useConsumableOrWaitForCooldown(consumableToUse, true)

        if ( Manager.isOptimized && consumableToUse.isCakeDecoration() ) {
            const updatedMotivation = Vajda.updateJobsMotivationOnRefill(consumableToUse.motivation)
            await Manager.createSchedule(updatedMotivation)
        }

        return true
    }

    Vajda.updateStatistics = function(oldXp, oldMoney) {
        const xpDiff = Character.experience - oldXp
        const moneyDiff = Character.money - oldMoney
        
        Vajda.statistics.sessionXpCount += xpDiff
        Vajda.statistics.totalXpCount += xpDiff

        if ( moneyDiff > 0 ) {      //spending money while vajda is running would make this a bit funky
            Vajda.statistics.sessionMoneyCount += moneyDiff
            Vajda.statistics.totalMoneyCount += moneyDiff
        }
    }

    Vajda.run = async function() {
        Vajda.checkMotivation(0, [], async function(result) {
            if ( ( Vajda.isMotivationAbove(result) || Vajda.isStopMotivationZero()) && Character.energy > 0 && !Vajda.isHealthBelowLimit() ) {
                Vajda.currentState = 1
                Vajda.selectTab("chosenJobs")
                Vajda.prepareJobRun(Vajda.currentJob.job)
            } else {
                if ( !Vajda.canAddMissing(result) ) {
                    Vajda.finishRun()
                } else {
                    let answer = await Vajda.fillUp(result)
                    if ( !answer ) {
                        Vajda.finishRun()
                    }
                }
            }
        })
    }

    Vajda.formatNumber = function(number) {
        if ( typeof number === 'number' ) {
            number = String(number)
        }

        const numberString = number.replace(/[ ,]/g, '')
    
        let formattedNumber = ''
        for ( let i = 0; i < numberString. length; i++ ) {
            if ( i > 0 && i % 3 === 0 ) {
                formattedNumber = ' ' + formattedNumber
            }

            formattedNumber = numberString[numberString.length - 1 - i] + formattedNumber
        }
        
        return formattedNumber
    }

    Vajda.createConsumablesTable = function() {
        let htmlSkel = $(`<div id='consumables_overview'></div>`)
        let html = $(`
            <div class ='consumables_filter' style='position: relative'>
                <div id='energy_consumables' style='position: absolute; top: 10px; left: 15px'></div> 
                <div id='motivation_consumables' style='position: absolute; top: 10px; left: 160px'></div> 
                <div id='health_consumables' style='position: absolute; top: 10px; left: 320px'></div> 
                <div id='buff_consumables' style='position: absolute; top: 10px; left: 460px'></div>
            </div>`
        )

        let table = new west.gui.Table()
        let consumableList = Vajda.filterConsumables(Vajda.consumableSelection.energy, Vajda.consumableSelection.motivation, Vajda.consumableSelection.health, Vajda.consumableSelection.hideBuffs)
        table.addColumn("consumIcon","consumIcon").addColumn("consumCount","consumCount").addColumn("consumEnergy","consumEnergy").addColumn("consumMotivation","consumMotivation").addColumn("consumHealth","consumHealth").addColumn("consumBuffs", "consumBuffs").addColumn("consumSelected","consumSelected")
        table.appendToCell("head","consumIcon","Image").appendToCell("head","consumCount","Count").appendToCell("head","consumEnergy","Energy").appendToCell("head","consumMotivation","Motivation").appendToCell("head","consumHealth","Health").appendToCell("head", "consumBuffs", "Buffs").appendToCell("head","consumSelected","Use")
        for ( const consumable of consumableList ) {
            const checkbox = new west.gui.Checkbox()
            checkbox.setSelected(consumable.isSelected)
            checkbox.setId(consumable.id)
            checkbox.setCallback(function() {
                Vajda.consumableTablePosition.content = $(".vajda-window .tw2gui_scrollpane_clipper_contentpane").css("top")
                Vajda.consumableTablePosition.scrollbar = $(".vajda-window .tw2gui_scrollbar_pulley").css("top")
                if ( consumable instanceof Buff && !consumable.hasCooldown() ) {
                    Manager.selectedBuffs = consumable
                    Vajda.selectTab('consumables')
                    Vajda.setCookies()
                    return
                }
                Vajda.changeConsumableSelection(parseInt(this.divMain.attr("id")), this.isSelected())
                Vajda.selectTab("consumables")
                Vajda.setCookies()
            })
            table.appendRow().appendToCell(-1,"consumIcon", Vajda.getConsumableIcon(consumable.image)).appendToCell(-1,"consumCount",consumable.count).appendToCell(-1,"consumEnergy",consumable.energy).appendToCell(-1,"consumMotivation",consumable.motivation).appendToCell(-1,"consumHealth",consumable.health).appendToCell(-1, "consumBuffs", consumable.getBuffHTML()).appendToCell(-1,"consumSelected",checkbox.getMainDiv())
        }
        const buttonSelect = new west.gui.Button("Select all", function() {
            Vajda.changeSelectionAllConsumables(true)
            Vajda.selectTab("consumables")
            Vajda.setCookies()
        })
        const buttonDeselect = new west.gui.Button("Deselect all", function() {
            Vajda.changeSelectionAllConsumables(false)
            Vajda.selectTab("consumables")
            Vajda.setCookies()
        })
        table.appendToFooter("consumEnergy",buttonSelect.getMainDiv())
        table.appendToFooter("consumHealth",buttonDeselect.getMainDiv())
        htmlSkel.append(table.getMainDiv())
        const checkboxEnergyConsumes = new west.gui.Checkbox()
        checkboxEnergyConsumes.setLabel("Energy consumables")
        checkboxEnergyConsumes.setSelected(Vajda.consumableSelection.energy)
        checkboxEnergyConsumes.setCallback(function() {
            Vajda.consumableSelection.energy = this.isSelected()
            Vajda.selectTab("consumables")
        })
        const checkboxMotivationConsumes = new west.gui.Checkbox()
        checkboxMotivationConsumes.setLabel("Motivation consumables")
        checkboxMotivationConsumes.setSelected(this.consumableSelection.motivation)
        checkboxMotivationConsumes.setCallback(function() {
            Vajda.consumableSelection.motivation = this.isSelected()
            Vajda.selectTab("consumables")
        })
        const checkboxHealthConsumes = new west.gui.Checkbox()
        checkboxHealthConsumes.setLabel("Health consumables")
        checkboxHealthConsumes.setSelected(this.consumableSelection.health)
        checkboxHealthConsumes.setCallback(function() {
            Vajda.consumableSelection.health = this.isSelected()
            Vajda.selectTab("consumables")
        })
        const buffsFilter = new west.gui.Checkbox()
        buffsFilter.setLabel("Hide buffs")
        buffsFilter.setSelected(this.consumableSelection.hideBuffs)
        buffsFilter.setCallback(function() {
            Vajda.consumableSelection.hideBuffs = this.isSelected()
            Vajda.selectTab("consumables")
        })
        $("#energy_consumables", html).append(checkboxEnergyConsumes.getMainDiv())
        $("#motivation_consumables", html).append(checkboxMotivationConsumes.getMainDiv())
        $("#health_consumables", html).append(checkboxHealthConsumes.getMainDiv())
        $("#buff_consumables", html).append(buffsFilter.getMainDiv())
        htmlSkel.append(html)
        return htmlSkel
    }

    Vajda.createAddedJobsTab = function() {
        const htmlSkel = $(`<div id='added_jobs_overview'></div>`)
        const footerHtml = $(`
            <div id='start_vajda' style='position: relative'>
                <span id='vajda-state-info' class='vajda_state' style='position: absolute; left: 20px; top: 10px; font-family: Arial, Helvetica, sans-serif; font-size: 15px; font-weight: bold;'> 
                    Current state: ${Vajda.states[Vajda.currentState]}
                </span>
                <div class='vajda_run' style='position: absolute; left: 350px; top: 20px'></div>
            </div>
        `)
        const table = new west.gui.Table()
        table.addColumn("jobIcon","jobIcon").addColumn("jobName","jobName").addColumn("jobStopMotivation","jobStopMotivation").addColumn("jobSet","jobSet").addColumn("jobRemove","jobRemove")
        table.appendToCell("head","jobIcon","Job icon").appendToCell("head","jobName","Job name").appendToCell("head","jobStopMotivation","Stop motivation").appendToCell("head","jobSet","Job set").appendToCell("head","jobRemove","")
        for ( let job = 0; job < Vajda.addedJobs.length; job++ ) {
            table.appendRow().appendToCell(-1,"jobIcon", Vajda.getJobIcon(Vajda.addedJobs[job].isSilver, Vajda.addedJobs[job].id, Vajda.addedJobs[job].x, Vajda.addedJobs[job].y)).appendToCell(-1,"jobName", Vajda.getJobName(Vajda.addedJobs[job].id)).appendToCell(-1,"jobStopMotivation", Vajda.createMinMotivationTextfield(Vajda.addedJobs[job].x, Vajda.addedJobs[job].y, Vajda.addedJobs[job].id, Vajda.addedJobs[job].stopMotivation)).appendToCell(-1,"jobSet", Vajda.createComboxJobSets(Vajda.addedJobs[job].x, Vajda.addedJobs[job].y, Vajda.addedJobs[job].id)).appendToCell(-1,"jobRemove", Vajda.createRemoveJobButton(Vajda.addedJobs[job].x, Vajda.addedJobs[job].y, Vajda.addedJobs[job].id))
        }
        const buttonStart = new west.gui.Button("Start", async function() {
            const parseSuccesful = Vajda.parseStopMotivation()
            if ( parseSuccesful ) {
                Vajda.currentState = 3
                $(`#vajda-state-info`).text(`Current state: ${Vajda.states[3]}`)
                Vajda.createRoute()
                Vajda.isRunning = true
                Vajda.setCookies()
                if ( Manager.isOptimized && Manager.hasEnoughPlugsAndDecorations() ) {
                    await Manager.createSchedule()
                }
                Observer.start()
                Vajda.run()
            } else {
                new UserMessage("Wrong format of set stop motivation", UserMessage.TYPE_ERROR).show()
            }
        });
        const buttonStop = new west.gui.Button("Stop",function() {
            Vajda.isRunning = false
            Vajda.currentState = 0
            Vajda.selectTab("chosenJobs")
            Observer.stop()
        })
        htmlSkel.append(table.getMainDiv())
        $(".vajda_run",footerHtml).append(buttonStart.getMainDiv())
        $(".vajda_run",footerHtml).append(buttonStop.getMainDiv())
        htmlSkel.append(footerHtml)
        return htmlSkel
    }


    Vajda.createStatisticsGui = function() {
        const offsetLeft = '.5rem'

        const statsBubbleStyle = `'
            padding: 1rem 2rem;
            border-radius: 10px;
            background-color: rgba(255, 255, 228, .3);
            position: relative;
            box-shadow: 0 0 5px rgba(0, 0, 0, .2)
        '`

        const pseudoHeadingStyle = `'
            display: block;
            width: calc(100% - ${offsetLeft});
            padding-left: ${offsetLeft};
            border-bottom: 1px solid black;
            margin-bottom: .5rem
        '`

        const refreshStats = `
            <div style=${statsBubbleStyle}>
                <b style=${pseudoHeadingStyle}>REFRESH COUNT</b>
                <p style='padding-left:${offsetLeft}'>Total in this session: ${Observer.refreshCount}</p>
            </div>
        `

        return $(`
            <div id='statistics_overview' style='padding: 0 2rem; display: grid; gap: 1rem'>
                <div style=${statsBubbleStyle}>
                    <b style=${pseudoHeadingStyle}>EXPERIENCE</b>
                    <p style='padding-left: ${offsetLeft}'>Total in this session: ${Vajda.formatNumber(Vajda.statistics.sessionXpCount)}</p>
                    <p style='padding-left: ${offsetLeft}'>Total overall: ${Vajda.formatNumber(Vajda.statistics.totalXpCount)}</p>
                </div>

                <div style=${statsBubbleStyle}>
                    <b style=${pseudoHeadingStyle}>MONEY</b>
                    <p style='padding-left: ${offsetLeft}'>Total in this session: ${Vajda.formatNumber(Vajda.statistics.sessionMoneyCount)}</p>
                    <p style='padding-left: ${offsetLeft}'>Total overall: ${Vajda.formatNumber(Vajda.statistics.totalMoneyCount)}</p>
                </div>

                <div style=${statsBubbleStyle}>
                    <b style=${pseudoHeadingStyle}>JOBS</b>
                    <p style='padding-left: ${offsetLeft}'>Total in this session: ${Vajda.formatNumber(Vajda.statistics.sessionJobsCount)}</p>
                    <p style='padding-left: ${offsetLeft}'>Total overall: ${Vajda.formatNumber(Vajda.statistics.totalJobsCount)}</p>
                </div>

                ${Observer.isEnabled ? refreshStats : ''}
            </div>
        `)
    }

    Vajda.createSettingsGui = function() {
        const htmlSkel = $(`<div id='settings_overview' style='padding: 10px'></div>`)
        const checkboxAddEnergy = new west.gui.Checkbox()
        checkboxAddEnergy.setLabel("Add energy")
        checkboxAddEnergy.setSelected(Vajda.settings.addEnergy)
        checkboxAddEnergy.setCallback(function() {
            Vajda.settings.addEnergy = !Vajda.settings.addEnergy
        })
        const checkboxAddMotivation = new west.gui.Checkbox()
        checkboxAddMotivation.setLabel("Add motivation")
        checkboxAddMotivation.setSelected(Vajda.settings.addMotivation)
        checkboxAddMotivation.setCallback(function() {
            Vajda.settings.addMotivation = !Vajda.settings.addMotivation
        })
        const checkboxAddHealth = new west.gui.Checkbox()
        checkboxAddHealth.setLabel("Add health")
        checkboxAddHealth.setSelected(Vajda.settings.addHealth)
        checkboxAddHealth.setCallback(function() {
            Vajda.settings.addHealth = !Vajda.settings.addHealth
        })

        const buttPlugsCheckbox = new west.gui.Checkbox()
        buttPlugsCheckbox.setLabel('Optimize for butt plugs and cake decorations')
        buttPlugsCheckbox.setSelected(Manager.isOptimized)
        buttPlugsCheckbox.setCallback(() => {
            Manager.isOptimized = p => !p
        })

        const enableObserverCheckbox = new west.gui.Checkbox()
        enableObserverCheckbox.setLabel('[Experimental] Enable auto refresh → Read user manual')
        enableObserverCheckbox.setSelected(Observer.isEnabled)
        enableObserverCheckbox.setCallback(() => {
            Observer.isEnabled = previous => {
                $('#observer-delay-input').css('max-height', previous ? '0px' : '30px')
                   
                Vajda.isRunning && !previous && Observer.start(forceStart = true)
                Vajda.isRunning && previous && Observer.stop()

                return !previous
            }
        })

        const observerDelayInput = new west.gui.Textfield()
        observerDelayInput.setWidth(100)
        observerDelayInput.setValue(Observer.getTimeOut(true))
        observerDelayInput.getMainDiv()[0].querySelector('input').addEventListener('change', e => {
            const val = e.target.value

            if ( Vajda.isNumber(val) ) {
                Observer.timeOut = val
            }
        })

        const style = (isOpen) => `'
            overflow: hidden;
            max-height: ${isOpen ? '30' : '0'}px;
            transition: max-height .5s;
            margin-bottom: 1rem;
        '`
        const observerDelay = $(`
            <div style=${style(Observer.isEnabled)} id='observer-delay-input'>
                Refresh page after 
            </div>
        `)
        observerDelay.append(observerDelayInput.getMainDiv())
        observerDelay.append(` minutes since the last action`)

        const saveMoneyCheckbox = new west.gui.Checkbox()
        saveMoneyCheckbox.setLabel('Deposit money in your hometown')
        saveMoneyCheckbox.setSelected(Vajda.settings.addDeposit.isEnabled)
        saveMoneyCheckbox.setCallback(() => {
            if ( Character.homeTown.town_id === 0 ) {
                new UserMessage("You don' have a home town", UserMessage.TYPE_HINT).show()
                return
            }

            const oldVal = !!Vajda.settings.addDeposit.isEnabled
            Vajda.settings.addDeposit.isEnabled = !oldVal

            $(`#deposit-limit-input`).css('max-height', oldVal ? '0px' : '30px')
        })

        const depositLimit = $(`
            <div style=${style(Vajda.settings.addDeposit.isEnabled)} id='deposit-limit-input'>
                Deposit when more than 
            </div>
        `)

        const limitInput = new west.gui.Textfield()
        limitInput.setWidth(100)
        limitInput.setValue(Number.isNaN(Vajda.settings.addDeposit.limit) ? '' : Vajda.settings.addDeposit.limit)
        limitInput.getMainDiv()[0].querySelector('input').addEventListener('change', e => {
            const val = e.target.value

            if ( Vajda.isNumber(val) ) {
                Vajda.settings.addDeposit.limit = val
            }
        })

        depositLimit.append(limitInput.getMainDiv())
        depositLimit.append(` $ in cash`)


  
        const htmlHealthStop = $("<div></div>")
        htmlHealthStop.append(`<span>Stoppage health percent value</span>`)
        const healthStopTextfiled = new west.gui.Textfield("healthStop")
        healthStopTextfiled.setValue(Vajda.settings.healthStop)
        healthStopTextfiled.setWidth(100)
        htmlHealthStop.append(healthStopTextfiled.getMainDiv())
        const htmlSetWearDelay = $("<div></div>")
        htmlSetWearDelay.append("<span> Job set equip delay </span>")
        const setWearDelayTextfiled = new west.gui.Textfield("setWearDelay")
        setWearDelayTextfiled.setValue(Vajda.settings.setWearDelay)
        setWearDelayTextfiled.setWidth(100)
        htmlSetWearDelay.append(setWearDelayTextfiled.getMainDiv())

        const htmlJobDelay = $("<div></div>")
        htmlJobDelay.append(`<span>Random delay between jobs(seconds)</span>`)
        const jobDelayTextFieldMin = new west.gui.Textfield("jobDelay")
        jobDelayTextFieldMin.setValue(Vajda.settings.jobDelayMin)
        jobDelayTextFieldMin.setWidth(50)
        const jobDelayTextFieldMax = new west.gui.Textfield("jobDelay")
        jobDelayTextFieldMax.setValue(Vajda.settings.jobDelayMax)
        jobDelayTextFieldMax.setWidth(50)

        htmlJobDelay.append(jobDelayTextFieldMin.getMainDiv())
        htmlJobDelay.append("<span> - </span>")
        htmlJobDelay.append(jobDelayTextFieldMax.getMainDiv())


        const manualLink = `
            <div onclick=Vajda.selectTab('manual') style="cursor: pointer; text-decoration: underline; position: absolute; top: 20px; right: 25px;">
                Open User Manual <svg style="width: 10px" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512"><!--!Font Awesome Free 6.5.1 by @fontawesome - https://fontawesome.com License - https://fontawesome.com/license/free Copyright 2024 Fonticons, Inc.--><path d="M352 0c-12.9 0-24.6 7.8-29.6 19.8s-2.2 25.7 6.9 34.9L370.7 96 201.4 265.4c-12.5 12.5-12.5 32.8 0 45.3s32.8 12.5 45.3 0L416 141.3l41.4 41.4c9.2 9.2 22.9 11.9 34.9 6.9s19.8-16.6 19.8-29.6V32c0-17.7-14.3-32-32-32H352zM80 32C35.8 32 0 67.8 0 112V432c0 44.2 35.8 80 80 80H400c44.2 0 80-35.8 80-80V320c0-17.7-14.3-32-32-32s-32 14.3-32 32V432c0 8.8-7.2 16-16 16H80c-8.8 0-16-7.2-16-16V112c0-8.8 7.2-16 16-16H192c17.7 0 32-14.3 32-32s-14.3-32-32-32H80z"/></svg>
            </div>
        `

        const buttonApply = new west.gui.Button("Apply", function() {
            Vajda.settings.addEnergy = checkboxAddEnergy.isSelected()
            Vajda.settings.addMotivation = checkboxAddMotivation.isSelected()
            Vajda.settings.addHealth = checkboxAddHealth.isSelected()
            if ( Vajda.isNumber(healthStopTextfiled.getValue()) ) {
                let healthStop = parseInt(healthStopTextfiled.getValue())
                healthStop = Math.min(30,healthStop)
                Vajda.settings.healthStop = healthStop
            }

            if ( Vajda.isNumber(setWearDelayTextfiled.getValue()) ) {
                let setWearDelay = parseInt(setWearDelayTextfiled.getValue())
                setWearDelay = Math.min(10,setWearDelay)
                Vajda.settings.setWearDelay = setWearDelay
            }

            if ( Vajda.isNumber(jobDelayTextFieldMin.getValue()) ) {
                let jobDelayTimeMin = parseInt(jobDelayTextFieldMin.getValue())
                Vajda.settings.jobDelayMin = jobDelayTimeMin
            } else {
                Vajda.settings.jobDelayMin = 0
                Vajda.settings.jobDelayMax = 0
                new UserMessage("Wrong format of delay job min value. Please set a number.", UserMessage.TYPE_ERROR).show()
            }

            if ( Vajda.isNumber(jobDelayTextFieldMax.getValue()) ) {
                let jobDelayTimeMax = parseInt(jobDelayTextFieldMax.getValue())
                Vajda.settings.jobDelayMax = jobDelayTimeMax
            } else {
                Vajda.settings.jobDelayMin = 0
                Vajda.settings.jobDelayMax = 0
                new UserMessage("Wrong format of delay job max value. Please set a number.", UserMessage.TYPE_ERROR).show()
            }
            Vajda.setCookies()
            Vajda.selectTab("settings")
        })

        htmlSkel.append(manualLink)

        htmlSkel.append(checkboxAddEnergy.getMainDiv())
        htmlSkel.append("<br>")
        htmlSkel.append(checkboxAddMotivation.getMainDiv())
        htmlSkel.append("<br>")
        htmlSkel.append(checkboxAddHealth.getMainDiv())
        htmlSkel.append("<br>")
        htmlSkel.append(buttPlugsCheckbox.getMainDiv())
        htmlSkel.append("<br>")
        htmlSkel.append(enableObserverCheckbox.getMainDiv())
        htmlSkel.append("<br>")
        htmlSkel.append(observerDelay)
        htmlSkel.append(saveMoneyCheckbox.getMainDiv())
        htmlSkel.append(depositLimit)
        htmlSkel.append(htmlHealthStop)
        htmlSkel.append("<br>")
        htmlSkel.append(htmlSetWearDelay)
        htmlSkel.append("<br>")
        htmlSkel.append(htmlJobDelay)
        htmlSkel.append("<br>")
        htmlSkel.append(buttonApply.getMainDiv())
        return htmlSkel
    }

    Vajda.createManualGui = function() {
        const listStyle = `'padding-inline-start: 14px'`
        const nestedListStyle = `'padding-inline-start: 30px'`
        const containerStyle = `'
            padding: 0 30px 30px
        '`

        const html = `
            <div style=${containerStyle}>
                <h3>Read Before use</h3>
                <ol style=${listStyle}>
                    <li>
                        <strong>Calculating optimal route</strong> takes time and memory, a lot of it. It might not be noticeable for up to 12-14 jobs,
                        but anything more than that... you will notice a short freeze of the window. I don't recommend selecting more than 18 jobs, as the
                        algorithm will take about a minute (depending on the hardware, of course) to execute, while it takes almost 8 minutes to
                        find the best route for 20 jobs. In these scenarios, your browser may appear unresponsive and prompt you to reload the page 
                        or wait. I recommend being patient and allowing some time for the algorithm to complete. If that doesn't work, remove one
                        or two jobs; that's the cheapest way to solve this problem.
                    </li>

                    <li>
                        <strong>The optimization for butt plugs and cake decorations</strong> setting, which is most likely why you are here:
                        <ul style=${nestedListStyle}>
                            <li>The setting has to be enabled <i>before</i> you start Zdenka, it won't work otherwise</li>
                            <li>The <i>Add motivation</i> setting has to be enabled as well</li>
                            <li>Do not start Zdenka with low energy and consumables on cooldown</li>
                            <li>Do not use consumables yourself while it's running</li>
                            <li>
                                You don't have to select consumables manually with this setting, Zdenka will do it for you. Just make sure
                                you have enough of them butt plugs and decorations
                            </li>
                        </ul>

                        These restrictions are simply the cost of having Zdenka refill energy <i>while</i> doing jobs.
                    </li>

                    <li>
                        Zdenka can <strong>travel</strong> to jobs that you can't do in travel set (unless you are super low level).
                    </li>

                    <li>
                        <strong>Buffs</strong> will be used automatically when selected (and when no buff is active, of course). Only one buff
                        of each type can be selected at once. Keep in mind that consumables with cooldown are <b>not</b> treated as buffs and will
                        only be used to refill whatever your character needs.
                    </li>

                    <li>
                        Zdenka will display <strong>all silver jobs</strong>, not just those you can do and are closest to yoor character. This
                        allows you <i>some</i> control of the route, for instance there might be multiple silver jobs of the same kind and one 
                        of them might simply be in a more advantageous position.
                    </li>

                    <li>
                        When <strong>Auto Refresh</strong> is enabled, Zdenka will refresh the page and start automatically <i>n</i> minutes
                        after starting a job (or walking to one). Starting new jobs will restart this countdown. Do not set the delay to low numbers,
                        Zdenka enforces a minimum of 2.5 minutes anyway, however, I do recommend a slightly longer delay.
                    </li>
                </ol>
            </div>
        `

        const container = new west.gui.Scrollpane().getMainDiv()
        container.style = 'height: 400px; overflow-y: scroll'
        container.innerHTML = html


        return container
    }

    Vajda.createMenuIcon = function() {
        const menuImage = ''
        let div = $('<div class="ui_menucontainer" />')
        let link = $('<div id="vajda_menu" class="menulink" onclick=Vajda.loadJobs() title="Zdenka Studenková" />').css('background-image', 'url(' + menuImage + ')')
        $('#ui_menubar').append((div).append(link).append('<div class="menucontainer_bottom" />'))
    }


    Vajda.createWindow = function(isHumanAction = true) {
        const window = wman.open("vajda").setResizeable(false).setMinSize(650, 480).setSize(650, 480).setMiniTitle("Vajda Jožo")
        const tabs = {
            "jobs": "Jobs",
            "chosenJobs": "Chosen Jobs",
            "sets": "Sets",
            "consumables": "Consumables",
            "stats": "Statistics",
            "settings": "Settings",
            "manual": "User manual"
        }

        let tabLogic = function(win,id) {
            const content = $(`<div class='vajda-window'></div>`)
            switch(id) {
                case 'jobs': 
                    Vajda.loadJobData(function(){
                        Vajda.removeActiveTab(this)
                        Vajda.removeWindowContent()
                        Vajda.addActiveTab("jobs",this)
                        content.append(Vajda.createJobsTab())
                        Vajda.window.appendToContentPane(content)
                        Vajda.addJobTableCss()
                        $(".vajda-window .tw2gui_scrollpane_clipper_contentpane").css({"top": Vajda.jobTablePosition.content})
                        $(".vajda-window .tw2gui_scrollbar_pulley").css({"top": Vajda.jobTablePosition.scrollbar})
                        Vajda.addEventsHeader()
                    })
                    break
                case 'chosenJobs': 
                    Vajda.removeActiveTab(this)
                    Vajda.removeWindowContent()
                    Vajda.addActiveTab("chosenJobs", this)
                    content.append(Vajda.createAddedJobsTab())
                    Vajda.window.appendToContentPane(content)
                    $(".vajda-window .tw2gui_scrollpane_clipper_contentpane").css({"top": Vajda.addedJobTablePosition.content})
                    $(".vajda-window .tw2gui_scrollbar_pulley").css({"top": Vajda.addedJobTablePosition.scrollbar})
                    Vajda.addAddedJobsTableCss()
                    break
                case 'sets': 
                    Vajda.loadSets(function() {
                        Vajda.removeActiveTab(this)
                        Vajda.removeWindowContent()
                        Vajda.addActiveTab("sets",this)
                        content.append(Vajda.createSetGui())
                        Vajda.window.appendToContentPane(content)
                    })
                    break
                case 'consumables': 
                    Vajda.removeActiveTab(this)
                    Vajda.removeWindowContent()
                    Vajda.addActiveTab("consumables",this)
                    Vajda.findAllConsumables()
                    content.append(Vajda.createConsumablesTable())
                    Vajda.window.appendToContentPane(content)
                    $(".vajda-window .tw2gui_scrollpane_clipper_contentpane").css({"top": Vajda.consumableTablePosition.content})
                    $(".vajda-window .tw2gui_scrollbar_pulley").css({"top": Vajda.consumableTablePosition.scrollbar})
                    Vajda.addConsumableTableCss()
                    break
                case 'stats':
                    Vajda.removeActiveTab(this)
                    Vajda.removeWindowContent()
                    Vajda.addActiveTab("stats",this)
                    content.append(Vajda.createStatisticsGui())
                    Vajda.window.appendToContentPane(content)
                    break
                case 'settings': 
                    Vajda.removeActiveTab(this)
                    Vajda.removeWindowContent()
                    Vajda.addActiveTab("settings",this)
                    content.append(Vajda.createSettingsGui())
                    Vajda.window.appendToContentPane(content)
                    break
                case 'manual':
                    Vajda.removeActiveTab(this)
                    Vajda.removeWindowContent()
                    Vajda.addActiveTab("manual", this)
                    content.append(Vajda.createManualGui())
                    Vajda.window.appendToContentPane(content)
                    break
            }
        }

        for(let tab in tabs) {
            window.addTab(tabs[tab],tab,tabLogic)
        }

        Vajda.window = window

        if ( !isHumanAction ) wman.close('vajda')

        Vajda.selectTab('jobs')
    }

    Vajda.selectTab = function(key) {
        Vajda.window.tabIds[key].f(Vajda.window,key)
    }

    Vajda.removeActiveTab = function(window) {
        $('div.tw2gui_window_tab', window.divMain).removeClass('tw2gui_window_tab_active')
    }

    Vajda.addActiveTab = function(key, window) {
        $(`div._tab_id_${key}`, window.divMain).addClass('tw2gui_window_tab_active')
    }
    
    Vajda.removeWindowContent = function() {
        $(".vajda-window").remove()
    }

    Vajda.addConsumableTableCss = function() {
        $(".vajda-window .consumIcon").css({"width":"80px"})
        $(".vajda-window .consumCount").css({"width":"60px"})
        $(".vajda-window .consumEnergy").css({"width":"60px"})
        $(".vajda-window .consumMotivation").css({"width":"70px"})
        $(".vajda-window .consumHealth").css({"width":"60px"})
        $(".vajda-window .consumBuffs").css({"width": "150px"})
        $(".vajda-window .row").css({"height":"80px"})
        $('.vajda-window').find('.tw2gui_scrollpane').css('height', '250px')
    }

    Vajda.addJobTableCss = function() {
        $(".vajda-window .jobIcon").css({"width":"80px"})
        $(".vajda-window .jobName").css({"width":"150px"})
        $(".vajda-window .jobXp").css({"width":"40px"})
        $(".vajda-window .jobMoney").css({"width":"40px"})
        $(".vajda-window .jobMotivation").css({"width":"40px"})
        $(".vajda-window .jobDistance").css({"width":"100px"})
        $(".vajda-window .row").css({"height":"60px"})
        $('.vajda-window').find('.tw2gui_scrollpane').css('height', '250px')
    }

    Vajda.addAddedJobsTableCss = function() {
        $(".vajda-window .jobIcon").css({"width":"80px"})
        $(".vajda-window .jobName").css({"width":"130px"})
        $(".vajda-window .jobStopMotivation").css({"width":"110px"})
        $(".vajda-window .jobRemove").css({"width":"105px"})
        $(".vajda-window .jobSet").css({"width":"100px"})
        $(".vajda-window .row").css({"height":"60px"})
        $('.vajda-window').find('.tw2gui_scrollpane').css('height', '250px')
    }

    Vajda.createJobsTab = function() {
        const htmlSkel = $(`<div id='jobs_overview'></div>`)
        const html = $(`
            <div class='jobs_search' style='position: relative'>
                <div id='jobFilter' style='position: absolute; top: 10px; left: 15px'></div>
                <div id='job_only_silver' style='position: absolute; top:10px; left: 200px;'></div>
                <div id='job_no_silver' style='position: absolute; top: 10px; left: 270px;'></div>
                <div id='job_center' style='position: absolute; top: 10px; left: 350px;'></div>
                <div id='button_filter_jobs' style='position: absolute; top: 5px; left: 450px;'></div>
            </div>
        `)
        
        const table = new west.gui.Table()
        const xpIcon = '<img src="/images/icons/star.png">'
        const dollarIcon = '<img src="/images/icons/dollar.png">'
        const motivationIcon = '<img src="/images/icons/motivation.png">'
        const arrow_desc = '&nbsp;<img src="../images/window/jobs/sortarrow_desc.png"/>'
        const arrow_asc = '&nbsp;<img src="../images/window/jobs/sortarrow_asc.png"/>'
        const uniqueJobs = Vajda.getAllUniqueJobs()
        table
            .addColumn("jobIcon","jobIcon")
            .addColumn("jobName","jobName")
            .addColumn("jobXp","jobXp")
            .addColumn("jobMoney","jobMoney")
            .addColumn("jobMotivation","jobMotivation")
            .addColumn("jobDistance","jobDistance")
            .addColumn("jobAdd","jobAdd")
        table.appendToCell("head","jobIcon","Job icon").appendToCell("head","jobName","Job name").appendToCell("head","jobXp",xpIcon + (Vajda.sortJobTableXp == 1 ? arrow_asc : Vajda.sortJobTableXp == -1 ? arrow_desc : "")).appendToCell("head","jobMoney",dollarIcon).appendToCell("head","jobMotivation",motivationIcon).appendToCell("head","jobDistance","Distance " + (Vajda.sortJobTableDistance == 1 ? arrow_asc : Vajda.sortJobTableDistance == -1 ? arrow_desc : "")).appendToCell("head","jobAdd","")
        for ( let job = 0; job < uniqueJobs.length; job++ ) {
            table
                .appendRow()
                .appendToCell(-1,"jobIcon",Vajda.getJobIcon(uniqueJobs[job].isSilver,uniqueJobs[job].id,uniqueJobs[job].x,uniqueJobs[job].y))
                .appendToCell(-1,"jobName",Vajda.getJobName(uniqueJobs[job].id))
                .appendToCell(-1,"jobXp",uniqueJobs[job].experience)
                .appendToCell(-1,"jobMoney",uniqueJobs[job].money)
                .appendToCell(-1,"jobMotivation",uniqueJobs[job].motivation)
                .appendToCell(-1,"jobDistance",uniqueJobs[job].distance.formatDuration())
                .appendToCell(-1,"jobAdd", Vajda.createAddJobButton(uniqueJobs[job].x,uniqueJobs[job].y,uniqueJobs[job].id))
        }
        const textfield = new west.gui.Textfield("jobsearch").setPlaceholder("Select job name")
        if ( Vajda.jobFilter.filterJob !== "" ) {
            textfield.setValue(Vajda.jobFilter.filterJob)
        }

        const checkboxOnlySilver = new west.gui.Checkbox()
        checkboxOnlySilver.setLabel("Silvers")
        checkboxOnlySilver.setSelected(Vajda.jobFilter.filterOnlySilver)
        checkboxOnlySilver.setCallback(function() {
            if ( this.isSelected() ) {
                Vajda.jobFilter.filterOnlySilver = true
            }else {
                Vajda.jobFilter.filterOnlySilver = false
            }
        })
        const checkboxNoSilver = new west.gui.Checkbox()
        checkboxNoSilver.setLabel("No silvers")
        checkboxNoSilver.setSelected(Vajda.jobFilter.filterNoSilver)
        checkboxNoSilver.setCallback(function() {
            if ( this.isSelected() ) {
                Vajda.jobFilter.filterNoSilver = true
            } else {
                Vajda.jobFilter.filterNoSilver = false
            }
        })
        const checkboxCenterJobs = new west.gui.Checkbox()
        checkboxCenterJobs.setLabel("Center jobs")
        checkboxCenterJobs.setSelected(Vajda.jobFilter.filterCenterJobs)
        checkboxCenterJobs.setCallback(function() {
            if ( this.isSelected() ) {
                Vajda.jobFilter.filterCenterJobs = true
            } else {
                Vajda.jobFilter.filterCenterJobs = false
            }
        })
        const buttonFilter = new west.gui.Button("Filter", function() {
            Vajda.jobFilter.filterJob = textfield.getValue()
            Vajda.jobTablePosition.content = "0px"
            Vajda.jobTablePosition.scrollbar = "0px"
            Vajda.selectTab("jobs")
        })
        htmlSkel.append(table.getMainDiv())
        $('#jobFilter', html).append(textfield.getMainDiv())
        $("#job_only_silver",html).append(checkboxOnlySilver.getMainDiv())
        $("#job_no_silver",html).append(checkboxNoSilver.getMainDiv())
        $("#job_center",html).append(checkboxCenterJobs.getMainDiv())
        $("#button_filter_jobs",html).append(buttonFilter.getMainDiv())
        htmlSkel.append(html)
        return htmlSkel
    }

    Vajda.createAddJobButton = function(x, y, id) {
        const buttonAdd = new west.gui.Button("Add new job", function() {
            Vajda.addJob(x, y, id)
            Vajda.jobTablePosition.content = $(".vajda-window .tw2gui_scrollpane_clipper_contentpane").css("top")
            Vajda.jobTablePosition.scrollbar = $(".vajda-window .tw2gui_scrollbar_pulley").css("top")
            Vajda.selectTab("jobs")
        })
        buttonAdd.setWidth(100)
        return buttonAdd.getMainDiv()
    }

    Vajda.createMinMotivationTextfield = function(x, y, id, placeholder) {
        const componentId = `x-${x}y-${y}id-${id}`
        const textfield = new west.gui.Textfield()
        textfield.setId(componentId)
        textfield.setWidth(40)
        textfield.setValue(placeholder)
        return textfield.getMainDiv()
    }

    Vajda.createRemoveJobButton = function(x, y, id) {
        const buttonRemove = new west.gui.Button("Remove job", function() {
            Vajda.removeJob(x, y, id)
            Vajda.addedJobTablePosition.content = $(".vajda-window .tw2gui_scrollpane_clipper_contentpane").css("top")
            Vajda.addedJobTablePosition.scrollbar = $(".vajda-window .tw2gui_scrollbar_pulley").css("top")
            Vajda.selectTab("chosenJobs")
        })
        buttonRemove.setWidth(100)
        return buttonRemove.getMainDiv()
    }


    $(document).ready(() => {
        try {
            Vajda.loadLanguage()
            Vajda.loadSets()
            Vajda.createMenuIcon()
            Vajda.getCookies()
            Observer.resumeSession()
        } catch(e) {
            console.log(e)
            console.log("exception occured")
        }
    })
})()