diff --git a/mod_examples/buildings_have_cost.js b/mod_examples/buildings_have_cost.js new file mode 100644 index 00000000..41b7ad86 --- /dev/null +++ b/mod_examples/buildings_have_cost.js @@ -0,0 +1,102 @@ +/** + * This shows how to patch existing methods by making belts have a cost + */ +registerMod(() => { + return class ModImpl extends shapez.Mod { + constructor(app, modLoader) { + super( + app, + { + website: "https://tobspr.io", + author: "tobspr", + name: "Mod Example: Patch Methods", + version: "1", + id: "patch-methods", + description: + "Shows how to patch existing methods to change the game by making the belts cost shapes", + }, + modLoader + ); + } + + init() { + // Define our currency + const CURRENCY = "CyCyCyCy:--------:CuCuCuCu"; + + // Make sure the currency is always pinned + this.modInterface.runAfterMethod(shapez.HUDPinnedShapes, "rerenderFull", function () { + this.internalPinShape({ + key: CURRENCY, + canUnpin: false, + className: "currency", + }); + }); + + // Style it + this.modInterface.registerCss(` + #ingame_HUD_PinnedShapes .shape.currency::after { + content: " "; + position: absolute; + display: inline-block; + width: $scaled(8px); + height: $scaled(8px); + top: $scaled(4px); + left: $scaled(-7px); + background: url('${RESOURCES["currency.png"]}') center center / contain no-repeat; + } + + .currencyIcon { + display: inline-block; + vertical-align: middle; + background: url('${RESOURCES["currency.png"]}') center center / contain no-repeat; + } + + .currencyIcon.small { + width: $scaled(11px); + height: $scaled(11px); + } + `); + + // Make the player start with some currency + this.modInterface.runAfterMethod(shapez.GameCore, "initNewGame", function () { + console.log("Giving player start currency"); + this.root.hubGoals.storedShapes[CURRENCY] = 100; + }); + + // Make belts have a cost + this.modInterface.replaceMethod(shapez.MetaBeltBuilding, "getAdditionalStatistics", function ( + $original, + [root, variant] + ) { + const oldStats = $original(root, variant); + oldStats.push(["Cost", "1 x "]); + return oldStats; + }); + + // Only allow placing an entity when there is enough currency + this.modInterface.replaceMethod(shapez.GameLogic, "checkCanPlaceEntity", function ( + $original, + [entity, offset] + ) { + const storedCurrency = this.root.hubGoals.storedShapes[CURRENCY] || 0; + return storedCurrency > 0 && $original(entity, offset); + }); + + // Take shapes when placing a building + this.modInterface.replaceMethod(shapez.GameLogic, "tryPlaceBuilding", function ($original, args) { + const result = $original(...args); + if (result && result.components.Belt) { + console.log("BELT PLACED"); + + this.root.hubGoals.storedShapes[CURRENCY]--; + } + return result; + }); + } + }; +}); + +const RESOURCES = { + "currency.png": + "", +}; diff --git a/src/js/game/hud/parts/HUDPuzzleNextPuzzle.js b/src/js/game/hud/parts/next_puzzle.js similarity index 100% rename from src/js/game/hud/parts/HUDPuzzleNextPuzzle.js rename to src/js/game/hud/parts/next_puzzle.js diff --git a/src/js/game/modes/puzzle_play.js b/src/js/game/modes/puzzle_play.js index fc9a8f11..19957eee 100644 --- a/src/js/game/modes/puzzle_play.js +++ b/src/js/game/modes/puzzle_play.js @@ -30,7 +30,7 @@ import { HUDPuzzlePlaySettings } from "../hud/parts/puzzle_play_settings"; import { MetaBlockBuilding } from "../buildings/block"; import { MetaBuilding } from "../meta_building"; import { gMetaBuildingRegistry } from "../../core/global_registries"; -import { HUDPuzzleNextPuzzle } from "../hud/parts/HUDPuzzleNextPuzzle"; +import { HUDPuzzleNextPuzzle } from "../hud/parts/next_puzzle"; const logger = createLogger("puzzle-play"); const copy = require("clipboard-copy"); diff --git a/src/js/mods/mod_interface.js b/src/js/mods/mod_interface.js index 895d64c7..784e3d9d 100644 --- a/src/js/mods/mod_interface.js +++ b/src/js/mods/mod_interface.js @@ -36,6 +36,10 @@ export class ModInterface { } registerCss(cssString) { + // Preprocess css + cssString = cssString.replace(/\$scaled\(([^\)]*)\)/gim, (substr, expression) => { + return "calc((" + expression + ") * var(--ui-scale))"; + }); const element = document.createElement("style"); element.textContent = cssString; document.head.appendChild(element); @@ -360,6 +364,26 @@ export class ModInterface { * Patches a method on a given object */ replaceMethod(classHandle, methodName, override) { - classHandle.prototype[methodName] = override; + const oldMethod = classHandle.prototype[methodName]; + classHandle.prototype[methodName] = function () { + return override.call(this, oldMethod.bind(this), arguments); + }; + } + + runBeforeMethod(classHandle, methodName, executeBefore) { + const oldHandle = classHandle.prototype[methodName]; + classHandle.prototype[methodName] = function () { + executeBefore.apply(this, arguments); + return oldHandle.apply(this, arguments); + }; + } + + runAfterMethod(classHandle, methodName, executeAfter) { + const oldHandle = classHandle.prototype[methodName]; + classHandle.prototype[methodName] = function () { + const returnValue = oldHandle.apply(this, arguments); + executeAfter.apply(this, arguments); + return returnValue; + }; } } diff --git a/src/js/mods/modloader.js b/src/js/mods/modloader.js index 715a0e33..301b5ec7 100644 --- a/src/js/mods/modloader.js +++ b/src/js/mods/modloader.js @@ -84,12 +84,21 @@ export class ModLoader { mods = await ipcRenderer.invoke("get-mods"); } if (G_IS_DEV && globalConfig.debug.externalModUrl) { - const mod = await ( - await fetch(globalConfig.debug.externalModUrl, { - method: "GET", - }) - ).text(); - mods.push(mod); + const response = await fetch(globalConfig.debug.externalModUrl, { + method: "GET", + }); + if (response.status !== 200) { + throw new Error( + "Failed to load " + + globalConfig.debug.externalModUrl + + ": " + + response.status + + " " + + response.statusText + ); + } + + mods.push(await response.text()); } mods.forEach(modCode => {