From c005d8e41c744077a6b482df5230700547e3b5d1 Mon Sep 17 00:00:00 2001
From: radmanplays <95340057+radmanplays@users.noreply.github.com>
Date: Thu, 26 Sep 2024 08:13:31 +0330
Subject: [PATCH 01/41] Update modgui.js
---
modgui.js | 3 ++-
1 file changed, 2 insertions(+), 1 deletion(-)
diff --git a/modgui.js b/modgui.js
index c6fb267..45bae37 100644
--- a/modgui.js
+++ b/modgui.js
@@ -8,7 +8,8 @@
"https://github.com/EaglerForge",
"hey you should check out https://github.com/ZXMushroom63/scratch-gui",
"99% of people stop gambling before they win big.",
- "Now with free estradiol!"
+ "Now with free estradiol!",
+ "Now with H.I.V (Hyper Injected Virtual-debugger)"
];
var gui = `
From bc662a14d423d24c0c2adfc94244af6d792c93bd Mon Sep 17 00:00:00 2001
From: radmanplays <95340057+radmanplays@users.noreply.github.com>
Date: Thu, 26 Sep 2024 08:13:43 +0330
Subject: [PATCH 02/41] Update modgui.injector.js
---
modgui.injector.js | 3 ++-
1 file changed, 2 insertions(+), 1 deletion(-)
diff --git a/modgui.injector.js b/modgui.injector.js
index 4e466e8..54f537b 100644
--- a/modgui.injector.js
+++ b/modgui.injector.js
@@ -8,7 +8,8 @@ globalThis.modapi_guikit = `// ModAPI GUI made by TheIdiotPlays
"https://github.com/EaglerForge",
"hey you should check out https://github.com/ZXMushroom63/scratch-gui",
"99% of people stop gambling before they win big.",
- "Now with free estradiol!"
+ "Now with free estradiol!",
+ "Now with H.I.V (Hyper Injected Virtual-debugger)"
];
var gui = \`
From 7d7c06475bd68136fc38ba0f04ccf6470e2096bd Mon Sep 17 00:00:00 2001
From: ZXMushroom63
Date: Thu, 26 Sep 2024 12:50:21 +0800
Subject: [PATCH 03/41] test
---
examplemods/AsyncSink.js | 21 ++++++++++++++++++++-
examplemods/npcspawner.js | 18 ++++++++++++++++++
2 files changed, 38 insertions(+), 1 deletion(-)
diff --git a/examplemods/AsyncSink.js b/examplemods/AsyncSink.js
index bb9aab2..9e2cd08 100644
--- a/examplemods/AsyncSink.js
+++ b/examplemods/AsyncSink.js
@@ -76,7 +76,11 @@
}
return wrap(AsyncSink.getFile(ModAPI.util.ustr(args[1])));
}
-
+ var ev = {method: "read", file: ModAPI.util.ustr(args[1]), shim: false, shimOutput: new ArrayBuffer()};
+ AsyncSink.MIDDLEWARE.forEach((fn)=>{fn(ev)});
+ if (ev.shim) {
+ return wrap(ev.shimOutput);
+ }
return originalReadWholeFile.apply(this, args);
};
@@ -92,6 +96,11 @@
AsyncSink.setFile(ModAPI.util.ustr(args[1]), args[2]);
return booleanResult(true);
}
+ var ev = {method: "write", file: ModAPI.util.ustr(args[1]), data: args[2], shim: false, shimOutput: true};
+ AsyncSink.MIDDLEWARE.forEach((fn)=>{fn(ev)});
+ if (ev.shim) {
+ return booleanResult(ev.shimOutput);
+ }
return originalWriteWholeFile.apply(this, args);
};
@@ -107,6 +116,11 @@
AsyncSink.deleteFile(ModAPI.util.ustr(args[1]));
return booleanResult(true);
}
+ var ev = {method: "delete", file: ModAPI.util.ustr(args[1]), shim: false, shimOutput: true};
+ AsyncSink.MIDDLEWARE.forEach((fn)=>{fn(ev)});
+ if (ev.shim) {
+ return booleanResult(ev.shimOutput);
+ }
return originalDeleteFile.apply(this, args);
};
@@ -122,6 +136,11 @@
var result = AsyncSink.fileExists(ModAPI.util.ustr(args[1]));
return booleanResult(result);
}
+ var ev = {method: "exists", file: ModAPI.util.ustr(args[1]), shim: false, shimOutput: true};
+ AsyncSink.MIDDLEWARE.forEach((fn)=>{fn(ev)});
+ if (ev.shim) {
+ return booleanResult(ev.shimOutput);
+ }
return originalFileExists.apply(this, args);
};
globalThis.AsyncSink = AsyncSink;
diff --git a/examplemods/npcspawner.js b/examplemods/npcspawner.js
index a07fbb2..f17bb07 100644
--- a/examplemods/npcspawner.js
+++ b/examplemods/npcspawner.js
@@ -1,5 +1,20 @@
(() => {
PluginAPI.dedicatedServer.appendCode(function () {
+ var ready = false;
+ var killFS = false;
+ function setup_filesystem_middleware() {
+ if (!ready) {
+ AsyncSink.MIDDLEWARE.push((ev)=>{
+ if (killFS) {
+ ev.shim = true;
+ if (typeof ev.shimOutput === "boolean") {
+ ev.shimOutput = true;
+ }
+ }
+ });
+ ready = true;
+ }
+ }
PluginAPI.addEventListener("processcommand", (event) => {
if (!ModAPI.reflect.getClassById("net.minecraft.entity.player.EntityPlayerMP").instanceOf(event.sender.getRef())) { return; }
@@ -7,6 +22,7 @@
if (!globalThis.AsyncSink) {
return alert("NPC Spawner relies on the AsyncSink library.");
}
+ setup_filesystem_middleware();
const world = event.sender.getServerForPlayer();
const senderPos = event.sender.getPosition();
@@ -23,6 +39,7 @@
const playerInteractionManager = PlayerInteractionManagerClass.constructors[0](world.getRef());
// Get the EntityPlayerMP class to spawn the fake player
+ killFS = true;
AsyncSink.startDebuggingFS();
const EntityPlayerMPClass = ModAPI.reflect.getClassById("net.minecraft.entity.player.EntityPlayerMP");
var worldNameProp = ModAPI.util.getNearestProperty(ModAPI.server.getRef(), "$worldName");
@@ -31,6 +48,7 @@
const fakePlayer = EntityPlayerMPClass.constructors[0](
ModAPI.server.getRef(), world.getRef(), fakeProfile, playerInteractionManager
);
+ killFS = false;
// Set the fake player position to be near the command sender
fakePlayer.setPosition(senderPos.$getX(), senderPos.$getY(), senderPos.$getZ());
From 6fa256ae0e6f1357841ff5751c127fee3a8de91c Mon Sep 17 00:00:00 2001
From: ZXMushroom63
Date: Thu, 26 Sep 2024 12:54:27 +0800
Subject: [PATCH 04/41] fix bug
---
examplemods/npcspawner.js | 3 ++-
1 file changed, 2 insertions(+), 1 deletion(-)
diff --git a/examplemods/npcspawner.js b/examplemods/npcspawner.js
index f17bb07..bbf7c75 100644
--- a/examplemods/npcspawner.js
+++ b/examplemods/npcspawner.js
@@ -51,7 +51,8 @@
killFS = false;
// Set the fake player position to be near the command sender
- fakePlayer.setPosition(senderPos.$getX(), senderPos.$getY(), senderPos.$getZ());
+ console.log(senderPos);
+ fakePlayer.setPosition(senderPos.getX(), senderPos.getY(), senderPos.getZ());
// Add the fake player to the world
world.addEntityToWorld(fakePlayer.getEntityId(), fakePlayer);
From f45b065bc3c91bab4171df272dda42cfa9cca47f Mon Sep 17 00:00:00 2001
From: radmanplays <95340057+radmanplays@users.noreply.github.com>
Date: Thu, 26 Sep 2024 08:32:31 +0330
Subject: [PATCH 05/41] Create slippery.js
---
examplemods/slippery.js | 23 +++++++++++++++++++++++
1 file changed, 23 insertions(+)
create mode 100644 examplemods/slippery.js
diff --git a/examplemods/slippery.js b/examplemods/slippery.js
new file mode 100644
index 0000000..9b6c37b
--- /dev/null
+++ b/examplemods/slippery.js
@@ -0,0 +1,23 @@
+ModAPI.meta.title("Slippery mod");
+ModAPI.meta.credits("By radmanplays");
+ModAPI.meta.icon("");
+ModAPI.meta.description("make everything ice!");
+
+ModAPI.dedicatedServer.appendCode(()=>{
+ ModAPI.addEventListener("serverstart", ()=>{
+ var blockKeys = Object.keys(ModAPI.blocks);
+ blockKeys.forEach(key=>{
+ if(ModAPI?.blocks?.[key]?.slipperiness) {// TeaVM likes to add metadata properties which are `null` or `undefined`
+ ModAPI.blocks[key].slipperiness = 0.98; //Ice slipperiness value.
+ ModAPI.blocks[key].reload();// load the value
+ }
+ });
+ });
+});
+var blockKeys = Object.keys(ModAPI.blocks);
+blockKeys.forEach(key=>{
+ if(ModAPI?.blocks?.[key]?.slipperiness) {// TeaVM likes to add metadata properties which are `null` or `undefined`
+ ModAPI.blocks[key].slipperiness = 0.98; //Ice slipperiness value.
+ ModAPI.blocks[key].reload();// load the value
+ }
+});
From b77e77fa6617d0f095cbd4afab7b71374533870d Mon Sep 17 00:00:00 2001
From: ZXMushroom63
Date: Thu, 26 Sep 2024 13:02:35 +0800
Subject: [PATCH 06/41] AsyncSink icon
---
examplemods/AsyncSink.js | 4 ++++
1 file changed, 4 insertions(+)
diff --git a/examplemods/AsyncSink.js b/examplemods/AsyncSink.js
index 9e2cd08..c57baed 100644
--- a/examplemods/AsyncSink.js
+++ b/examplemods/AsyncSink.js
@@ -1,3 +1,7 @@
+ModAPI.meta.title("AsyncSink");
+ModAPI.meta.description("Library for patching and hooking into asynchronous filesystem requests for EaglercraftX.");
+ModAPI.meta.icon("");
+ModAPI.meta.credits("By ZXMushroom63");
(function AsyncSinkFn() {
//AsyncSink is a plugin to debug and override asynchronous methods in EaglercraftX
function runtimeComponent() {
From 0589b2c4fb6536c0bfc39bb20da021769ff3c075 Mon Sep 17 00:00:00 2001
From: ZXMushroom63
Date: Thu, 26 Sep 2024 13:04:02 +0800
Subject: [PATCH 07/41] npcspawner fix 153
---
examplemods/npcspawner.js | 6 +++---
1 file changed, 3 insertions(+), 3 deletions(-)
diff --git a/examplemods/npcspawner.js b/examplemods/npcspawner.js
index bbf7c75..f0b8c33 100644
--- a/examplemods/npcspawner.js
+++ b/examplemods/npcspawner.js
@@ -45,9 +45,9 @@
var worldNameProp = ModAPI.util.getNearestProperty(ModAPI.server.getRef(), "$worldName");
var worldName = ModAPI.server.getRef()[worldNameProp];
console.log(ModAPI.util.ustr(worldName));
- const fakePlayer = EntityPlayerMPClass.constructors[0](
+ const fakePlayer = ModAPI.util.wrap(EntityPlayerMPClass.constructors[0](
ModAPI.server.getRef(), world.getRef(), fakeProfile, playerInteractionManager
- );
+ ));
killFS = false;
// Set the fake player position to be near the command sender
@@ -55,7 +55,7 @@
fakePlayer.setPosition(senderPos.getX(), senderPos.getY(), senderPos.getZ());
// Add the fake player to the world
- world.addEntityToWorld(fakePlayer.getEntityId(), fakePlayer);
+ world.addEntityToWorld(fakePlayer.getEntityId(), fakePlayer.getRef());
// Notify the player that the fake player has been spawned
const ChatComponentTextClass = ModAPI.reflect.getClassById("net.minecraft.util.ChatComponentText");
From b8af5ad3563cdc66779c51ffaefa72a7dff4cebc Mon Sep 17 00:00:00 2001
From: ZXMushroom63
Date: Thu, 26 Sep 2024 13:04:45 +0800
Subject: [PATCH 08/41] Fix 154
---
examplemods/npcspawner.js | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/examplemods/npcspawner.js b/examplemods/npcspawner.js
index f0b8c33..0e003e3 100644
--- a/examplemods/npcspawner.js
+++ b/examplemods/npcspawner.js
@@ -20,7 +20,7 @@
if (event.command.toLowerCase().startsWith("/spawnnpc")) {
if (!globalThis.AsyncSink) {
- return alert("NPC Spawner relies on the AsyncSink library.");
+ return console.log("NPC Spawner relies on the AsyncSink library.");
}
setup_filesystem_middleware();
const world = event.sender.getServerForPlayer();
From 4d69375e3550e1a34db015402e8491ca28df621c Mon Sep 17 00:00:00 2001
From: ZXMushroom63
Date: Thu, 26 Sep 2024 13:05:05 +0800
Subject: [PATCH 09/41] use error instead of log
---
examplemods/npcspawner.js | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/examplemods/npcspawner.js b/examplemods/npcspawner.js
index 0e003e3..79cd67b 100644
--- a/examplemods/npcspawner.js
+++ b/examplemods/npcspawner.js
@@ -20,7 +20,7 @@
if (event.command.toLowerCase().startsWith("/spawnnpc")) {
if (!globalThis.AsyncSink) {
- return console.log("NPC Spawner relies on the AsyncSink library.");
+ return console.error("NPC Spawner relies on the AsyncSink library.");
}
setup_filesystem_middleware();
const world = event.sender.getServerForPlayer();
From d4a9919cd8ed971cfbeed505570472b4f16d107f Mon Sep 17 00:00:00 2001
From: ZXMushroom63
Date: Thu, 26 Sep 2024 13:10:45 +0800
Subject: [PATCH 10/41] fix bug 156
---
examplemods/npcspawner.js | 3 +--
1 file changed, 1 insertion(+), 2 deletions(-)
diff --git a/examplemods/npcspawner.js b/examplemods/npcspawner.js
index 79cd67b..19bef2c 100644
--- a/examplemods/npcspawner.js
+++ b/examplemods/npcspawner.js
@@ -40,7 +40,6 @@
// Get the EntityPlayerMP class to spawn the fake player
killFS = true;
- AsyncSink.startDebuggingFS();
const EntityPlayerMPClass = ModAPI.reflect.getClassById("net.minecraft.entity.player.EntityPlayerMP");
var worldNameProp = ModAPI.util.getNearestProperty(ModAPI.server.getRef(), "$worldName");
var worldName = ModAPI.server.getRef()[worldNameProp];
@@ -55,7 +54,7 @@
fakePlayer.setPosition(senderPos.getX(), senderPos.getY(), senderPos.getZ());
// Add the fake player to the world
- world.addEntityToWorld(fakePlayer.getEntityId(), fakePlayer.getRef());
+ world.spawnEntityInWorld(fakePlayer.getRef());
// Notify the player that the fake player has been spawned
const ChatComponentTextClass = ModAPI.reflect.getClassById("net.minecraft.util.ChatComponentText");
From 45e6a4b56fe052daf59a6bc51ee7c388e74381ad Mon Sep 17 00:00:00 2001
From: ZXMushroom63
Date: Thu, 26 Sep 2024 18:37:14 +0800
Subject: [PATCH 11/41] re-add simple vclip
---
examplemods/vclip.js | 17 +++++++++++++++++
1 file changed, 17 insertions(+)
create mode 100644 examplemods/vclip.js
diff --git a/examplemods/vclip.js b/examplemods/vclip.js
new file mode 100644
index 0000000..8582db3
--- /dev/null
+++ b/examplemods/vclip.js
@@ -0,0 +1,17 @@
+ModAPI.meta.title("Simple VClip Exploit");
+ModAPI.meta.description("Use .vclip to vertically phase through blocks.");
+ModAPI.meta.credits("By ZXMushroom63");
+ModAPI.require("player");
+ModAPI.addEventListener("sendchatmessage", (ev) => {
+ var msg = ev.message.toLowerCase();
+ if (msg.startsWith(".vclip")) {
+ ev.preventDefault();
+ var yOffset = 1;
+ if (msg.split(" ")[1]) {
+ yOffset = parseFloat(msg.split(" ")[1]) || 0;
+ }
+ ModAPI.player.setPosition(ModAPI.player.posX, ModAPI.player.posY
+ + yOffset, ModAPI.player.posZ);
+ ModAPI.displayToChat("[SimpleVClip] VClipped " + yOffset + " blocks.");
+ }
+});
\ No newline at end of file
From 93b62ec6117d3f1caed419bea4479939f8ed5df7 Mon Sep 17 00:00:00 2001
From: ZXMushroom63
Date: Thu, 26 Sep 2024 18:39:08 +0800
Subject: [PATCH 12/41] fix issues
---
examplemods/advanced_vclip.js | 2 +-
examplemods/vclip.js | 2 +-
2 files changed, 2 insertions(+), 2 deletions(-)
diff --git a/examplemods/advanced_vclip.js b/examplemods/advanced_vclip.js
index b6471c9..0c62f91 100644
--- a/examplemods/advanced_vclip.js
+++ b/examplemods/advanced_vclip.js
@@ -11,7 +11,7 @@ ModAPI.require("player");
ModAPI.addEventListener("sendchatmessage", (ev) => {
var msg = ev.message.toLowerCase();
if (msg.startsWith(".vclip")) {
- ev.preventDefault();
+ ev.preventDefault == true;
var args = msg.split(" ");
if (args.length != 2) {
diff --git a/examplemods/vclip.js b/examplemods/vclip.js
index 8582db3..a91ff79 100644
--- a/examplemods/vclip.js
+++ b/examplemods/vclip.js
@@ -5,7 +5,7 @@ ModAPI.require("player");
ModAPI.addEventListener("sendchatmessage", (ev) => {
var msg = ev.message.toLowerCase();
if (msg.startsWith(".vclip")) {
- ev.preventDefault();
+ ev.preventDefault = true;
var yOffset = 1;
if (msg.split(" ")[1]) {
yOffset = parseFloat(msg.split(" ")[1]) || 0;
From 6da933d82eb1b1bf981a4682dbc484c11fe64f65 Mon Sep 17 00:00:00 2001
From: ZXMushroom63
Date: Sat, 28 Sep 2024 16:42:16 +0800
Subject: [PATCH 13/41] WIP baritone port
---
docs/apidoc/utils.md | 14 ++++-
examplemods/astar.js | 117 +++++++++++++++++++++++++++++++++++
examplemods/threadtesting.js | 2 +-
postinit.js | 39 ++++++++++++
4 files changed, 170 insertions(+), 2 deletions(-)
create mode 100644 examplemods/astar.js
diff --git a/docs/apidoc/utils.md b/docs/apidoc/utils.md
index 8446a71..fa80f21 100644
--- a/docs/apidoc/utils.md
+++ b/docs/apidoc/utils.md
@@ -43,4 +43,16 @@ Methods:
- `ModAPI.util.wrap(obj: Object) : object`
- Returns a wrapper around native java objects, removing prefixes and fixing method outputs.
- `ModAPI.util.getNearestProperty(object: Object, property: string) : string`
- - Finds the nearest property name to the one you specify (suffix based). This is used to mitigate teavm adding random suffixes to properties.
\ No newline at end of file
+ - Finds the nearest property name to the one you specify (suffix based). This is used to mitigate teavm adding random suffixes to properties.
+- `ModAPI.util.modifyFunction(fn: Function, patcherFunction: Function) : string`
+ - Returns a modifies version of a function, where the patcher function can be used to modify the contents of a function. Example:
+ ```javascript
+ function add(a, b) {
+ return a + b;
+ }
+ var multiply = ModAPI.util.modifyFunction(add, (code)=>{
+ return code.replaceAll("a + b", "a * b");
+ });
+ console.log(multiply(2, 3));
+ //Logs 6
+ ```
\ No newline at end of file
diff --git a/examplemods/astar.js b/examplemods/astar.js
new file mode 100644
index 0000000..93ef7cf
--- /dev/null
+++ b/examplemods/astar.js
@@ -0,0 +1,117 @@
+(function AStarPathfinding() {
+ ModAPI.meta.title("A* Pathfinding Bot");
+ ModAPI.meta.description("Use #move to instruct the bot to move somewhere. Use #stop to terminate the job.");
+ ModAPI.meta.credits("By ZXMushroom63");
+
+ ModAPI.require("player");
+ ModAPI.require("world");
+
+ var blockBreakCostMultiplier = 2;
+ const costMap = Object.fromEntries(Object.keys(ModAPI.blocks).flatMap(x => {
+ ModAPI.blocks[x].readableId = x;
+ return [x, (Math.floor(ModAPI.blocks[x].blockHardness * 10 * blockBreakCostMultiplier) + 1) || 99999]; //Block hardness is in decimals, unbreakable blocks are negative one, and base movement cost is 1. -1 + 1 = 0, and 0 || 99999 is 99999
+ }));
+
+ var makeBlockPos = ModAPI.reflect.getClassById("net.minecraft.util.BlockPos").constructors.find(x=>x.length === 3);
+
+ function shouldPause(x, y, z) {
+ return !ModAPI.world.isAreaLoaded0(makeBlockPos(x, y, z), 2);
+ }
+
+ class APNode {
+ constructor(x, y, z, g, h, parent = null) {
+ this.x = x;
+ this.y = y;
+ this.z = z;
+ this.g = g;
+ this.h = h;
+ this.f = g + h;
+ this.parent = parent;
+ }
+ }
+
+ function heuristic(a, b) {
+ return Math.abs(a.x - b.x) + Math.abs(a.y - b.y) + Math.abs(a.z - b.z);
+ }
+
+ function getNeighbors(node) {
+ const neighbors = [];
+ const directions = [
+ [1, 0, 0], [-1, 0, 0],
+ [0, 1, 0], [0, -1, 0],
+ [0, 0, 1], [0, 0, -1]
+ ];
+
+ for (const [dx, dy, dz] of directions) {
+ const x = node.x + dx;
+ const y = node.y + dy;
+ const z = node.z + dz;
+
+ if (ModAPI.world.isBlockLoaded(makeBlockPos(Math.round(x), Math.round(y), Math.round(z)))) {
+ neighbors.push(new APNode(x, y, z, 0, 0));
+ }
+ }
+
+ return neighbors;
+ }
+
+ function lookupCost(x, y, z) {
+ var block = ModAPI.world.getBlockState(makeBlockPos(Math.round(x), Math.round(y), Math.round(z))).block;
+ return costMap[block.readableId];
+ }
+
+ function* aStar(start, goal) {
+ const openSet = [];
+ const closedSet = new Set();
+ openSet.push(start);
+
+ while (openSet.length > 0) {
+ let current = openSet.reduce((prev, curr) => (prev.f < curr.f ? prev : curr));
+
+ if (current.x === goal.x && current.y === goal.y && current.z === goal.z) {
+ const path = [];
+ while (current) {
+ path.push([current.x, current.y, current.z]);
+ current = current.parent;
+ }
+ yield* path.reverse();
+ return;
+ }
+
+ openSet.splice(openSet.indexOf(current), 1);
+ closedSet.add(`${current.x},${current.y},${current.z}`);
+
+ for (const neighbor of getNeighbors(current)) {
+ if (closedSet.has(`${neighbor.x},${neighbor.y},${neighbor.z}`)) {
+ continue;
+ }
+
+ const tentativeG = current.g + lookupCost(neighbor.x, neighbor.y, neighbor.z);
+
+ if (!openSet.some(node => node.x === neighbor.x && node.y === neighbor.y && node.z === neighbor.z)) {
+ neighbor.g = tentativeG;
+ neighbor.h = heuristic(neighbor, goal);
+ neighbor.f = neighbor.g + neighbor.h;
+ neighbor.parent = current;
+ openSet.push(neighbor);
+ } else {
+ const existingNode = openSet.find(node => node.x === neighbor.x && node.y === neighbor.y && node.z === neighbor.z);
+ if (tentativeG < existingNode.g) {
+ existingNode.g = tentativeG;
+ existingNode.f = existingNode.g + existingNode.h;
+ existingNode.parent = current;
+ }
+ }
+ }
+
+ yield [current.x, current.y, current.z];
+ }
+
+ return [];
+ }
+
+ const start = new APNode(0, 0, 0, 0, 0);
+ const goal = new APNode(2, 2, 2, 0, 0);
+
+ const pathGenerator = aStar(start, goal);
+})();
\ No newline at end of file
diff --git a/examplemods/threadtesting.js b/examplemods/threadtesting.js
index 1b0a633..35006aa 100644
--- a/examplemods/threadtesting.js
+++ b/examplemods/threadtesting.js
@@ -1,4 +1,4 @@
-//SUCCESS - While there is no TeaVM thread actively running, I am able to run an asyncronous function, and get a result.
+//SUCCESS - While there is no TeaVM thread actively running, I am able to run an asynchronous function, and get a result.
ModAPI.hooks._teavm.$rt_startThread(() => {
return ModAPI.hooks.methods.nlevi_PlatformRuntime_downloadRemoteURI(ModAPI.util.str("data:text/plain,hi"))
}, function (...args) { console.log(this, args) })
diff --git a/postinit.js b/postinit.js
index e5f7e0d..5801eb0 100644
--- a/postinit.js
+++ b/postinit.js
@@ -584,6 +584,25 @@ globalThis.modapi_postinit = "(" + (() => {
})[0] || null;
}
+ ModAPI.util.modifyFunction = function (fn, patcherFn) {
+ // Convert the original function to a string
+ let functionString = fn.toString();
+
+ // Extract the function body
+ let bodyStart = functionString.indexOf('{') + 1;
+ let bodyEnd = functionString.lastIndexOf('}');
+ let functionBody = functionString.substring(bodyStart, bodyEnd);
+
+ // Replace the function body with new instructions
+ let modifiedFunctionBody = patcherFn(functionBody) || 'return;';
+
+ // Create a new function with the same arguments and the modified body
+ let args = functionString.substring(functionString.indexOf('(') + 1, functionString.indexOf(')'));
+ let modifiedFunction = new Function(args, modifiedFunctionBody);
+
+ return modifiedFunction;
+ }
+
ModAPI.clickMouse = function () {
ModAPI.hooks.methods["nmc_Minecraft_clickMouse"](ModAPI.javaClient);
}
@@ -661,6 +680,26 @@ globalThis.modapi_postinit = "(" + (() => {
return sendChatMessage.apply(this, [$this, $message]);
}
+ ModAPI.events.newEvent("render", "client");
+ const renderMethodName = ModAPI.util.getMethodFromPackage("net.minecraft.client.renderer.EntityRenderer", "renderWorldPass");
+ const renderMethod = ModAPI.hooks.methods[renderMethodName];
+ ModAPI.hooks.methods[renderMethodName] = function ($this, $int_pass, $float_partialTicks, $long_finishTimeNano) {
+ var shouldRenderHand = $this.$renderHand;
+ $this.$renderHand = 0; //Rendering the hand clears the depth bit, which we don't want to do.
+ var out = renderMethod.apply(this, [$this, $int_pass, $float_partialTicks, $long_finishTimeNano]);
+ var data = {
+ partialTicks: $float_partialTicks
+ }
+ ModAPI.events.callEvent("render", data);
+ if (shouldRenderHand) {
+ ModAPI.hooks.methods.nlevo_GlStateManager_clear(256); //GL_DEPTH_BUFFER_BIT, found in class RealOpenGLEnums
+ ModAPI.hooks.methods.nmcr_EntityRenderer_renderHand($this, $float_partialTicks, $int_pass);
+ ModAPI.hooks.methods.nmcr_EntityRenderer_renderWorldDirections($this, $float_partialTicks);
+ }
+ $this.$renderHand = shouldRenderHand;
+ return out;
+ }
+
const ScaledResolutionConstructor = ModAPI.reflect.getClassByName("ScaledResolution").constructors[0];
ModAPI.events.newEvent("frame", "client");
const frameMethodName = ModAPI.util.getMethodFromPackage("net.minecraft.client.Minecraft", "runTick");
From 51f06875f369dd605c09bdc4ed469a890fa7d18b Mon Sep 17 00:00:00 2001
From: ZXMushroom63
Date: Sun, 29 Sep 2024 14:58:32 +0800
Subject: [PATCH 14/41] roadmap update
---
examplemods/astar.js | 1 -
roadmap/index.html | 7 ++++---
2 files changed, 4 insertions(+), 4 deletions(-)
diff --git a/examplemods/astar.js b/examplemods/astar.js
index 93ef7cf..dc90951 100644
--- a/examplemods/astar.js
+++ b/examplemods/astar.js
@@ -51,7 +51,6 @@
neighbors.push(new APNode(x, y, z, 0, 0));
}
}
-
return neighbors;
}
diff --git a/roadmap/index.html b/roadmap/index.html
index acc2f89..6b735a1 100644
--- a/roadmap/index.html
+++ b/roadmap/index.html
@@ -10,11 +10,12 @@
EaglerForgeInjector Roadmap
- Add makeItemStack to LCI [done]
+ Add custom itemstack meta to LCI [done]
Fix blocklook.js [todo]
Fix setblocktest.js [done]
- `ModAPI.asyncMode(10)# and `ModAPI.exitAsync()# to allow using `await# on things. I will have to create a new `TeaVMThread#, and create a `ModAPI.util.async(fn)# method to promisify. [wip]
- Full backwards compatibility for mods [todo]
+ Async toolkit library [done]
+ Object.keys() and `in` operator on proxy objects [todo]
+ Add textures to LCI. [todo]