From 7f1fae38b0fd967d4a85cc5aec89af71c6e1469a Mon Sep 17 00:00:00 2001
From: Awakened-Redstone <40528665+Awakened-Redstone@users.noreply.github.com>
Date: Sun, 7 Jul 2024 01:34:15 -0300
Subject: [PATCH] chore[noci]: Improve island teleport, add client testmod
---
README.md | 20 ++
build.gradle.kts | 85 +++++++--
gradle.properties | 4 +-
.../neoskies/NeoSkiesClient.java | 10 +
.../resources/neoskies.client.mixins.json | 10 +
.../neoskies/command/NeoSkiesCommands.java | 2 +-
.../command/admin/IslandDataCommand.java | 82 ++++----
.../neoskies/command/island/HomeCommand.java | 18 --
.../neoskies/command/island/VisitCommand.java | 26 ++-
.../neoskies/config/IslandRankingConfig.java | 7 -
.../awakenedredstone/neoskies/logic/Hub.java | 7 +-
.../neoskies/logic/Island.java | 68 ++++---
.../neoskies/logic/IslandStuck.java | 11 ++
.../registry/NeoSkiesIslandSettings.java | 4 +-
.../mixin/block/EndPortalBlockMixin.java | 36 ++--
.../neoskies/mixin/entity/EntityAccessor.java | 10 +
.../mixin/entity/PlayerManagerMixin.java | 6 +-
.../entity/ServerPlayerEntityAccessor.java | 11 ++
.../mixin/entity/ServerPlayerEntityMixin.java | 4 +-
.../neoskies/util/Worlds.java | 39 ++++
.../resources/data/neoskies/lang/en_us.json | 2 +-
src/main/resources/fabric.mod.json | 10 +-
src/main/resources/neoskies.mixins.json | 2 +
src/testmod/resources/fabric.mod.json | 8 +-
.../test/DebugDearImGuiEntrypoint.java | 17 ++
.../neoskies/test/DebugImGuiRenderer.java | 176 ++++++++++++++++++
.../neoskies/test/NeoSkiesTestClientMain.java | 30 +++
src/testmodClient/resources/fabric.mod.json | 23 +++
.../neoskies.test.client.mixins.json | 11 ++
29 files changed, 584 insertions(+), 155 deletions(-)
create mode 100644 src/client/java/com/awakenedredstone/neoskies/NeoSkiesClient.java
create mode 100644 src/client/resources/neoskies.client.mixins.json
create mode 100644 src/main/java/com/awakenedredstone/neoskies/mixin/entity/EntityAccessor.java
create mode 100644 src/main/java/com/awakenedredstone/neoskies/mixin/entity/ServerPlayerEntityAccessor.java
create mode 100644 src/testmodClient/java/com/awakenedredstone/neoskies/test/DebugDearImGuiEntrypoint.java
create mode 100644 src/testmodClient/java/com/awakenedredstone/neoskies/test/DebugImGuiRenderer.java
create mode 100644 src/testmodClient/java/com/awakenedredstone/neoskies/test/NeoSkiesTestClientMain.java
create mode 100644 src/testmodClient/resources/fabric.mod.json
create mode 100644 src/testmodClient/resources/neoskies.test.client.mixins.json
diff --git a/README.md b/README.md
index 59a4ef3..ac0aecd 100644
--- a/README.md
+++ b/README.md
@@ -1,3 +1,23 @@
+[//]: # (A skyblock mod with customizable islands, advanced island protection and support for expansion mods.)
+
+[//]: # ()
+[//]: # (NeoSkies provides it's own API and registries for other mods to add content and features and integrate with the island system.)
+
+[//]: # ()
+[//]: # (Each island have their own settings, those settings allow the owner to decide how other players can interact with the island, such as allowing interaction with doors, redstone, containers, etc.)
+
+[//]: # ()
+[//]: # (The mod uses the Common Economy API for currency, each island has it's account, that is shared between all island members.)
+
+[//]: # ()
+[//]: # (The mod is still in development and will change over time.)
+
+[//]: # ()
+[//]: # (This mod is a fork of Skylands, and the current island system and island templates are (mostly) from the original mod.)
+
+[//]: # ()
+[//]: # (Please provide feedback on the [discord server](https://discord.gg/MTqsjwMpN2), it helps a lot to continue developing the mod)
+
## About
diff --git a/build.gradle.kts b/build.gradle.kts
index c627313..9e0f2f0 100644
--- a/build.gradle.kts
+++ b/build.gradle.kts
@@ -1,6 +1,7 @@
import java.io.File
import com.modrinth.minotaur.dependencies.ModDependency
+//region Setup
plugins {
id("signing")
id("maven-publish")
@@ -8,15 +9,16 @@ plugins {
id("com.modrinth.minotaur") version "2.+"
}
-val CHANGELOG: String =
- if (file("CHANGELOG.md").exists()) {
- file("CHANGELOG.md").readText()
- } else {
- "No changelog provided"
- }
+loom {
+ accessWidenerPath.set(file("src/main/resources/neoskies.accesswidener"))
+ splitEnvironmentSourceSets()
-fun file(path: String): File {
- return rootProject.file(path)
+ mods {
+ create("neoskies") {
+ sourceSet(sourceSets.main.get())
+ sourceSet(sourceSets.getByName("client"))
+ }
+ }
}
sourceSets {
@@ -30,37 +32,48 @@ sourceSets {
compileClasspath += sourceSets.main.get().compileClasspath
runtimeClasspath += sourceSets.main.get().runtimeClasspath
}
+
+ create("testmodClient") {
+ compileClasspath += main.get().compileClasspath
+ runtimeClasspath += main.get().runtimeClasspath
+
+ compileClasspath += sourceSets.getByName("client").compileClasspath
+ runtimeClasspath += sourceSets.getByName("client").runtimeClasspath
+
+ compileClasspath += sourceSets.getByName("testmod").compileClasspath
+ runtimeClasspath += sourceSets.getByName("testmod").runtimeClasspath
+ }
}
loom {
- accessWidenerPath.set(file("src/main/resources/neoskies.accesswidener"))
-
runs {
create("datagenServer") {
server()
- name = "Data Generation"
+ name("Data Generation")
vmArg("-Dfabric-api.datagen")
vmArg("-Dfabric-api.datagen.output-dir=${file("src/main/generated")}")
vmArg("-Dfabric-api.datagen.modid=neoskies")
-
- //ideConfigGenerated = true
- runDir = "build/datagen"
+ runDir("build/datagen")
+ ideConfigGenerated(true)
}
-
create("testmodServer") {
server()
- name = "Testmod Server"
+ name("Testmod Server")
+ ideConfigGenerated(true)
source(sourceSets.getByName("testmod"))
}
-
create("testmodClient") {
client()
- name = "Testmod Client"
+ name("Testmod Client")
runDir("run_client")
- source(sourceSets.getByName("testmod"))
+ //TODO: Generate dev world
+ programArgs("--quickPlaySingleplayer \"world\"")
+ ideConfigGenerated(true)
+ source(sourceSets.getByName("testmodClient"))
}
}
}
+//endregion
var archivesBaseName: String = property("archives_base_name").toString()
base {
@@ -76,6 +89,7 @@ repositories {
maven("https://oss.sonatype.org/content/repositories/snapshots")
maven("https://maven.ladysnake.org/releases")
maven("https://maven.isxander.dev/releases")
+ maven("https://maven.deftu.dev/snapshots")
maven("https://jitpack.io")
maven {
name = "Modrinth"
@@ -122,11 +136,30 @@ dependencies {
include(api("blue.endless:jankson:${property("jankson_version")}") as Any)
//endregion
+ //region Client
+ "clientImplementation"("dev.deftu:dearimguimc-1.20.6-fabric:0.1.0") {
+ exclude(group = "net.fabricmc.fabric-api")
+ }
+ listOf(
+ "binding",
+ "lwjgl3",
+ "natives-windows",
+ "natives-linux",
+ "natives-macos"
+ ).forEach { module ->
+ val version = "1.86.11"
+ implementation("io.github.spair:imgui-java-$module:$version") {
+ exclude(group = "org.lwjgl")
+ }
+ }
+ //endregion
+
// region Tests
"testmodImplementation"(sourceSets.main.get().output)
//endregion
}
+//region Misc
tasks.processResources {
val map = mapOf(
"version" to version
@@ -156,7 +189,9 @@ tasks.jar {
rename { "${it}_${property("archivesBaseName")}" }
}
}
+//endregion
+//region Publishing
publishing {
repositories {
maven {
@@ -181,6 +216,17 @@ signing {
sign(publishing.publications["main"])
}
+val CHANGELOG: String =
+ if (file("CHANGELOG.md").exists()) {
+ file("CHANGELOG.md").readText()
+ } else {
+ "No changelog provided"
+ }
+
+fun file(path: String): File {
+ return rootProject.file(path)
+}
+
modrinth {
val projectVersion: String = property("mod_version").toString()
val projectVersionNumber: List = projectVersion.split(Regex("-"), 2)
@@ -212,3 +258,4 @@ modrinth {
ModDependency("placeholder-api", "embedded")
)
}
+//endregion
diff --git a/gradle.properties b/gradle.properties
index e251cd1..6cbdadf 100644
--- a/gradle.properties
+++ b/gradle.properties
@@ -9,9 +9,9 @@ mod_version=1.0.0-alpha.1
# Dependencies | Check these on https://fabricmc.net/develop
minecraft_version=1.20.6
yarn_mappings=1.20.6+build.1
-fabric_loader=0.15.10
+fabric_loader=0.15.11
-fabric_api=0.98.0+1.20.6
+fabric_api=0.100.4+1.20.6
cardinal_components_api=6.0.0-beta.3
diff --git a/src/client/java/com/awakenedredstone/neoskies/NeoSkiesClient.java b/src/client/java/com/awakenedredstone/neoskies/NeoSkiesClient.java
new file mode 100644
index 0000000..3d038a2
--- /dev/null
+++ b/src/client/java/com/awakenedredstone/neoskies/NeoSkiesClient.java
@@ -0,0 +1,10 @@
+package com.awakenedredstone.neoskies;
+
+import net.fabricmc.api.ClientModInitializer;
+
+public class NeoSkiesClient implements ClientModInitializer {
+ @Override
+ public void onInitializeClient() {
+
+ }
+}
diff --git a/src/client/resources/neoskies.client.mixins.json b/src/client/resources/neoskies.client.mixins.json
new file mode 100644
index 0000000..4de9c26
--- /dev/null
+++ b/src/client/resources/neoskies.client.mixins.json
@@ -0,0 +1,10 @@
+{
+ "required": true,
+ "package": "com.awakenedredstone.neoskies.mixin.client",
+ "compatibilityLevel": "JAVA_17",
+ "client": [
+ ],
+ "injectors": {
+ "defaultRequire": 1
+ }
+}
diff --git a/src/main/java/com/awakenedredstone/neoskies/command/NeoSkiesCommands.java b/src/main/java/com/awakenedredstone/neoskies/command/NeoSkiesCommands.java
index 83a1354..fa6c7e3 100644
--- a/src/main/java/com/awakenedredstone/neoskies/command/NeoSkiesCommands.java
+++ b/src/main/java/com/awakenedredstone/neoskies/command/NeoSkiesCommands.java
@@ -127,7 +127,6 @@ private static void registerAdminCommands(CommandDispatcher
.putAny("current", integer)
.build()));
}, (timeTaken, scannedBlocks) -> {
- Map.Entry i = scannedBlocks.entrySet().stream().findFirst().get();
source.sendMessage(Texts.of("Scanned %total% blocks in %time%", new MapBuilder.StringMap()
.putAny("total", UnitConvertions.readableNumber(scannedBlocks.values().stream().mapToInt(value -> value).sum()))
.putAny("time", UnitConvertions.formatTimings(timeTaken))
@@ -140,6 +139,7 @@ private static void registerAdminCommands(CommandDispatcher
})
)
).then(CommandManager.literal("list")
+ .requires(Permissions.require("neoskies.admin.island.list", 4))
.executes(context -> {
LinedStringBuilder builder = new LinedStringBuilder();
List islands = IslandLogic.getInstance().islands.stuck;
diff --git a/src/main/java/com/awakenedredstone/neoskies/command/admin/IslandDataCommand.java b/src/main/java/com/awakenedredstone/neoskies/command/admin/IslandDataCommand.java
index 170869d..eeb6fef 100644
--- a/src/main/java/com/awakenedredstone/neoskies/command/admin/IslandDataCommand.java
+++ b/src/main/java/com/awakenedredstone/neoskies/command/admin/IslandDataCommand.java
@@ -24,36 +24,38 @@
public class IslandDataCommand {
public static void init(CommandDispatcher dispatcher) {
registerAdmin(dispatcher, adminNode()
- .then(literal("island-data")
- .then(literal("find")
- .requires(Permissions.require("neoskies.admin.island.data.find", 4))
- .then(argument("player", StringArgumentType.word())
- .suggests((context, builder) -> {
- List islands = IslandLogic.getInstance().islands.stuck;
- for (Island island : islands) {
- builder.suggest(island.owner.name);
- island.members.forEach(member -> {
- builder.suggest(member.name);
- });
- }
- return builder.buildFuture();
- }).executes(context -> {
- String playerName = StringArgumentType.getString(context, "player");
- Optional islandOptional = NeoSkiesAPI.getIslandByPlayer(playerName);
- return getIslandData(context.getSource(), islandOptional.orElse(null));
- })
- )
- ).then(literal("get")
- .then(argument("id", StringArgumentType.word())
- .suggests(CommandUtils.ISLAND_SUGGESTIONS)
- .executes(context -> {
- String islandId = StringArgumentType.getString(context, "id");
- Optional islandOptional = NeoSkiesAPI.getIsland(UUID.fromString(islandId));
- return getIslandData(context.getSource(), islandOptional.orElse(null));
- })
- )
- )
+ .then(literal("island-data")
+ .requires(Permissions.require("neoskies.admin.island.data", 4))
+ .then(literal("find")
+ .requires(Permissions.require("neoskies.admin.island.data.find", 4))
+ .then(argument("player", StringArgumentType.word())
+ .suggests((context, builder) -> {
+ List islands = IslandLogic.getInstance().islands.stuck;
+ for (Island island : islands) {
+ builder.suggest(island.owner.name);
+ island.members.forEach(member -> {
+ builder.suggest(member.name);
+ });
+ }
+ return builder.buildFuture();
+ }).executes(context -> {
+ String playerName = StringArgumentType.getString(context, "player");
+ Optional islandOptional = NeoSkiesAPI.getIslandByPlayer(playerName);
+ return getIslandData(context.getSource(), islandOptional.orElse(null));
+ })
+ )
+ ).then(literal("get")
+ .requires(Permissions.require("neoskies.admin.island.data.get", 4))
+ .then(argument("id", StringArgumentType.word())
+ .suggests(CommandUtils.ISLAND_SUGGESTIONS)
+ .executes(context -> {
+ String islandId = StringArgumentType.getString(context, "id");
+ Optional islandOptional = NeoSkiesAPI.getIsland(UUID.fromString(islandId));
+ return getIslandData(context.getSource(), islandOptional.orElse(null));
+ })
+ )
)
+ )
);
}
@@ -73,17 +75,17 @@ private static int getIslandData(ServerCommandSource source, @Nullable Island is
}
MapBuilder.StringMap map = new MapBuilder.StringMap()
- .put("id", island.getIslandId().toString())
- .put("owner", island.owner.name)
- .put("members", members.toString())
- .putAny("balance", island.getWallet().balance())
- .putAny("has_nether", island.hasNether)
- .putAny("has_end", island.hasEnd)
- .putAny("spawn_pos", island.spawnPos)
- .putAny("visit_pos", island.visitsPos)
- .putAny("radius", island.radius)
- .putAny("locked", island.locked)
- .putAny("created", island.getCreated().toEpochMilli());
+ .put("id", island.getIslandId().toString())
+ .put("owner", island.owner.name)
+ .put("members", members.toString())
+ .putAny("balance", island.getWallet().balance())
+ .putAny("has_nether", island.hasNether)
+ .putAny("has_end", island.hasEnd)
+ .putAny("spawn_pos", island.spawnPos)
+ .putAny("visit_pos", island.visitsPos)
+ .putAny("radius", island.radius)
+ .putAny("locked", island.locked)
+ .putAny("created", island.getCreated().toEpochMilli());
source.sendFeedback(() -> Texts.of("message.neoskies.island_data", map.build()), false);
return 1;
}
diff --git a/src/main/java/com/awakenedredstone/neoskies/command/island/HomeCommand.java b/src/main/java/com/awakenedredstone/neoskies/command/island/HomeCommand.java
index 034dc58..5a80f37 100644
--- a/src/main/java/com/awakenedredstone/neoskies/command/island/HomeCommand.java
+++ b/src/main/java/com/awakenedredstone/neoskies/command/island/HomeCommand.java
@@ -1,6 +1,5 @@
package com.awakenedredstone.neoskies.command.island;
-import com.awakenedredstone.neoskies.NeoSkies;
import com.awakenedredstone.neoskies.api.NeoSkiesAPI;
import com.awakenedredstone.neoskies.logic.Island;
import com.awakenedredstone.neoskies.logic.IslandLogic;
@@ -42,21 +41,4 @@ static void run(ServerPlayerEntity player) {
}
}, () -> player.sendMessage(Texts.prefixed("message.neoskies.home.no_island")));
}
-
- static void run(ServerPlayerEntity visitor, String islandOwner) {
- IslandLogic.getInstance().islands.getByPlayer(islandOwner).ifPresentOrElse(island -> {
- if (visitor.getWorld().getRegistryKey().getValue().equals(NeoSkies.id(island.owner.uuid.toString())) && !IslandLogic.getConfig().allowVisitCurrentIsland) {
- visitor.sendMessage(Texts.prefixed("message.neoskies.visit_home.fail", map -> map.put("owner", islandOwner)));
- } else {
- if (island.isMember(visitor)) {
- visitor.sendMessage(Texts.prefixed("message.neoskies.visit_home.success", map -> map.put("owner", islandOwner)));
- island.visitAsMember(visitor);
- } else {
- visitor.sendMessage(Texts.prefixed("message.neoskies.visit_home.not_member"));
- }
- }
- }, () -> {
- visitor.sendMessage(Texts.prefixed("message.neoskies.visit_home.no_island"));
- });
- }
}
diff --git a/src/main/java/com/awakenedredstone/neoskies/command/island/VisitCommand.java b/src/main/java/com/awakenedredstone/neoskies/command/island/VisitCommand.java
index 4e0b9cf..d7ff07d 100644
--- a/src/main/java/com/awakenedredstone/neoskies/command/island/VisitCommand.java
+++ b/src/main/java/com/awakenedredstone/neoskies/command/island/VisitCommand.java
@@ -4,10 +4,12 @@
import com.awakenedredstone.neoskies.logic.IslandLogic;
import com.awakenedredstone.neoskies.util.Texts;
import com.mojang.brigadier.CommandDispatcher;
+import com.mojang.brigadier.arguments.StringArgumentType;
import me.lucko.fabric.api.permissions.v0.Permissions;
import net.minecraft.command.argument.EntityArgumentType;
import net.minecraft.server.command.ServerCommandSource;
import net.minecraft.server.network.ServerPlayerEntity;
+import net.minecraft.text.Text;
import static com.awakenedredstone.neoskies.command.utils.CommandUtils.node;
import static com.awakenedredstone.neoskies.command.utils.CommandUtils.register;
@@ -21,10 +23,16 @@ public static void init(CommandDispatcher dispatcher) {
register(dispatcher, node()
.then(literal("visit")
.requires(Permissions.require("neoskies.island.visit", true))
- .then(argument("player", player())
+ .then(argument("player", StringArgumentType.word())
+ .suggests((context, builder) -> {
+ for (String playerName : context.getSource().getServer().getPlayerManager().getPlayerNames()) {
+ builder.suggest(playerName);
+ }
+ return builder.buildFuture();
+ })
.executes(context -> {
var visitor = context.getSource().getPlayer();
- var owner = EntityArgumentType.getPlayer(context, "player");
+ var owner = StringArgumentType.getString(context, "player");
if (visitor != null && owner != null) {
VisitCommand.run(visitor, owner);
}
@@ -35,25 +43,23 @@ public static void init(CommandDispatcher dispatcher) {
);
}
- static void run(ServerPlayerEntity visitor, ServerPlayerEntity owner) {
- String ownerName = owner.getName().getString();
-
+ static void run(ServerPlayerEntity visitor, String owner) {
IslandLogic.getInstance().islands.getByPlayer(owner).ifPresentOrElse(island -> {
if (!island.isMember(visitor) && island.isBanned(visitor)) {
- visitor.sendMessage(Texts.prefixed("message.neoskies.island_visit.ban", map -> map.put("owner", ownerName)));
+ visitor.sendMessage(Texts.prefixed("message.neoskies.island_visit.ban", map -> map.put("owner", owner)));
} else {
if (!island.locked) {
if (visitor.getWorld().getRegistryKey().getValue().equals(NeoSkies.id(island.owner.uuid.toString())) && !IslandLogic.getConfig().allowVisitCurrentIsland) {
- visitor.sendMessage(Texts.prefixed("message.neoskies.island_visit.fail", map -> map.put("owner", ownerName)));
+ visitor.sendMessage(Texts.prefixed("message.neoskies.island_visit.fail", map -> map.put("owner", owner)));
} else {
- visitor.sendMessage(Texts.prefixed("message.neoskies.island_visit.success", map -> map.put("owner", ownerName)));
+ visitor.sendMessage(Texts.prefixed("message.neoskies.island_visit.success", map -> map.put("owner", owner)));
island.visitAsVisitor(visitor);
}
} else {
- visitor.sendMessage(Texts.prefixed("message.neoskies.island_visit.no_visits", map -> map.put("owner", ownerName)));
+ visitor.sendMessage(Texts.prefixed("message.neoskies.island_visit.no_visits", map -> map.put("owner", owner)));
}
}
- }, () -> visitor.sendMessage(Texts.prefixed("message.neoskies.island_visit.no_island", map -> map.put("owner", ownerName))));
+ }, () -> visitor.sendMessage(Texts.prefixed("message.neoskies.island_visit.no_island", map -> map.put("owner", owner))));
}
}
diff --git a/src/main/java/com/awakenedredstone/neoskies/config/IslandRankingConfig.java b/src/main/java/com/awakenedredstone/neoskies/config/IslandRankingConfig.java
index 7306f91..8fe0157 100644
--- a/src/main/java/com/awakenedredstone/neoskies/config/IslandRankingConfig.java
+++ b/src/main/java/com/awakenedredstone/neoskies/config/IslandRankingConfig.java
@@ -9,19 +9,12 @@
import com.awakenedredstone.neoskies.config.source.annotation.SkipThis;
import com.awakenedredstone.neoskies.mixin.accessor.TagEntryAccessor;
import com.google.gson.JsonElement;
-import com.mojang.serialization.Codec;
import com.mojang.serialization.DataResult;
import com.mojang.serialization.JsonOps;
-import com.mojang.serialization.MapCodec;
-import com.mojang.serialization.codecs.RecordCodecBuilder;
-import net.minecraft.block.ButtonBlock;
-import net.minecraft.client.gui.screen.option.KeybindsScreen;
import net.minecraft.registry.Registries;
import net.minecraft.registry.RegistryKeys;
import net.minecraft.registry.tag.TagEntry;
import net.minecraft.registry.tag.TagKey;
-import net.minecraft.text.Text;
-import net.minecraft.text.TranslatableTextContent;
import net.minecraft.util.Identifier;
import java.util.HashMap;
diff --git a/src/main/java/com/awakenedredstone/neoskies/logic/Hub.java b/src/main/java/com/awakenedredstone/neoskies/logic/Hub.java
index cc5893b..988fce4 100644
--- a/src/main/java/com/awakenedredstone/neoskies/logic/Hub.java
+++ b/src/main/java/com/awakenedredstone/neoskies/logic/Hub.java
@@ -2,6 +2,7 @@
import com.awakenedredstone.neoskies.api.events.IslandEvents;
import com.awakenedredstone.neoskies.util.Texts;
+import com.awakenedredstone.neoskies.util.Worlds;
import net.fabricmc.fabric.api.dimension.v1.FabricDimensions;
import net.minecraft.entity.player.PlayerEntity;
import net.minecraft.nbt.NbtCompound;
@@ -14,14 +15,14 @@ public class Hub {
public boolean hasProtection = false;
//public PositionSongPlayer songPlayer = null;
- public void visit(PlayerEntity player) {
+ public void visit(ServerPlayerEntity player) {
visit(player, false);
}
- public void visit(PlayerEntity player, boolean silent) {
+ public void visit(ServerPlayerEntity player, boolean silent) {
var world = IslandLogic.getServer().getOverworld();
if (!silent) player.sendMessage(Texts.prefixed("message.neoskies.hub_visit"));
- FabricDimensions.teleport(player, world, new TeleportTarget(this.pos, new Vec3d(0, 0, 0), 0, 0));
+ Worlds.teleport(player, world, this.pos, 0, 0);
IslandEvents.ON_HUB_VISIT.invoker().invoke(player, world);
}
diff --git a/src/main/java/com/awakenedredstone/neoskies/logic/Island.java b/src/main/java/com/awakenedredstone/neoskies/logic/Island.java
index 24a5cde..8bac4c9 100644
--- a/src/main/java/com/awakenedredstone/neoskies/logic/Island.java
+++ b/src/main/java/com/awakenedredstone/neoskies/logic/Island.java
@@ -11,12 +11,15 @@
import com.awakenedredstone.neoskies.util.Constants;
import com.awakenedredstone.neoskies.util.Players;
import com.awakenedredstone.neoskies.util.Texts;
+import com.awakenedredstone.neoskies.util.Worlds;
import eu.pb4.common.economy.api.EconomyAccount;
import net.minecraft.block.Block;
import net.minecraft.entity.player.PlayerEntity;
import net.minecraft.nbt.NbtCompound;
+import net.minecraft.registry.RegistryKey;
import net.minecraft.registry.RegistryKeys;
import net.minecraft.server.MinecraftServer;
+import net.minecraft.server.network.ServerPlayerEntity;
import net.minecraft.server.world.ServerWorld;
import net.minecraft.structure.StructurePlacementData;
import net.minecraft.structure.StructureTemplate;
@@ -27,6 +30,7 @@
import net.minecraft.util.math.Vec3d;
import net.minecraft.util.math.random.RandomSeed;
import net.minecraft.world.Difficulty;
+import net.minecraft.world.World;
import net.minecraft.world.biome.BiomeKeys;
import net.minecraft.world.dimension.DimensionTypes;
import net.minecraft.world.gen.chunk.FlatChunkGenerator;
@@ -308,14 +312,7 @@ public boolean isInteractionAllowed(Identifier identifier, PermissionLevel sourc
return source.getLevel() >= settings.getPermissionLevel().getLevel();
}
- public RuntimeWorldHandle getOverworldHandler() {
- if (this.islandConfig == null) {
- this.islandConfig = createIslandConfig();
- }
- return this.fantasy.getOrOpenPersistentWorld(NeoSkies.id(this.owner.uuid.toString()), this.islandConfig);
- }
-
- private RuntimeWorldConfig createIslandConfig() {
+ private RuntimeWorldConfig createOverworldConfig() {
var biome = IslandLogic.getServer().getRegistryManager().get(RegistryKeys.BIOME).getEntry(IslandLogic.getServer().getRegistryManager().get(RegistryKeys.BIOME).getOrThrow(BiomeKeys.PLAINS));
FlatChunkGeneratorConfig flat = new FlatChunkGeneratorConfig(Optional.empty(), biome, List.of());
FlatChunkGenerator generator = new FlatChunkGenerator(flat);
@@ -329,13 +326,6 @@ private RuntimeWorldConfig createIslandConfig() {
.setSeed(0L);
}
- public RuntimeWorldHandle getNetherHandler() {
- if (this.netherConfig == null) {
- this.netherConfig = createNetherConfig();
- }
- return this.fantasy.getOrOpenPersistentWorld(new Identifier(Constants.NAMESPACE_NETHER, this.owner.uuid.toString()), this.netherConfig);
- }
-
private RuntimeWorldConfig createNetherConfig() {
var biome = IslandLogic.getServer().getRegistryManager().get(RegistryKeys.BIOME).getEntry(IslandLogic.getServer().getRegistryManager().get(RegistryKeys.BIOME).getOrThrow(BiomeKeys.NETHER_WASTES));
FlatChunkGeneratorConfig flat = new FlatChunkGeneratorConfig(Optional.empty(), biome, List.of());
@@ -349,13 +339,6 @@ private RuntimeWorldConfig createNetherConfig() {
.setSeed(RandomSeed.getSeed());
}
- public RuntimeWorldHandle getEndHandler() {
- if (this.endConfig == null) {
- this.endConfig = createEndConfig();
- }
- return this.fantasy.getOrOpenPersistentWorld(new Identifier(Constants.NAMESPACE_END, this.owner.uuid.toString()), this.endConfig);
- }
-
private RuntimeWorldConfig createEndConfig() {
var biome = IslandLogic.getServer().getRegistryManager().get(RegistryKeys.BIOME).getEntry(IslandLogic.getServer().getRegistryManager().get(RegistryKeys.BIOME).getOrThrow(BiomeKeys.THE_END));
FlatChunkGeneratorConfig flat = new FlatChunkGeneratorConfig(Optional.empty(), biome, List.of());
@@ -369,6 +352,27 @@ private RuntimeWorldConfig createEndConfig() {
.setSeed(RandomSeed.getSeed());
}
+ public RuntimeWorldHandle getOverworldHandler() {
+ if (this.islandConfig == null) {
+ this.islandConfig = createOverworldConfig();
+ }
+ return this.fantasy.getOrOpenPersistentWorld(NeoSkies.id(this.owner.uuid.toString()), this.islandConfig);
+ }
+
+ public RuntimeWorldHandle getNetherHandler() {
+ if (this.netherConfig == null) {
+ this.netherConfig = createNetherConfig();
+ }
+ return this.fantasy.getOrOpenPersistentWorld(new Identifier(Constants.NAMESPACE_NETHER, this.owner.uuid.toString()), this.netherConfig);
+ }
+
+ public RuntimeWorldHandle getEndHandler() {
+ if (this.endConfig == null) {
+ this.endConfig = createEndConfig();
+ }
+ return this.fantasy.getOrOpenPersistentWorld(new Identifier(Constants.NAMESPACE_END, this.owner.uuid.toString()), this.endConfig);
+ }
+
public ServerWorld getOverworld() {
RuntimeWorldHandle handler = this.getOverworldHandler();
handler.setTickWhenEmpty(false);
@@ -391,6 +395,18 @@ public ServerWorld getNether() {
return world;
}
+ public RegistryKey getOverworldKey() {
+ return getOverworld().getRegistryKey();
+ }
+
+ public RegistryKey getNetherKey() {
+ return getNether().getRegistryKey();
+ }
+
+ public RegistryKey getEndKey() {
+ return getEnd().getRegistryKey();
+ }
+
public void updateBlocks(@Nullable Map blocks) {
if (blocks != null) this.blocks = blocks;
this.points = 0;
@@ -400,9 +416,9 @@ public void updateBlocks(@Nullable Map blocks) {
});
}
- public void visit(PlayerEntity player, Vec3d pos) {
+ public void visit(ServerPlayerEntity player, Vec3d pos) {
ServerWorld world = this.getOverworld();
- player.teleport(world, pos.getX(), pos.getY(), pos.getZ(), Set.of(), 0, 0);
+ Worlds.teleport(player, world, pos.getX(), pos.getY(), pos.getZ(), 0, 0);
if (!isMember(player)) {
Players.get(this.owner.name).ifPresent(owner -> {
@@ -420,11 +436,11 @@ public void visit(PlayerEntity player, Vec3d pos) {
}
}
- public void visitAsMember(PlayerEntity player) {
+ public void visitAsMember(ServerPlayerEntity player) {
this.visit(player, this.spawnPos);
}
- public void visitAsVisitor(PlayerEntity player) {
+ public void visitAsVisitor(ServerPlayerEntity player) {
this.visit(player, this.visitsPos);
}
diff --git a/src/main/java/com/awakenedredstone/neoskies/logic/IslandStuck.java b/src/main/java/com/awakenedredstone/neoskies/logic/IslandStuck.java
index 201b74e..123c522 100644
--- a/src/main/java/com/awakenedredstone/neoskies/logic/IslandStuck.java
+++ b/src/main/java/com/awakenedredstone/neoskies/logic/IslandStuck.java
@@ -99,6 +99,17 @@ public Optional getByPlayer(UUID playerUuid) {
return Optional.empty();
}
+ public Optional getById(String islandId) {
+ return getById(UUID.fromString(islandId));
+ }
+
+ public Optional getById(UUID islandId) {
+ for (var island : this.stuck) {
+ if (island.getIslandId().equals(islandId)) return Optional.of(island);
+ }
+ return Optional.empty();
+ }
+
public Optional get(UUID islandId) {
for (var island : this.stuck) {
if (island.getIslandId().equals(islandId)) return Optional.of(island);
diff --git a/src/main/java/com/awakenedredstone/neoskies/logic/registry/NeoSkiesIslandSettings.java b/src/main/java/com/awakenedredstone/neoskies/logic/registry/NeoSkiesIslandSettings.java
index ead5e80..db5302b 100644
--- a/src/main/java/com/awakenedredstone/neoskies/logic/registry/NeoSkiesIslandSettings.java
+++ b/src/main/java/com/awakenedredstone/neoskies/logic/registry/NeoSkiesIslandSettings.java
@@ -39,7 +39,7 @@ public class NeoSkiesIslandSettings {
public static final IslandSettings USE_LODESTONE = register("use/lodestone", Items.LODESTONE);
public static final IslandSettings USE_REDSTONE = register("use/redstone", Items.REDSTONE);
public static final IslandSettings USE_RESPAWN_ANCHOR = register("use/respawn_anchor", Items.RESPAWN_ANCHOR);
- public static final IslandSettings USE_SIGNS = register("use/signs", Items.SPAWNER);
+ public static final IslandSettings USE_SIGNS = register("use/signs", Items.OAK_SIGN);
public static final IslandSettings USE_SPAWNER = register("use/spawner", Items.SPAWNER);
public static final IslandSettings USE_TNT = register("use/tnt", Items.TNT);
@@ -51,7 +51,7 @@ public class NeoSkiesIslandSettings {
public static final IslandSettings HARVEST = register("harvest", Items.SWEET_BERRIES);
public static final IslandSettings RIDE_MINECARTS = register("ride/minecarts", Items.MINECART);
- public static final IslandSettings RIDE_BOATS = register("ride/boats", Items.MINECART);
+ public static final IslandSettings RIDE_BOATS = register("ride/boats", Items.OAK_BOAT);
public static final IslandSettings RIDE_OTHERS = register("ride/others", Items.SADDLE);
public static final IslandSettings LEASH_ENTITY = register("leash/entity", Items.LEAD);
public static final IslandSettings SHEAR_ENTITY = register("shear/entity", Items.SHEARS);
diff --git a/src/main/java/com/awakenedredstone/neoskies/mixin/block/EndPortalBlockMixin.java b/src/main/java/com/awakenedredstone/neoskies/mixin/block/EndPortalBlockMixin.java
index 2a3892a..d1b2bd0 100644
--- a/src/main/java/com/awakenedredstone/neoskies/mixin/block/EndPortalBlockMixin.java
+++ b/src/main/java/com/awakenedredstone/neoskies/mixin/block/EndPortalBlockMixin.java
@@ -3,43 +3,41 @@
import com.awakenedredstone.neoskies.api.NeoSkiesAPI;
import com.awakenedredstone.neoskies.logic.Island;
import com.awakenedredstone.neoskies.logic.IslandLogic;
-import net.minecraft.block.BlockState;
+import com.llamalad7.mixinextras.sugar.Local;
import net.minecraft.block.EndPortalBlock;
-import net.minecraft.entity.Entity;
-import net.minecraft.server.world.ServerWorld;
-import net.minecraft.util.math.BlockPos;
+import net.minecraft.registry.RegistryKey;
import net.minecraft.world.World;
import org.spongepowered.asm.mixin.Mixin;
import org.spongepowered.asm.mixin.injection.At;
-import org.spongepowered.asm.mixin.injection.Inject;
-import org.spongepowered.asm.mixin.injection.callback.CallbackInfo;
-import org.spongepowered.asm.mixin.injection.callback.LocalCapture;
+import org.spongepowered.asm.mixin.injection.ModifyVariable;
import java.util.Optional;
@Mixin(EndPortalBlock.class)
public class EndPortalBlockMixin {
- @Inject(method = "onEntityCollision", at = @At(value = "INVOKE", target = "Lnet/minecraft/entity/Entity;moveToWorld(Lnet/minecraft/server/world/ServerWorld;)Lnet/minecraft/entity/Entity;"), locals = LocalCapture.CAPTURE_FAILHARD, cancellable = true)
- public void resourceKey(BlockState state, World world, BlockPos pos, Entity entity, CallbackInfo ci) {
+ @ModifyVariable(method = "onEntityCollision", at = @At(value = "STORE"), ordinal = 0)
+ public RegistryKey resourceKey(RegistryKey original, @Local(argsOnly = true) World world) {
if (NeoSkiesAPI.isIsland(world)) {
+ Optional island = NeoSkiesAPI.getIsland(world);
if (!IslandLogic.getConfig().enableEndIsland) {
- ci.cancel();
- return;
+ if (island.isPresent()) {
+ return island.get().getOverworldKey();
+ } else {
+ return world.getRegistryKey();
+ }
}
- Optional island = NeoSkiesAPI.getIsland(world);
if (island.isPresent()) {
- ServerWorld targetWorld;
+ RegistryKey targetWorld;
if (NeoSkiesAPI.isEnd(world.getRegistryKey())) {
- targetWorld = island.get().getOverworld();
+ targetWorld = island.get().getOverworldKey();
} else {
- targetWorld = island.get().getEnd();
- }
- if (targetWorld != null) {
- entity.moveToWorld(targetWorld);
+ targetWorld = island.get().getEndKey();
}
- ci.cancel();
+ return targetWorld;
}
}
+
+ return original;
}
}
diff --git a/src/main/java/com/awakenedredstone/neoskies/mixin/entity/EntityAccessor.java b/src/main/java/com/awakenedredstone/neoskies/mixin/entity/EntityAccessor.java
new file mode 100644
index 0000000..7c38dde
--- /dev/null
+++ b/src/main/java/com/awakenedredstone/neoskies/mixin/entity/EntityAccessor.java
@@ -0,0 +1,10 @@
+package com.awakenedredstone.neoskies.mixin.entity;
+
+import net.minecraft.entity.Entity;
+import org.spongepowered.asm.mixin.Mixin;
+import org.spongepowered.asm.mixin.gen.Invoker;
+
+@Mixin(Entity.class)
+public interface EntityAccessor {
+ @Invoker void callUnsetRemoved();
+}
diff --git a/src/main/java/com/awakenedredstone/neoskies/mixin/entity/PlayerManagerMixin.java b/src/main/java/com/awakenedredstone/neoskies/mixin/entity/PlayerManagerMixin.java
index ae966ba..17bdda6 100644
--- a/src/main/java/com/awakenedredstone/neoskies/mixin/entity/PlayerManagerMixin.java
+++ b/src/main/java/com/awakenedredstone/neoskies/mixin/entity/PlayerManagerMixin.java
@@ -104,7 +104,11 @@ private void loadIsland(ClientConnection connection, ServerPlayerEntity player,
@ModifyExpressionValue(method = "respawnPlayer", at = @At(value = "INVOKE", target = "Lnet/minecraft/server/network/ServerPlayerEntity;getSpawnPointDimension()Lnet/minecraft/registry/RegistryKey;"))
private RegistryKey neoskies$fixRespawnDimension(RegistryKey original, ServerPlayerEntity player) {
- return player.getWorld().getRegistryKey();
+ World playerWorld = player.getWorld();
+ if (NeoSkiesAPI.isIsland(playerWorld)) {
+ return NeoSkiesAPI.getIsland(playerWorld).get().getOverworldKey();
+ }
+ return IslandLogic.getServer().getOverworld().getRegistryKey();
}
@Redirect(method = "respawnPlayer", at = @At(value = "INVOKE", target = "Lnet/minecraft/entity/player/PlayerEntity;findRespawnPosition(Lnet/minecraft/server/world/ServerWorld;Lnet/minecraft/util/math/BlockPos;FZZ)Ljava/util/Optional;"))
diff --git a/src/main/java/com/awakenedredstone/neoskies/mixin/entity/ServerPlayerEntityAccessor.java b/src/main/java/com/awakenedredstone/neoskies/mixin/entity/ServerPlayerEntityAccessor.java
new file mode 100644
index 0000000..b89a49c
--- /dev/null
+++ b/src/main/java/com/awakenedredstone/neoskies/mixin/entity/ServerPlayerEntityAccessor.java
@@ -0,0 +1,11 @@
+package com.awakenedredstone.neoskies.mixin.entity;
+
+import net.minecraft.server.network.ServerPlayerEntity;
+import net.minecraft.server.world.ServerWorld;
+import org.spongepowered.asm.mixin.Mixin;
+import org.spongepowered.asm.mixin.gen.Invoker;
+
+@Mixin(ServerPlayerEntity.class)
+public interface ServerPlayerEntityAccessor {
+ @Invoker void callWorldChanged(ServerWorld world);
+}
diff --git a/src/main/java/com/awakenedredstone/neoskies/mixin/entity/ServerPlayerEntityMixin.java b/src/main/java/com/awakenedredstone/neoskies/mixin/entity/ServerPlayerEntityMixin.java
index cb10979..bd04bf9 100644
--- a/src/main/java/com/awakenedredstone/neoskies/mixin/entity/ServerPlayerEntityMixin.java
+++ b/src/main/java/com/awakenedredstone/neoskies/mixin/entity/ServerPlayerEntityMixin.java
@@ -5,6 +5,7 @@
import com.awakenedredstone.neoskies.event.PlayerEvents;
import com.awakenedredstone.neoskies.util.Worlds;
import com.llamalad7.mixinextras.injector.ModifyExpressionValue;
+import com.llamalad7.mixinextras.sugar.Local;
import com.mojang.authlib.GameProfile;
import net.minecraft.entity.damage.DamageSource;
import net.minecraft.entity.player.PlayerEntity;
@@ -21,8 +22,6 @@
import org.spongepowered.asm.mixin.Shadow;
import org.spongepowered.asm.mixin.injection.At;
import org.spongepowered.asm.mixin.injection.Inject;
-import org.spongepowered.asm.mixin.injection.Redirect;
-import org.spongepowered.asm.mixin.injection.Slice;
import org.spongepowered.asm.mixin.injection.callback.CallbackInfo;
@Mixin(ServerPlayerEntity.class)
@@ -73,6 +72,7 @@ public RegistryKey getTeleportTarget_redirectRegistryKey(RegistryKey worldChanged_redirectRegistryKey(RegistryKey world) {
return Worlds.redirect(world);
}
+
@ModifyExpressionValue(method = "createCommonPlayerSpawnInfo", at = @At(value = "INVOKE", target = "Lnet/minecraft/server/world/ServerWorld;getRegistryKey()Lnet/minecraft/registry/RegistryKey;"))
public RegistryKey createCommonPlayerSpawnInfo_redirectRegistryKey(RegistryKey world) {
return Worlds.redirect(world);
diff --git a/src/main/java/com/awakenedredstone/neoskies/util/Worlds.java b/src/main/java/com/awakenedredstone/neoskies/util/Worlds.java
index f0c57b6..2e4e96c 100644
--- a/src/main/java/com/awakenedredstone/neoskies/util/Worlds.java
+++ b/src/main/java/com/awakenedredstone/neoskies/util/Worlds.java
@@ -4,12 +4,19 @@
import com.awakenedredstone.neoskies.logic.Hub;
import com.awakenedredstone.neoskies.logic.Island;
import com.awakenedredstone.neoskies.logic.IslandLogic;
+import com.awakenedredstone.neoskies.mixin.entity.EntityAccessor;
+import com.awakenedredstone.neoskies.mixin.entity.ServerPlayerEntityAccessor;
import net.fabricmc.fabric.api.dimension.v1.FabricDimensions;
+import net.minecraft.entity.Entity;
+import net.minecraft.network.packet.s2c.play.DifficultyS2CPacket;
+import net.minecraft.network.packet.s2c.play.PlayerRespawnS2CPacket;
import net.minecraft.registry.RegistryKey;
import net.minecraft.server.network.ServerPlayerEntity;
+import net.minecraft.server.world.ServerWorld;
import net.minecraft.util.math.Vec3d;
import net.minecraft.world.TeleportTarget;
import net.minecraft.world.World;
+import net.minecraft.world.WorldProperties;
import java.util.Optional;
@@ -53,4 +60,36 @@ public static RegistryKey redirect(RegistryKey registryKey) {
return registryKey;
}
+ public static RegistryKey redirect(World world) {
+ return redirect(world.getRegistryKey());
+ }
+
+ public static void teleport(ServerPlayerEntity player, ServerWorld targetWorld, Vec3d pos, float yaw, float pitch) {
+ teleport(player, targetWorld, pos.x, pos.y, pos.z, yaw, pitch);
+ }
+
+ public static void teleport(ServerPlayerEntity player, ServerWorld targetWorld, double x, double y, double z, float yaw, float pitch) {
+ player.setCameraEntity(player);
+ player.stopRiding();
+ if (targetWorld == player.getWorld()) {
+ player.networkHandler.requestTeleport(x, y, z, yaw, pitch);
+ } else {
+ ServerWorld serverWorld = player.getServerWorld();
+ WorldProperties worldProperties = targetWorld.getLevelProperties();
+ if (!redirect(targetWorld).equals(redirect(player.getWorld()))) {
+ player.networkHandler.sendPacket(new PlayerRespawnS2CPacket(player.createCommonPlayerSpawnInfo(targetWorld), PlayerRespawnS2CPacket.KEEP_ALL));
+ }
+ player.networkHandler.sendPacket(new DifficultyS2CPacket(worldProperties.getDifficulty(), worldProperties.isDifficultyLocked()));
+ player.server.getPlayerManager().sendCommandTree(player);
+ serverWorld.removePlayer(player, Entity.RemovalReason.CHANGED_DIMENSION);
+ ((EntityAccessor) player).callUnsetRemoved();
+ player.refreshPositionAndAngles(x, y, z, yaw, pitch);
+ player.setServerWorld(targetWorld);
+ targetWorld.onPlayerTeleport(player);
+ ((ServerPlayerEntityAccessor) player).callWorldChanged(serverWorld);
+ player.networkHandler.requestTeleport(x, y, z, yaw, pitch);
+ player.server.getPlayerManager().sendWorldInfo(player, targetWorld);
+ player.server.getPlayerManager().sendPlayerStatus(player);
+ }
+ }
}
diff --git a/src/main/resources/data/neoskies/lang/en_us.json b/src/main/resources/data/neoskies/lang/en_us.json
index 321cca4..86d5ff5 100644
--- a/src/main/resources/data/neoskies/lang/en_us.json
+++ b/src/main/resources/data/neoskies/lang/en_us.json
@@ -125,7 +125,7 @@
"island_settings.neoskies.ride/others.description": "Allows players to ride mobs",
"island_settings.neoskies.leash/entity.description": "Allows players to leash entities",
"island_settings.neoskies.shear/entity.description": "Allows players to shear entities",
- "island_settings.neoskies.bucket/passive.description": "Allow players to bucket passive mobs",
+ "island_settings.neoskies.bucket/passive.description": "Allow players to bucket passive mobs",
"island_settings.neoskies.harvest.description": "Allows players to harvest crops",
"island_settings/selected": "> %level%",
diff --git a/src/main/resources/fabric.mod.json b/src/main/resources/fabric.mod.json
index 90cf3fd..d1a1793 100644
--- a/src/main/resources/fabric.mod.json
+++ b/src/main/resources/fabric.mod.json
@@ -20,10 +20,18 @@
],
"cardinal-components": [
"com.awakenedredstone.neoskies.data.NeoSkiesComponents"
+ ],
+
+ "client": [
+ "com.awakenedredstone.neoskies.NeoSkiesClient"
]
},
"mixins": [
- "neoskies.mixins.json"
+ "neoskies.mixins.json",
+ {
+ "config": "neoskies.client.mixins.json",
+ "environment": "client"
+ }
],
"custom": {
"cardinal-components": [
diff --git a/src/main/resources/neoskies.mixins.json b/src/main/resources/neoskies.mixins.json
index 458e634..1b9021c 100644
--- a/src/main/resources/neoskies.mixins.json
+++ b/src/main/resources/neoskies.mixins.json
@@ -14,8 +14,10 @@
"compat.JanksonAcessor",
"compat.POJODeserializerAccessor",
"entity.DataTrackerAccessor",
+ "entity.EntityAccessor",
"entity.EntityMixin",
"entity.PlayerManagerMixin",
+ "entity.ServerPlayerEntityAccessor",
"entity.ServerPlayerEntityMixin",
"item.BoneMealItemMixin",
"item.ItemStackMixin",
diff --git a/src/testmod/resources/fabric.mod.json b/src/testmod/resources/fabric.mod.json
index 8b780b9..8699078 100644
--- a/src/testmod/resources/fabric.mod.json
+++ b/src/testmod/resources/fabric.mod.json
@@ -3,9 +3,11 @@
"id": "neoskies-test",
"version": "${version}",
"name": "NeoSkies TestMod",
- "description": "Library to work with server-side events with the ability to filter by event source",
- "authors": ["Nucleoid Contributors"],
- "license": "LGPLv3",
+ "description": "The testmod for NeoSkies | Also includes devtools",
+ "authors": [
+ "Awakened Redstone"
+ ],
+ "license": "LGPL-3.0",
"environment": "*",
"entrypoints": {
"main": [
diff --git a/src/testmodClient/java/com/awakenedredstone/neoskies/test/DebugDearImGuiEntrypoint.java b/src/testmodClient/java/com/awakenedredstone/neoskies/test/DebugDearImGuiEntrypoint.java
new file mode 100644
index 0000000..5393104
--- /dev/null
+++ b/src/testmodClient/java/com/awakenedredstone/neoskies/test/DebugDearImGuiEntrypoint.java
@@ -0,0 +1,17 @@
+package com.awakenedredstone.neoskies.test;
+
+import dev.deftu.imgui.DearImGuiEntrypoint;
+import dev.deftu.imgui.ImGuiRenderer;
+
+public class DebugDearImGuiEntrypoint implements DearImGuiEntrypoint {
+
+ @Override
+ public ImGuiRenderer createRenderer() {
+ return new DebugImGuiRenderer();
+ }
+
+ @Override
+ public void render() {
+ // Render ImGui here
+ }
+}
diff --git a/src/testmodClient/java/com/awakenedredstone/neoskies/test/DebugImGuiRenderer.java b/src/testmodClient/java/com/awakenedredstone/neoskies/test/DebugImGuiRenderer.java
new file mode 100644
index 0000000..5a80263
--- /dev/null
+++ b/src/testmodClient/java/com/awakenedredstone/neoskies/test/DebugImGuiRenderer.java
@@ -0,0 +1,176 @@
+package com.awakenedredstone.neoskies.test;
+
+import com.awakenedredstone.neoskies.logic.Island;
+import com.awakenedredstone.neoskies.logic.IslandLogic;
+import com.awakenedredstone.neoskies.util.MapBuilder;
+import com.awakenedredstone.neoskies.util.Texts;
+import com.awakenedredstone.neoskies.util.UnitConvertions;
+import dev.deftu.imgui.ImGuiRenderer;
+import imgui.ImGui;
+import imgui.extension.implot.ImPlot;
+import imgui.extension.implot.ImPlotContext;
+import imgui.extension.implot.flag.ImPlotStyleVar;
+import imgui.flag.ImGuiCond;
+import imgui.flag.ImGuiWindowFlags;
+import imgui.type.ImBoolean;
+import net.minecraft.client.MinecraftClient;
+import net.minecraft.client.gui.screen.Screen;
+import net.minecraft.text.Text;
+
+import java.util.*;
+import java.util.concurrent.atomic.AtomicInteger;
+
+public class DebugImGuiRenderer implements ImGuiRenderer {
+ private static final ImPlotContext IMPLOT_CONTEXT;
+ public static final ImBoolean OPEN = new ImBoolean(true);
+ private static final Map MESSAGES = new HashMap<>();
+
+ static {
+ IMPLOT_CONTEXT = ImPlot.createContext();
+ }
+
+ public static int bufferLength = 600;
+ public static final List MEMORY = new LinkedList<>() {
+ @Override
+ public boolean add(Long aFloat) {
+ if (this.size() >= 10000) {
+ this.removeFirst();
+ }
+ return super.add(aFloat);
+ }
+ };
+
+ @Override
+ public void render() {
+ if (!OPEN.get()) return;
+
+ /*ImPlot.showDemoWindow(OPEN);
+ ImGui.showDemoWindow();*/
+
+ ImGui.setNextWindowSize(600, 750, ImGuiCond.FirstUseEver);
+ if (ImGui.begin("NeoSkies debug tools", OPEN, ImGuiWindowFlags.MenuBar)) {
+ if (MinecraftClient.getInstance().player != null && IslandLogic.getInstance() != null) {
+ ImGui.text(IslandLogic.getServer().getPlayerManager().getPlayer(MinecraftClient.getInstance().player.getUuid()).getWorld().getRegistryKey().getValue().toString());
+ ImGui.text(String.valueOf(MinecraftClient.getInstance().player.getWorld().getWorldBorder().getSize()));
+ }
+
+ if (ImGui.collapsingHeader("Server RAM")) {
+ final Float[] samples = new Float[bufferLength];
+ Arrays.fill(samples, 0f);
+
+ final Float[] samples2 = new Float[bufferLength];
+ for (int i = 0; i < samples.length; i++) {
+ try {
+ if (MEMORY.size() < samples.length && i < MEMORY.size()) {
+ samples[samples.length - MEMORY.size() + i] = (float) toMiB(MEMORY.get(i));
+ } else if (i < MEMORY.size()) {
+ samples[i] = (float) toMiB(MEMORY.get(Math.max(0, MEMORY.size() - bufferLength) + i));
+ }
+ samples2[i] = (float) i;
+ } catch (Throwable ignored) {}
+ }
+
+ if (ImGui.button("Clear")) {
+ MEMORY.clear();
+ }
+ ImGui.sameLine();
+
+ ImGui.text("Buffer size: %s".formatted(formatSeconds(bufferLength / 10)));
+ ImGui.sameLine();
+ if (ImGui.button("+")) {
+ if (Screen.hasShiftDown()) {
+ bufferLength += 100;
+ } else {
+ bufferLength += 10;
+ }
+ }
+ ImGui.sameLine();
+ if (ImGui.button("-")) {
+ if (Screen.hasShiftDown()) {
+ bufferLength -= 100;
+ } else {
+ bufferLength -= 10;
+ }
+ }
+
+ long maxMemory = Runtime.getRuntime().maxMemory();
+ long totalMemory = Runtime.getRuntime().totalMemory();
+ long freeMemory = Runtime.getRuntime().freeMemory();
+ long usedMemory = totalMemory - freeMemory;
+ ImGui.text("Memory: %s%% (%s/%sMiB)".formatted(usedMemory * 100L / maxMemory, toMiB(usedMemory), toMiB(maxMemory)));
+
+ ImPlot.setNextPlotLimits(0, bufferLength, 0, toMiB(maxMemory), ImGuiCond.Always);
+ if (ImPlot.beginPlot("Example Plot")) {
+ ImPlot.pushStyleVar(ImPlotStyleVar.FillAlpha, 0.25f);
+ try {
+ ImPlot.plotLine("Line", samples2, samples);
+ ImPlot.plotShaded("Line", samples2, samples, 0, bufferLength);
+ } catch (Throwable ignored) {}
+ ImPlot.endPlot();
+ }
+ }
+
+ if (ImGui.collapsingHeader("Islands")) {
+ if (MESSAGES.containsKey("island.info")) {
+ ImGui.text(MESSAGES.get("island.info").getString());
+ }
+
+ for (Island island : IslandLogic.getInstance().islands.stuck) {
+ if (ImGui.button("Scan " + island.hashCode())) { //Each button must have a unique id, and the id is the label ._.
+ if (island.isScanning()) {
+ MESSAGES.put("island.info", Texts.of("Can not queue a scan for an island that is already scanning!"));
+ } else {
+ MESSAGES.put("island.info", Texts.of("Scan queued"));
+
+ AtomicInteger total = new AtomicInteger();
+ IslandLogic.getInstance().islandScanner.queueScan(island, integer -> {
+ MESSAGES.put("island.info", Texts.of("Scanning %total% chunks", new MapBuilder.StringMap().putAny("total", integer).build()));
+ total.set(integer);
+ }, integer -> {
+ MESSAGES.put("island.info", Texts.of("Scanned %current%/%total% chunks", new MapBuilder.StringMap()
+ .putAny("total", total.get())
+ .putAny("current", integer)
+ .build()));
+ }, (timeTaken, scannedBlocks) -> {
+ MESSAGES.put("island.info", Texts.of("Scanned %total% blocks in %time%", new MapBuilder.StringMap()
+ .putAny("total", UnitConvertions.readableNumber(scannedBlocks.values().stream().mapToInt(value -> value).sum()))
+ .putAny("time", UnitConvertions.formatTimings(timeTaken))
+ .build()));
+ }, () -> {
+ MESSAGES.put("island.info", Texts.of("Island scan failed"));
+ });
+ }
+ }
+ ImGui.sameLine();
+ ImGui.text("%s: %s (%s)".formatted(island.owner.name, island.getIslandId(), island.getPoints()));
+ }
+ }
+
+ ImGui.end();
+ }
+ }
+
+ private static long toMiB(long bytes) {
+ return bytes / 1024L / 1024L;
+ }
+
+ private static String formatSeconds(int seconds) {
+ StringBuilder builder = new StringBuilder();
+ if (seconds >= 60) {
+ int minutes = seconds / 60;
+ if (minutes >= 60) {
+ int hours = minutes / 60;
+ builder.append(hours).append("h");
+ }
+ if (minutes % 60 > 0) {
+ builder.append(minutes % 60).append("m");
+ }
+ }
+
+ if (seconds % 60 > 0) {
+ builder.append(seconds % 60).append("s");
+ }
+
+ return builder.toString();
+ }
+}
diff --git a/src/testmodClient/java/com/awakenedredstone/neoskies/test/NeoSkiesTestClientMain.java b/src/testmodClient/java/com/awakenedredstone/neoskies/test/NeoSkiesTestClientMain.java
new file mode 100644
index 0000000..61e0ef7
--- /dev/null
+++ b/src/testmodClient/java/com/awakenedredstone/neoskies/test/NeoSkiesTestClientMain.java
@@ -0,0 +1,30 @@
+package com.awakenedredstone.neoskies.test;
+
+import net.fabricmc.api.ClientModInitializer;
+import net.fabricmc.fabric.api.client.command.v2.ClientCommandManager;
+import net.fabricmc.fabric.api.client.command.v2.ClientCommandRegistrationCallback;
+import net.fabricmc.fabric.api.client.event.lifecycle.v1.ClientTickEvents;
+
+public final class NeoSkiesTestClientMain implements ClientModInitializer {
+ private static int tick = 0;
+
+ @Override
+ public void onInitializeClient() {
+ ClientTickEvents.END_CLIENT_TICK.register(client -> {
+ if (tick++ % 2 == 0) {
+ DebugImGuiRenderer.MEMORY.add((Runtime.getRuntime().totalMemory() - Runtime.getRuntime().freeMemory()));
+ }
+ });
+
+ ClientCommandRegistrationCallback.EVENT.register((dispatcher, registryAccess) -> {
+ dispatcher.register(ClientCommandManager.literal("sbac")
+ .then(ClientCommandManager.literal("debug")
+ .executes(context -> {
+ DebugImGuiRenderer.OPEN.set(true);
+ return 0;
+ })
+ )
+ );
+ });
+ }
+}
diff --git a/src/testmodClient/resources/fabric.mod.json b/src/testmodClient/resources/fabric.mod.json
new file mode 100644
index 0000000..5a8525a
--- /dev/null
+++ b/src/testmodClient/resources/fabric.mod.json
@@ -0,0 +1,23 @@
+{
+ "schemaVersion": 1,
+ "id": "neoskies-test-client",
+ "version": "${version}",
+ "name": "NeoSkies TestMod (Client Edition)",
+ "description": "The testmod for NeoSkies (client edition) | Also includes devtools",
+ "authors": [
+ "Awakened Redstone"
+ ],
+ "license": "LGPL-3.0",
+ "environment": "client",
+ "mixins": [
+ "neoskies.test.client.mixins.json"
+ ],
+ "entrypoints": {
+ "client": [
+ "com.awakenedredstone.neoskies.test.NeoSkiesTestClientMain"
+ ],
+ "imgui": [
+ "com.awakenedredstone.neoskies.test.DebugDearImGuiEntrypoint"
+ ]
+ }
+}
diff --git a/src/testmodClient/resources/neoskies.test.client.mixins.json b/src/testmodClient/resources/neoskies.test.client.mixins.json
new file mode 100644
index 0000000..2f98377
--- /dev/null
+++ b/src/testmodClient/resources/neoskies.test.client.mixins.json
@@ -0,0 +1,11 @@
+{
+ "required": true,
+ "minVersion": "0.8",
+ "package": "com.awakenedredstone.neoskies.test.mixin",
+ "compatibilityLevel": "JAVA_17",
+ "mixins": [
+ ],
+ "injectors": {
+ "defaultRequire": 1
+ }
+}