IdlePixel Slap Chop

Ain't nobody got time for that! Adds some QoL 1-click actions.

// ==UserScript==
// @name         IdlePixel Slap Chop
// @namespace    com.anwinity.idlepixel
// @version      1.1.22
// @description  Ain't nobody got time for that! Adds some QoL 1-click actions.
// @author       Anwinity
// @icon         
// @license      MIT
// @match        *://idle-pixel.com/login/play*
// @grant        none
// @require      https://greasyfork.org/scripts/441206-idlepixel/code/IdlePixel+.js?anticache=20220905
// ==/UserScript==

(function() {
    'use strict';

    const IMAGE_URL_BASE = $("itembox[data-item=copper] img").attr("src").replace(/\/[^/]+.png$/, "");
    const SMELTABLES =  ["copper", "iron", "silver", "gold", "promethium", "titanium"];
    const PLANTABLES = $('itembox[data-item$="_seeds"]').toArray().map(el => el.getAttribute("data-item"));
    const EDIBLES = Object.keys(Cooking.ENERGY_MAP).filter(s => !s.startsWith("raw_"));
    const COOKABLES = Object.keys(Cooking.FOOD_HEAT_REQ_MAP);
    const BONEMEALABLE = ["bones", "big_bones", "ice_bones", "ashes", "blood_bones"];
    const POTIONS = Object.keys(Brewing.POTION_TIMERS);
    const BOATS = $(`itembox[data-item$="_boat"]`).toArray().map(el => el.getAttribute("data-item"));
    const NEEDLEABLE = ["lizard_mask", "lizard_body", "lizard_legs", "lizard_boots", "lizard_gloves", "bat_mask", "bat_body", "bat_legs", "bat_boots", "bat_gloves", "bear_mask", "bear_body", "bear_legs", "bear_boots", "bear_gloves"];
    const LOOT_BAGS = $(`itembox[data-item^="gathering_loot_bag_"]`).toArray().map(el => el.getAttribute("data-item"));
    const GRINDABLE = $(`#panel-invention itembox[data-item^="blood_"][onclick^="Invention.clicks_limb"]`).toArray().map(el => el.getAttribute("data-item"));
    const LOGS = Object.keys(Cooking.LOG_HEAT_MAP);
    const FEATHER2ARROW = {
        feathers: {
            craft: "wooden_arrows",
            required: {
                feathers: 15,
                logs: 5,
                iron_bar: 5
            }
        },
        fire_feathers: {
            craft: "fire_arrows",
            required: {
                fire_feathers: 15,
                oak_logs: 5,
                silver_bar: 5
            }
        },
        ice_feathers: {
            craft: "ice_arrows",
            required: {
                ice_feathers: 15,
                willow_logs: 5,
                gold_bar: 5
            }
        }
    };

    class SlapChopPlugin extends IdlePixelPlusPlugin {
        constructor() {
            super("slapchop", {
                about: {
                    name: GM_info.script.name,
                    version: GM_info.script.version,
                    author: GM_info.script.author,
                    description: GM_info.script.description
                },
                config: [
                    {
                        id: "primaryActionKey",
                        label: "Primary Action Key",
                        type: "select",
                        options: [
                            {value:"none", label:"None"},
                            {value:"altKey", label:"Alt"},
                            {value:"shiftKey", label:"Shift"},
                            {value:"ctrlKey", label:"Ctrl"}
                        ],
                        default: "none"
                    },
                    {
                        id: "altActionKey",
                        label: "Alt Action Key",
                        type: "select",
                        options: [
                            {value:"altKey", label:"Alt"},
                            {value:"shiftKey", label:"Shift"},
                            {value:"ctrlKey", label:"Ctrl"}
                        ],
                        default: "altKey"
                    },
                    {
                        id: "quickFightEnabled",
                        label: "Quick Fight: Enabled",
                        type: "boolean",
                        default: true
                    },
                    {
                        id: "quickPresetsEnabled",
                        label: "Quick Presets: Enabled",
                        type: "boolean",
                        default: true
                    },
                    {
                        id: "quickFightFPBar",
                        label: "Quick Fight: FP Bar",
                        type: "boolean",
                        default: true
                    },
                    {
                        id: "quickFightEnergyBar",
                        label: "Quick Fight: Energy Bar",
                        type: "boolean",
                        default: true
                    },
                    {
                        id: "quickFightConfirm",
                        label: "Quick Fight: Confirm",
                        type: "boolean",
                        default: false
                    },
                    {
                        id: "quickSmeltEnabled",
                        label: "Quick Smelt (buttons): Enabled",
                        type: "boolean",
                        default: true
                    },
                    {
                        id: "quickSmeltRightClickEnabled",
                        label: "Quick Smelt (right-click, primary=max): Enabled",
                        type: "boolean",
                        default: true
                    },
                    {
                        id: "quickFoundryEnabled",
                        label: "Quick Foundry (buttons): Enabled",
                        type: "boolean",
                        default: true
                    },
                    {
                        id: "quickCookRightClickEnabled",
                        label: "Quick Cook (right-click, primary=max, alt=keep-1): Enabled",
                        type: "boolean",
                        default: true
                    },
                    {
                        id: "quickEatRightClickEnabled",
                        label: "Quick Eat (right-click, primary=max, alt=keep-1): Enabled",
                        type: "boolean",
                        default: true
                    },
                    {
                        id: "quickPlantRightClickEnabled",
                        label: "Quick Plant (right-click, primary=1, alt=max): Enabled",
                        type: "boolean",
                        default: true
                    },
                    {
                        id: "quickChopEnabled",
                        label: "Quick Chop (Lumberjack): Enabled",
                        type: "boolean",
                        default: true
                    },
                    {
                        id: "quickHarvestEnabled",
                        label: "Quick Harvest (Bob): Enabled",
                        type: "boolean",
                        default: true
                    },
                    {
                        id: "quickBoneRightClickEnabled",
                        label: "Quick Bonemeal (right-click, primary=max, alt=keep-1): Enabled",
                        type: "boolean",
                        default: true
                    },
                    {
                        id: "quickBrewButtonEnabled",
                        label: "Quick Brew (buttons): Enabled",
                        type: "boolean",
                        default: true
                    },
                    {
                        id: "quickPotionRightClickEnabled",
                        label: "Quick Potion (right-click, primary=1): Enabled",
                        type: "boolean",
                        default: true
                    },
                    {
                        id: "quickBoatRightClickEnabled",
                        label: "Quick Boat (right-click): Enabled",
                        type: "boolean",
                        default: true
                    },
                    {
                        id: "quickNeedleRightClickEnabled",
                        label: "Quick Needle (right-click, primary=max, alt=keep-1): Enabled",
                        type: "boolean",
                        default: false
                    },
                    {
                        id: "quickGatherRightClickEnabled",
                        label: "Quick Gather (right-click, primary=max, alt=keep-1): Enabled",
                        type: "boolean",
                        default: true
                    },
                    {
                        id: "quickBurnRightClickEnabled",
                        label: "Quick Burn Logs (right-click, primary=max, alt=keep-1): Enabled",
                        type: "boolean",
                        default: true
                    },
                    {
                        id: "quickCraftArrowRightClickEnabled",
                        label: "Quick Craft Arrow (right-click feather, primary=max): Enabled",
                        type: "boolean",
                        default: true
                    },
                    {
                        id: "quickGrindRightClickEnabled",
                        label: "Quick Blood Grind (right-click, primary=1): Enabled",
                        type: "boolean",
                        default: true
                    },
                ]
            });
        }

        onConfigsChanged() {
            if(this.getConfig("quickFightEnabled")) {
                $("#slapchop-quickfight").show();
            }
            else {
                $("#slapchop-quickfight").hide();
            }

            if(this.getConfig("quickFoundryEnabled")) {
                $("#slapchop-quickfoundry").show();
            }
            else {
                $("#slapchop-quickfoundry").hide();
            }

            const presetsUnlocked = IdlePixelPlus.getVar("combat_presets") == "1";
            if(presetsUnlocked && this.getConfig("quickPresetsEnabled")) {
                $("#slapchop-quickpreset").show();
            }
            else {
                $("#slapchop-quickpreset").hide();
            }

            if(this.getConfig("quickFightFPBar")) {
                $(".slapchop-quickfight-fightpoints").show();
            }
            else {
                $(".slapchop-quickfight-fightpoints").hide();
            }

            if(this.getConfig("quickFightEnergyBar")) {
                $(".slapchop-quickfight-energy").show();
            }
            else {
                $(".slapchop-quickfight-energy").hide();
            }

            if(this.getConfig("quickSmeltEnabled")) {
                $("#slapchop-quicksmelt").show();
            }
            else {
                $("#slapchop-quicksmelt").hide();
            }

            if(this.getConfig("quickBrewButtonEnabled")) {
                $(".slapchop-quickbrew-button").show();
            }
            else {
                $(".slapchop-quickbrew-button").hide();
            }

            if(this.getConfig("quickHarvestEnabled")) {
                window.var_slapchop_bob = "1";
            }
            else {
                window.var_slapchop_bob = "0";
            }

            if(this.getConfig("quickChopEnabled")) {
                window.var_slapchop_lumberjack = "1";
            }
            else {
                window.var_slapchop_lumberjack = "0";
            }


        }

        isPrimaryAction(event) {
            const prop = this.getConfig("primaryActionKey") || "none";
            if(prop=="none") {
                return !(event.altKey || event.ctrlKey || event.shiftKey);
            }
            else {
                return event[prop];
            }
        }

        isAltAction(event) {
            const prop = this.getConfig("altActionKey") || "altKey";
            return event[prop];
        }

        setItemBoxOverlay(items, enabled) {
            items.forEach(item => {
                if(enabled) {
                    $(`itembox[data-item=${item}]`).addClass("slapchop-overlay");
                }
                else {
                    $(`itembox[data-item=${item}]`).removeClass("slapchop-overlay");
                }
            });
        }

        initStyles() {
            $("head").append(`
            <style id="styles-slapchop">

            #slapchop-quickfight, #slapchop-quickpreset {
              position: relative;
            }
            #slapchop-quickpreset > .slapchop-quickpreset-buttons {
              display: flex;
              flex-direction: row;
              justify-content: start;
            }
            #slapchop-quickpreset > .slapchop-quickpreset-buttons > div {
              display: flex;
              flex-direction: column;
              justify-content: start;
            }
            #slapchop-quickpreset > .slapchop-quickpreset-buttons > div > button {
              margin: 0.125em;
            }
            #slapchop-quickfight > .slapchop-quickfight-buttons {
              display: flex;
              flex-direction: row;
              flex-wrap: wrap;
            }
            #slapchop-quickfight > .slapchop-quickfight-buttons .slapchop-quickfight-zone {
              width: 150px;
              max-width: 150px;
              display: flex;
              flex-direction: column;
              justify-content: start;
              align-items: center;
              position: relative;
            }
            #slapchop-quickfight > .slapchop-quickfight-buttons .slapchop-quickfight-zone.blood button {
              font-weight: 550;
              background-color: rgb(136, 8, 8) !important;
              color: rgb(255,255,255);
            }
            #slapchop-quickfight > .slapchop-quickfight-buttons .slapchop-quickfight-zone.blood button:disabled {
              color: rgba(255,255,255,0.3);
              background-color: rgba(136, 8, 8, 0.3) !important;
            }
            #slapchop-quickfight > .slapchop-quickfight-buttons .slapchop-quickfight-zone > * {
              width: 100%;
            }
            #slapchop-quickfight > .slapchop-quickfight-buttons .slapchop-quickfight-zone > .slapchop-quickfight-progress-container {
              width: 100%;
              color: white;
              text-shadow: 1px 0 0 #000, 0 -1px 0 #000, 0 1px 0 #000, -1px 0 0 #000;
              text-align: left;
              position: relative;
            }
            #slapchop-quickfight > .slapchop-quickfight-buttons .slapchop-quickfight-zone > .slapchop-quickfight-progress-container > .slapchop-quickfight-progress-value {
              position: relative;
              z-index: 5;
              margin-left: 4px;
              font-weight: bold;
            }
            #slapchop-quickfight > .slapchop-quickfight-buttons .slapchop-quickfight-zone > .slapchop-quickfight-progress-container.slapchop-quickfight-fightpoints {
              background-color: rgba(255, 216, 0, 0.5);
              border: 1px solid rgb(255, 216, 0);
            }
            #slapchop-quickfight > .slapchop-quickfight-buttons .slapchop-quickfight-zone > .slapchop-quickfight-progress-container.slapchop-quickfight-fightpoints .slapchop-quickfight-progress {
              background-color: #ffd800;
            }
            #slapchop-quickfight > .slapchop-quickfight-buttons .slapchop-quickfight-zone > .slapchop-quickfight-progress-container.slapchop-quickfight-energy {
              background-color: rgba(215, 0, 71, 0.5);
              border: 1px solid rgb(215, 0, 71);
            }
            #slapchop-quickfight > .slapchop-quickfight-buttons .slapchop-quickfight-zone > .slapchop-quickfight-progress-container.slapchop-quickfight-energy .slapchop-quickfight-progress {
              background-color: #d70047;
            }
            #slapchop-quickfight > .slapchop-quickfight-buttons .slapchop-quickfight-zone > .slapchop-quickfight-progress-container > .slapchop-quickfight-progress {
              position: absolute;
              top: 0;
              left: 0;
              height: 100%;
              width: 0; /* will be overwritten inline */
              z-index: 3;
            }
            #brewing-table .slapchop-quickbrew-button {
              border: 1px solid rgba(124, 218, 255, 0.86);
              background-color: rgba(124, 218, 255, 0.1);
              padding: 2px;
              display: flex;
              justify-content: center;
              align-items: center;
              margin: 0.5em auto 0.125em auto;
              max-width: 100px;
            }
            #brewing-table .slapchop-quickbrew-button:hover {
              background-color: rgba(69, 177, 216, 0.5);
            }
            </style>
            `);
        }

        quickFight(zoneId) {
            const confirm = this.getConfig("quickFightConfirm");
            if(confirm) {
                if(window.confirm(`FIGHT: ${zoneId.replace(/_/g, " ").replace(/(^|\s)\w/g, s => s.toUpperCase())} ?`)) {
                    if(zoneId.startsWith("blood_")) {
                        Combat.modal_blood_area_last_selected = zoneId;
                    }
                    else {
                        Combat.modal_area_last_selected = zoneId;
                    }
                    IdlePixelPlus.sendMessage(`START_FIGHT=${zoneId}`);
                }
            }
            else {
                if(zoneId.startsWith("blood_")) {
                    Combat.modal_blood_area_last_selected = zoneId;
                }
                else {
                    Combat.modal_area_last_selected = zoneId;
                }
                IdlePixelPlus.sendMessage(`START_FIGHT=${zoneId}`);
            }
        }

        initQuickFight() {
            let html = `
            <div id="slapchop-quickfight">
              <h5>Quick Fight:</h5>
              <div class="slapchop-quickfight-buttons">
            `;
            Object.values(IdlePixelPlus.info.combatZones).forEach(zone => {
                html += `
                <div id="slapchop-quickfight-${zone.id}" class="slapchop-quickfight-zone m-1 ${zone.blood?'blood':''}">
                  <button type="button" onclick="IdlePixelPlus.plugins.slapchop.quickFight('${zone.id}')">${zone.id.replace(/_/g, " ").replace(/(^|\s)\w/g, s => s.toUpperCase())}</button>
                  <div class="slapchop-quickfight-fightpoints slapchop-quickfight-progress-container" title="Fight Points: ${zone.fightPointCost.toLocaleString()}">
                    <span class="slapchop-quickfight-progress-value">0</span>
                    <div class="slapchop-quickfight-progress"></div>
                  </div>
                  <div class="slapchop-quickfight-energy slapchop-quickfight-progress-container" title="Energy: ${zone.energyCost.toLocaleString()}">
                    <span class="slapchop-quickfight-progress-value">0</span>
                    <div class="slapchop-quickfight-progress"></div>
                  </div>
                </div>
                `;
            });
            html += `
              </div>
              <hr>
            </div>
            <div id="slapchop-quickpreset">
              <h5>Quick Presets:</h5>
              <div class="slapchop-quickpreset-buttons">
                <div>
                  <button onclick="IdlePixelPlus.sendMessage('PRESET_SAVE=1')">Save 1</button>
                  <button onclick="IdlePixelPlus.sendMessage('PRESET_LOAD=1~0')">Load 1</button>
                </div>
                <div>
                  <button onclick="IdlePixelPlus.sendMessage('PRESET_SAVE=2')">Save 2</button>
                  <button onclick="IdlePixelPlus.sendMessage('PRESET_LOAD=2~0')">Load 2</button>
                </div>
                <div>
                  <button onclick="IdlePixelPlus.sendMessage('PRESET_SAVE=3')">Save 3</button>
                  <button onclick="IdlePixelPlus.sendMessage('PRESET_LOAD=3~0')">Load 3</button>
                </div>
                <div>
                  <button onclick="IdlePixelPlus.sendMessage('PRESET_SAVE=4')">Save 4</button>
                  <button onclick="IdlePixelPlus.sendMessage('PRESET_LOAD=4~0')">Load 4</button>
                </div>
                <div>
                  <button onclick="IdlePixelPlus.sendMessage('PRESET_SAVE=5')">Save 5</button>
                  <button onclick="IdlePixelPlus.sendMessage('PRESET_LOAD=5~0')">Load 5</button>
                </div>
              </div>
              <hr>
            </div>
            `;
            $("#panel-combat hr").first().after(html);
        }

        updateQuickFight() {
            const fp = IdlePixelPlus.getVarOrDefault("fight_points", 0, "int");
            const energy = IdlePixelPlus.getVarOrDefault("energy", 0, "int");
            Object.values(IdlePixelPlus.info.combatZones).forEach(zone => {
                let disabled = fp < zone.fightPointCost || energy < zone.energyCost;
                let fpPercent = (fp / zone.fightPointCost).toFixed(2).split(".");
                let energyPercent = (energy / zone.energyCost).toFixed(2).split(".");

                let fpLabel = `&times; ${fpPercent[0]} + ${fpPercent[1].replace(/^0/, "")}%`;
                let energyLabel = `&times; ${energyPercent[0]} + ${energyPercent[1].replace(/^0/, "")}%`;

                if(zone.id == "volcano" && IdlePixelPlus.getVar("volcano_unlocked")!="1") {
                    disabled = true;
                }
                else if(zone.id == "northern_field" && IdlePixelPlus.getVar("northern_field_unlocked")!="1") {
                    disabled = true;
                }
                else if(zone.id == "mansion" && IdlePixelPlus.getVar("mansion_unlocked")!="1") {
                    disabled = true;
                }
                const button = $(`#slapchop-quickfight-${zone.id} button`);
                button.prop("disabled", disabled);

                $(`#slapchop-quickfight-${zone.id} .slapchop-quickfight-fightpoints .slapchop-quickfight-progress`).css("width", `${fpPercent[1]}%`);
                $(`#slapchop-quickfight-${zone.id} .slapchop-quickfight-energy .slapchop-quickfight-progress`).css("width", `${energyPercent[1]}%`);

                $(`#slapchop-quickfight-${zone.id} .slapchop-quickfight-fightpoints .slapchop-quickfight-progress-value`).html(fpLabel);
                $(`#slapchop-quickfight-${zone.id} .slapchop-quickfight-energy .slapchop-quickfight-progress-value`).html(energyLabel);
            });
        }

        maxSmeltable(ore) {
            const oilPerOre = Crafting.getOilPerBar(ore);
            const charcoalPerOre = Crafting.getCharcoalPerBar(ore);
            const lavaPerOre = Crafting.getLavaPerBar(ore);

            const oil = IdlePixelPlus.getVarOrDefault("oil", 0, "int");
            const capacity = Furnace.getFurnaceCapacity();
            const oreCount = IdlePixelPlus.getVarOrDefault(ore, 0, "int");
            const maxSmeltFromOil = Math.floor(oil / oilPerOre);
            let maxSmeltCount = Math.min(capacity, oreCount, maxSmeltFromOil);
            if(charcoalPerOre > 0) {
                const charcoal = IdlePixelPlus.getVarOrDefault("charcoal", 0, "int");
                const maxSmeltFromCharcoal = Math.floor(charcoal / charcoalPerOre);
                maxSmeltCount = Math.min(maxSmeltCount, maxSmeltFromCharcoal);
            }
            if(lavaPerOre > 0) {
                const lava = IdlePixelPlus.getVarOrDefault("lava", 0, "int");
                const maxSmeltFromLava = Math.floor(lava / lavaPerOre);
                maxSmeltCount = Math.min(maxSmeltCount, maxSmeltFromLava);
            }
            return maxSmeltCount || 0;
        }

        quickSmelt(ore) {
            const current = IdlePixelPlus.getVarOrDefault("furnace_ore_type", "none");
            if(current == "none") {
                const max = this.maxSmeltable(ore);
                if(max > 0) {
                    IdlePixelPlus.sendMessage(`SMELT=${ore}~${max}`);
                }
            }
        }

        initQuickSmelt() {
            let html = `
            <div class="slapchop-quickfight">
              <h5>Quick Smelt:</h5>
              <div class="slapchop-quicksmelt-buttons">
            `;
            SMELTABLES.forEach(ore => {
                html += `
                  <button type="button" onclick="IdlePixelPlus.plugins.slapchop.quickSmelt('${ore}')">
                    <img src="${IMAGE_URL_BASE}/${ore}.png" class="img-20" />
                    ${ore.replace(/_/g, " ").replace(/(^|\s)\w/g, s => s.toUpperCase())}
                    (<span data-slap="max-smelt-${ore}">?</span>)
                  </button>
                `;
            });
            html += `
              </div>
              <hr>
            </div>
            `;
            $("#panel-mining hr").first().after(html);
            $("#panel-crafting hr").first().after(html);

            SMELTABLES.forEach(ore => {
              $(`itembox[data-item="${ore}"]`).on("contextmenu", event => {
                  if(this.getConfig("quickSmeltRightClickEnabled")) {
                      const primary = this.isPrimaryAction(event);
                      const alt = this.isAltAction(event);
                      if(primary || alt) {
                          this.quickSmelt(item, !primary);
                          event.stopPropagation();
                          event.preventDefault();
                          return false;
                      }
                  }
                  return true;
              });
            });
        }

        updateQuickSmelt() {
            SMELTABLES.forEach(ore => {
                const max = this.maxSmeltable(ore);
                $(`[data-slap="max-smelt-${ore}"]`).text(max);
            });
        }

         initQuickFoundry() {
            let html = `
            <div id="slapchop-quickfoundry" class="slapchop-quickfight">
              <h5>Quick Foundry:</h5>
              <div class="slapchop-quicksmelt-buttons">
            `;
            LOGS.forEach(log => {
                html += `
                  <button id="slapchop-quickfoundry-${log}" type="button" onclick="IdlePixelPlus.plugins.slapchop.quickFoundry('${log}')">
                    <img src="${IMAGE_URL_BASE}/${log}.png" class="img-20" />
                    ${log.replace("_logs", "").replace(/_/g, " ").replace(/(^|\s)\w/g, s => s.toUpperCase())}
                    (<span data-slap="max-foundry-${log}">?</span>)
                  </button>
                `;
            });
            html += `
              </div>
              <hr>
            </div>
            `;
            $("#panel-woodcutting hr").first().after(html);
        }

        updateQuickFoundry() {
            const foundryBusy = IdlePixelPlus.getVarOrDefault("foundry_amount", 0, "int") != 0;
            LOGS.forEach(log => {
                const max = this.maxFoundry(log);
                $(`[data-slap="max-foundry-${log}"]`).text(max);
                if(!foundryBusy && max > 0) {
                    $(`#slapchop-quickfoundry-${log}`).prop("disabled", false);
                }
                else {
                    $(`#slapchop-quickfoundry-${log}`).prop("disabled", true);
                }
            });
        }

        quickFoundry(log) {
            const max = this.maxFoundry(log);
            if(max > 0) {
                IdlePixelPlus.sendMessage(`FOUNDRY=${log}~${max}`);
            }
        }

        maxFoundry(log) {
            if(IdlePixelPlus.getVarOrDefault("charcoal_foundry_crafted", "0") != "1") {
                return 0;
            }
            let max = IdlePixelPlus.getVarOrDefault(log, 0, "int");
            if(max > 100) {
                max = 100;
            }
            let oilMax = Math.floor(IdlePixelPlus.getVarOrDefault("oil", 0, "int")/10);
            if(max > oilMax) {
                max = oilMax;
            }
            return max;
        }

        initQuickCook() {
            COOKABLES.forEach(item => {
              $(`itembox[data-item="${item}"]`).on("contextmenu", event => {
                  if(this.getConfig("quickCookRightClickEnabled")) {
                      const primary = this.isPrimaryAction(event);
                      const alt = this.isAltAction(event);
                      if(primary || alt) {
                          this.quickCook(item, !primary);
                          event.stopPropagation();
                          event.preventDefault();
                          return false;
                      }
                  }
                  return true;
              });
            });
        }

        maxCookable(food) {
            return Cooking.can_cook_how_many(food) || 0;
        }

        quickCook(food, alt) {
            const max = this.maxCookable(food);
            let n = max;
            if(alt) {
                const owned = IdlePixelPlus.getVarOrDefault(food, 0, "int");
                if(owned==max) {
                    n--;
                }
            }
            if(n > 0) {
                IdlePixelPlus.sendMessage(`COOK=${food}~${n}`);
            }
        }

        quickEat(food, alt) {
            let n = IdlePixelPlus.getVarOrDefault(food, 0, "int");
            if(alt) {
                n--;
            }
            if(n > 0) {
                IdlePixelPlus.sendMessage(`CONSUME=${food}~${n}`);
            }
        }

        initQuickEat() {
            EDIBLES.forEach(item => {
              $(`itembox[data-item="${item}"]`).on("contextmenu", event => {
                  if(this.getConfig("quickEatRightClickEnabled")) {
                      const primary = this.isPrimaryAction(event);
                      const alt = this.isAltAction(event);
                      if(primary || alt) {
                          this.quickEat(item, !primary);
                          event.stopPropagation();
                          event.preventDefault();
                          return false;
                      }
                  }
                  return true;
              });
            });
        }

        quickPlant(seed, alt) {
            let n = IdlePixelPlus.getVarOrDefault(seed, 0, "int");
            if(!alt && n>1) {
                n = 1;
            }
            const donor = DonorShop.has_donor_active(Items.getItem("donor_farm_patches_timestamp"));
            const maxPlot = donor ? 5 : 3;
            for(let plot = 1; plot <= maxPlot && n > 0; plot++) {
                if(IdlePixelPlus.getVar(`farm_${plot}`) == "none") {
                    IdlePixelPlus.sendMessage(`PLANT=${seed}~${plot}`);
                    n--;
                }
            }
        }

        initQuickPlant() {
            PLANTABLES.forEach(item => {
              $(`itembox[data-item="${item}"]`).on("contextmenu", event => {
                  if(this.getConfig("quickPlantRightClickEnabled")) {
                      const primary = this.isPrimaryAction(event);
                      const alt = this.isAltAction(event);
                      if(primary || alt) {
                          this.quickPlant(item, !primary);
                          event.stopPropagation();
                          event.preventDefault();
                          return false;
                      }
                  }
                  return true;
              });
            });
        }

        quickBone(item, alt) { /* lol */
            if(IdlePixelPlus.getVarOrDefault("bonemeal_bin", 0, "int") != 0) {
                let n = IdlePixelPlus.getVarOrDefault(item, 0, "int");
                if(alt) {
                    n--;
                }
                if(n > 0) {
                    IdlePixelPlus.sendMessage(`ADD_BONEMEAL=${item}~${n}`);
                }
            }
        }

        initQuickBones() {
            BONEMEALABLE.forEach(item => {
              $(`itembox[data-item="${item}"]`).on("contextmenu", event => {
                  if(this.getConfig("quickBoneRightClickEnabled")) {
                      const primary = this.isPrimaryAction(event);
                      const alt = this.isAltAction(event);
                      if(primary || alt) {
                          this.quickBone(item, !primary);
                          event.stopPropagation();
                          event.preventDefault();
                          return false;
                      }
                  }
                  return true;
              });
            });
        }

        quickPotion(potion) {
            const n = IdlePixelPlus.getVarOrDefault(potion, 0, "int");
            if(n > 0) {
                IdlePixelPlus.sendMessage(`DRINK=${potion}`);
            }
        }

        initQuickPotions() {
            POTIONS.forEach(item => {
              $(`itembox[data-item="${item}"]`).on("contextmenu", event => {
                  if(this.getConfig("quickPotionRightClickEnabled")) {
                      const primary = this.isPrimaryAction(event);
                      const alt = this.isAltAction(event);
                      if(primary || alt) {
                          this.quickPotion(item, !primary);
                          event.stopPropagation();
                          event.preventDefault();
                          return false;
                      }
                  }
                  return true;
              });
            });
        }

        quickBoat(item) {
            const n = IdlePixelPlus.getVar(`${item}_timer`);
            if(n == "1") {
                IdlePixelPlus.sendMessage(`BOAT_COLLECT=${item}`);
            }
            else {
                IdlePixelPlus.sendMessage(`BOAT_SEND=${item}`);
            }
        }

        initQuickBoat() {
            BOATS.forEach(item => {
                $(`itembox[data-item="${item}"]`).on("contextmenu", event => {
                    if(this.getConfig("quickBoatRightClickEnabled")) {
                        const primary = this.isPrimaryAction(event);
                        const alt = this.isAltAction(event);
                        if(primary || alt) {
                            this.quickBoat(item, !primary);
                            event.stopPropagation();
                            event.preventDefault();
                            return false;
                        }
                    }
                    return true;
                });
            });
        }

        quickNeedle(item, alt) {
            let n = IdlePixelPlus.getVarOrDefault(item, 0, "int");
            if(alt) {
                n--;
            }
            if(n > 0) {
                IdlePixelPlus.sendMessage(`USE_NEEDLE=${item}~${n}`);
            }
        }

        initQuickNeedle() {
            NEEDLEABLE.forEach(item => {
                $(`itembox[data-item="${item}"]`).on("contextmenu", event => {
                    if(this.getConfig("quickNeedleRightClickEnabled")) {
                        const primary = this.isPrimaryAction(event);
                        const alt = this.isAltAction(event);
                        if(primary || alt) {
                            this.quickNeedle(item, !primary);
                            event.stopPropagation();
                            event.preventDefault();
                            return false;
                        }
                    }
                    return true;
                });
            });
        }

        quickGather(bag, alt) {
            let n = IdlePixelPlus.getVarOrDefault(bag, 0, "int");
            if(alt) {
                n--;
            }
            if(n > 0) {
                IdlePixelPlus.sendMessage(`OPEN_GATHERING_LOOT=${bag.replace("gathering_loot_bag_", "")}~${n}`);
            }
        }

        initQuickGather() {
            LOOT_BAGS.forEach(item => {
              $(`itembox[data-item="${item}"]`).on("contextmenu", event => {
                  if(this.getConfig("quickGatherRightClickEnabled")) {
                      const primary = this.isPrimaryAction(event);
                      const alt = this.isAltAction(event);
                      if(primary || alt) {
                          this.quickGather(item, !primary);
                          event.stopPropagation();
                          event.preventDefault();
                          return false;
                      }
                  }
                  return true;
              });
            });
        }

        quickBrew(potion) {
            IdlePixelPlus.sendMessage(`BREW=${potion}~1`);
        }

        initQuickBrew() {
            $("#brewing-table tbody tr[data-brewing-item]").each(function() {
                const el = $(this);
                const potion = el.attr("data-brewing-item");
                if(!potion) {
                    return;
                }
                el.find("td:nth-child(4)").append(`
	            <div class="slapchop-quickbrew-button"
		            onclick="event.stopPropagation(); IdlePixelPlus.plugins.slapchop.quickBrew('${potion}')"">Quick Brew 1</div>
	            `);
            });
        }

        quickBurn(item, alt) {
            let n = IdlePixelPlus.getVarOrDefault(item, 0, "int");
            if(alt) {
                n--;
            }
            if(n > 0) {
                IdlePixelPlus.sendMessage(`ADD_HEAT=${item}~${n}`);
            }
        }

        initQuickBurn() {
            LOGS.forEach(item => {
              $(`itembox[data-item="${item}"]`).on("contextmenu", event => {
                  if(this.getConfig("quickBurnRightClickEnabled")) {
                      const primary = this.isPrimaryAction(event);
                      const alt = this.isAltAction(event);
                      if(primary || alt) {
                          this.quickBurn(item, !primary);
                          event.stopPropagation();
                          event.preventDefault();
                          return false;
                      }
                  }
                  return true;
              });
            });
        }

        maxCraftableArrows(feather) {
            const data = FEATHER2ARROW[feather];
            if(!data) return 0;

            let max = Number.MAX_SAFE_INTEGER;
            Object.keys(data.required).forEach(item => {
                const needed = data.required[item];
                const owned = IdlePixelPlus.getVarOrDefault(item, 0, "int");
                const craftable = Math.floor(owned/needed);
                max = Math.min(max, craftable);
            });
            return max;
        }

        quickFeather2Arrow(item, alt) {
            let n = this.maxCraftableArrows(item);
            if(n > 0) {
                IdlePixelPlus.sendMessage(`CRAFT=${FEATHER2ARROW[item].craft}~${n}`);
            }
        }

        initQuickFeather2Arrow() {
            Object.keys(FEATHER2ARROW).forEach(item => {
              $(`itembox[data-item="${item}"]`).on("contextmenu", event => {
                  if(this.getConfig("quickCraftArrowRightClickEnabled")) {
                      const primary = this.isPrimaryAction(event);
                      const alt = this.isAltAction(event);
                      if(primary || alt) {
                          this.quickFeather2Arrow(item, !primary);
                          event.stopPropagation();
                          event.preventDefault();
                          return false;
                      }
                  }
                  return true;
              });
            });
        }

        initQuickChop() {
            $("#panel-woodcutting itembox").first().before(`
	        <itembox id="slapchop-lumberjack" class="shadow hover" data-item="slapchop_lumberjack" onclick="IdlePixelPlus.plugins.slapchop.quickChop()">
        		<div class="center mt-1"><img width="50" height="50" src=""></div>
        		<div class="center mt-2">Chop</div>
        	</itembox>
            `);
        }

        quickChop() {
            for(let i = 1; i <= 5; i++) {
                let status = IdlePixelPlus.getVarOrDefault("tree_stage_"+i, 0, "int");
                if(status == 4) {
                    IdlePixelPlus.sendMessage("CHOP_TREE="+i);
                }
            }
        }

        initQuickHarvest() {
            $("#panel-farming itembox").first().before(`
	        <itembox id="slapchop-bob" class="shadow hover" data-item="slapchop_bob" onclick="IdlePixelPlus.plugins.slapchop.quickHarvest()">
        		<div class="center mt-1"><img width="50" height="50" src=""></div>
        		<div class="center mt-2">Harvest</div>
        	</itembox>
            `);
        }

        quickHarvest() {
            for(let i = 1; i <= 5; i++) {
                let status = IdlePixelPlus.getVarOrDefault("farm_stage_"+i, 0, "int");
                if(status == 4) {
                    IdlePixelPlus.sendMessage("CLICKS_PLOT="+i);
                }
            }
        }

        initQuickGrind() {
            GRINDABLE.forEach(item => {
                console.log(`#panel-invention itembox[data-item="${item}"]`);
                $(`#panel-invention itembox[data-item="${item}"]`).on("contextmenu", event => {
                    console.log("initQuickGrind", event);
                    console.log("quickGrind:rightclick:config", this.getConfig("quickGrindRightClickEnabled"));
                    if(this.getConfig("quickGrindRightClickEnabled")) {
                        const primary = this.isPrimaryAction(event);
                        const alt = this.isAltAction(event);
                        console.log("quickGrind:rightclick", {primary, alt});
                        if(primary || alt) {
                            this.quickGrind(item, !primary);
                            event.stopPropagation();
                            event.preventDefault();
                            return false;
                        }
                    }
                    return true;
                });
            });
        }

        quickGrind(item, alt) {
            console.log("quickGrind", {item, alt});
            let n = IdlePixelPlus.getVarOrDefault(item, 0, "int");
            if(n > 0) {
                IdlePixelPlus.sendMessage(`GRIND=${item}`);
            }
        }

        onLogin() {
            this.initStyles();
            this.initQuickFight();
            this.initQuickSmelt();
            this.initQuickCook();
            this.initQuickEat();
            this.initQuickPlant();
            this.initQuickBones();
            this.initQuickPotions();
            this.initQuickBoat();
            this.initQuickNeedle();
            this.initQuickBrew();
            this.initQuickGather();
            this.initQuickBurn();
            this.initQuickFeather2Arrow();
            this.initQuickFoundry();
            this.initQuickChop();
            this.initQuickHarvest();
            this.initQuickGrind();

            this.updateQuickFight();
            this.updateQuickSmelt();
            this.updateQuickFoundry();

            this.onConfigsChanged()
        }

        onVariableSet(key, valueBefore, valueAfter) {
            if(["fight_points", "energy", "volcano_unlocked", "northern_field_unlocked"].includes(key)) {
                this.updateQuickFight();
            }
            if([...SMELTABLES, "oil", "charcoal", "stone_furnace", "bronze_furnace", "iron_furnace", "silver_furnace", "gold_furnace"].includes(key)) {
                this.updateQuickSmelt();
            }
            if([...LOGS, "oil", "foundry_amount"].includes(key)) {
                this.updateQuickFoundry();
            }
        }

    }

    const plugin = new SlapChopPlugin();
    IdlePixelPlus.registerPlugin(plugin);

})();