aboutsummaryrefslogtreecommitdiff
path: root/src/main/java/me/xmrvizzy/skyblocker/skyblock/spidersden/Relics.java
blob: 12ce07150d0ac98eb85a03b1a0ea95fe3efedabe (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
package me.xmrvizzy.skyblocker.skyblock.spidersden;

import com.google.gson.JsonArray;
import com.google.gson.JsonElement;
import com.google.gson.JsonObject;
import com.google.gson.JsonParser;
import com.mojang.brigadier.CommandDispatcher;
import me.xmrvizzy.skyblocker.SkyblockerMod;
import me.xmrvizzy.skyblocker.config.SkyblockerConfig;
import me.xmrvizzy.skyblocker.utils.PosUtils;
import me.xmrvizzy.skyblocker.utils.Utils;
import me.xmrvizzy.skyblocker.utils.render.RenderHelper;
import net.fabricmc.fabric.api.client.command.v2.ClientCommandRegistrationCallback;
import net.fabricmc.fabric.api.client.command.v2.FabricClientCommandSource;
import net.fabricmc.fabric.api.client.event.lifecycle.v1.ClientLifecycleEvents;
import net.fabricmc.fabric.api.client.message.v1.ClientReceiveMessageEvents;
import net.fabricmc.fabric.api.client.rendering.v1.WorldRenderContext;
import net.fabricmc.fabric.api.client.rendering.v1.WorldRenderEvents;
import net.minecraft.client.MinecraftClient;
import net.minecraft.command.CommandRegistryAccess;
import net.minecraft.entity.player.PlayerEntity;
import net.minecraft.text.Text;
import net.minecraft.util.DyeColor;
import net.minecraft.util.Identifier;
import net.minecraft.util.math.BlockPos;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.io.*;
import java.util.*;
import java.util.concurrent.CompletableFuture;

import static net.fabricmc.fabric.api.client.command.v2.ClientCommandManager.literal;

public class Relics {
    private static final Logger LOGGER = LoggerFactory.getLogger(Relics.class);
    private static CompletableFuture<Void> relicsLoaded;
    @SuppressWarnings({"unused", "FieldCanBeLocal"})
    private static int totalRelics = 0;
    private static final List<BlockPos> relics = new ArrayList<>();
    private static final Map<String, Set<BlockPos>> foundRelics = new HashMap<>();

    public static void init() {
        ClientLifecycleEvents.CLIENT_STARTED.register(Relics::loadRelics);
        ClientLifecycleEvents.CLIENT_STOPPING.register(Relics::saveFoundRelics);
        ClientCommandRegistrationCallback.EVENT.register(Relics::registerCommands);
        WorldRenderEvents.AFTER_TRANSLUCENT.register(Relics::render);
        ClientReceiveMessageEvents.GAME.register(Relics::onChatMessage);
    }

    private static void loadRelics(MinecraftClient client) {
        relicsLoaded = CompletableFuture.runAsync(() -> {
            try (BufferedReader reader = client.getResourceManager().openAsReader(new Identifier(SkyblockerMod.NAMESPACE, "spidersden/relics.json"))) {
                for (Map.Entry<String, JsonElement> json : JsonParser.parseReader(reader).getAsJsonObject().asMap().entrySet()) {
                    if (json.getKey().equals("total")) {
                        totalRelics = json.getValue().getAsInt();
                    } else if (json.getKey().equals("locations")) {
                        for (JsonElement locationJson : json.getValue().getAsJsonArray().asList()) {
                            JsonObject posData = locationJson.getAsJsonObject();
                            relics.add(new BlockPos(posData.get("x").getAsInt(), posData.get("y").getAsInt(), posData.get("z").getAsInt()));
                        }
                    }
                }
                LOGGER.info("[Skyblocker] Loaded relics locations");
            } catch (IOException e) {
                LOGGER.error("[Skyblocker] Failed to load relics locations", e);
            }

            try (BufferedReader reader = new BufferedReader(new FileReader(SkyblockerMod.CONFIG_DIR.resolve("found_relics.json").toFile()))) {
                for (Map.Entry<String, JsonElement> profileJson : JsonParser.parseReader(reader).getAsJsonObject().asMap().entrySet()) {
                    Set<BlockPos> foundRelicsForProfile = new HashSet<>();
                    for (JsonElement foundRelicsJson : profileJson.getValue().getAsJsonArray().asList()) {
                        foundRelicsForProfile.add(PosUtils.parsePosString(foundRelicsJson.getAsString()));
                    }
                    foundRelics.put(profileJson.getKey(), foundRelicsForProfile);
                }
                LOGGER.debug("[Skyblocker] Loaded found relics");
            } catch (FileNotFoundException ignored) {
            } catch (IOException e) {
                LOGGER.error("[Skyblocker] Failed to load found relics", e);
            }
        });
    }

    private static void saveFoundRelics(MinecraftClient client) {
        try (BufferedWriter writer = new BufferedWriter(new FileWriter(SkyblockerMod.CONFIG_DIR.resolve("found_relics.json").toFile()))) {
            JsonObject json = new JsonObject();
            for (Map.Entry<String, Set<BlockPos>> foundRelicsForProfile : foundRelics.entrySet()) {
                JsonArray foundRelicsJson = new JsonArray();
                for (BlockPos foundRelic : foundRelicsForProfile.getValue()) {
                    foundRelicsJson.add(PosUtils.getPosString(foundRelic));
                }
                json.add(foundRelicsForProfile.getKey(), foundRelicsJson);
            }
            SkyblockerMod.GSON.toJson(json, writer);
            LOGGER.debug("[Skyblocker] Saved found relics");
        } catch (IOException e) {
            LOGGER.error("[Skyblocker] Failed to write found relics to file", e);
        }
    }

    private static void registerCommands(CommandDispatcher<FabricClientCommandSource> dispatcher, CommandRegistryAccess registryAccess) {
        dispatcher.register(literal(SkyblockerMod.NAMESPACE)
                .then(literal("relics")
                        .then(literal("markAllFound").executes(context -> {
                            Relics.markAllFound();
                            context.getSource().sendFeedback(Text.translatable("skyblocker.relics.markAllFound"));
                            return 1;
                        }))
                        .then(literal("markAllMissing").executes(context -> {
                            Relics.markAllMissing();
                            context.getSource().sendFeedback(Text.translatable("skyblocker.relics.markAllMissing"));
                            return 1;
                        }))));
    }

    private static void render(WorldRenderContext context) {
        SkyblockerConfig.Relics config = SkyblockerConfig.get().locations.spidersDen.relics;

        if (config.enableRelicsHelper && relicsLoaded.isDone() && Utils.getLocationRaw().equals("combat_1")) {
            for (BlockPos fairySoulPos : relics) {
                boolean isRelicMissing = isRelicMissing(fairySoulPos);
                if (!isRelicMissing && !config.highlightFoundRelics) continue;
                float[] colorComponents = isRelicMissing ? DyeColor.YELLOW.getColorComponents() : DyeColor.BROWN.getColorComponents();
                RenderHelper.renderFilledThroughWallsWithBeaconBeam(context, fairySoulPos, colorComponents, 0.5F);
            }
        }
    }

    private static void onChatMessage(Text text, boolean overlay) {
        String message = text.getString();
        if (message.equals("You've already found this relic!") || message.startsWith("+10,000 Coins! (") && message.endsWith("/28 Relics)")) {
            markClosestRelicFound();
        }
    }

    private static void markClosestRelicFound() {
        if (!relicsLoaded.isDone()) return;
        PlayerEntity player = MinecraftClient.getInstance().player;
        if (player == null) {
            LOGGER.warn("[Skyblocker] Failed to mark closest relic as found because player is null");
            return;
        }
        relics.stream()
                .filter(Relics::isRelicMissing)
                .min(Comparator.comparingDouble(relicPos -> relicPos.getSquaredDistance(player.getPos())))
                .filter(relicPos -> relicPos.getSquaredDistance(player.getPos()) <= 16)
                .ifPresent(relicPos -> {
                    foundRelics.computeIfAbsent(Utils.getProfile(), profileKey -> new HashSet<>());
                    foundRelics.get(Utils.getProfile()).add(relicPos);
                });
    }

    private static boolean isRelicMissing(BlockPos relicPos) {
        Set<BlockPos> foundRelicsForProfile = foundRelics.get(Utils.getProfile());
        return foundRelicsForProfile == null || !foundRelicsForProfile.contains(relicPos);
    }

    private static void markAllFound() {
        foundRelics.computeIfAbsent(Utils.getProfile(), profileKey -> new HashSet<>());
        foundRelics.get(Utils.getProfile()).addAll(relics);
    }

    private static void markAllMissing() {
        Set<BlockPos> foundRelicsForProfile = foundRelics.get(Utils.getProfile());
        if (foundRelicsForProfile != null) {
            foundRelicsForProfile.clear();
        }
    }
}