/* * Dungeons Guide - The most intelligent Hypixel Skyblock Dungeons Mod * Copyright (C) 2021 cyoung06 * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License as published * by the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Affero General Public License for more details. * * You should have received a copy of the GNU Affero General Public License * along with this program. If not, see . */ package kr.syeyoung.dungeonsguide.dungeon; import com.google.common.collect.BiMap; import com.google.common.collect.HashBiMap; import com.google.common.collect.Sets; import kr.syeyoung.dungeonsguide.DungeonsGuide; import kr.syeyoung.dungeonsguide.dungeon.doorfinder.EDungeonDoorType; import kr.syeyoung.dungeonsguide.dungeon.events.DungeonMapUpdateEvent; import kr.syeyoung.dungeonsguide.dungeon.events.DungeonNodataEvent; import kr.syeyoung.dungeonsguide.dungeon.events.DungeonRoomDiscoverEvent; import kr.syeyoung.dungeonsguide.dungeon.events.SerializableBlockPos; import kr.syeyoung.dungeonsguide.dungeon.roomfinder.DungeonRoom; import kr.syeyoung.dungeonsguide.dungeon.doorfinder.DungeonSpecificDataProviderRegistry; import kr.syeyoung.dungeonsguide.dungeon.doorfinder.DungeonSpecificDataProvider; import kr.syeyoung.dungeonsguide.events.DungeonContextInitializationEvent; import kr.syeyoung.dungeonsguide.features.FeatureRegistry; import kr.syeyoung.dungeonsguide.stomp.StompPayload; import kr.syeyoung.dungeonsguide.utils.MapUtils; import kr.syeyoung.dungeonsguide.wsresource.StaticResource; import kr.syeyoung.dungeonsguide.wsresource.StaticResourceCache; import lombok.Getter; import lombok.Setter; import net.minecraft.client.Minecraft; import net.minecraft.entity.player.EntityPlayer; import net.minecraft.item.ItemMap; import net.minecraft.item.ItemStack; import net.minecraft.util.*; import net.minecraft.world.storage.MapData; import net.minecraftforge.common.MinecraftForge; import org.json.JSONObject; import javax.vecmath.Vector2d; import javax.vecmath.Vector2f; import java.awt.*; import java.util.*; import java.util.List; import java.util.concurrent.ExecutionException; public class MapProcessor { private final DungeonContext context; private byte[] lastMapData; @Getter private MapData lastMapData2; @Getter private final BiMap mapIconToPlayerMap = HashBiMap.create(); @Getter @Setter private Dimension unitRoomDimension; @Getter @Setter private Dimension doorDimension; // width: width of door, height: gap between rooms @Getter @Setter private Point topLeftMapPoint; @Setter private boolean bugged = false; private final List roomsFound = new ArrayList(); private boolean axisMatch = false; @Getter private boolean initialized = false; @Getter private int undiscoveredRoom = 0; public MapProcessor(DungeonContext context) { this.context = context; } private static final Set directions = Sets.newHashSet(new Vector2d(0,1), new Vector2d(0, -1), new Vector2d(1, 0), new Vector2d(-1 , 0)); private int waitCnt = 0; private void buildMap(final byte[] mapData) { final Point startroom = MapUtils.findFirstColorWithIn(mapData, (byte) 30, new Rectangle(0,0,128,128)); if (startroom == null){ Minecraft.getMinecraft().thePlayer.addChatMessage(new ChatComponentText("§eDungeons Guide §7:: §cCouldn't retrieve map data, disabling mod for this dungeon run")); bugged = true; return; } // Determine room dimension { int width = MapUtils.getWidthOfColorAt(mapData, (byte) 30, startroom); int height = MapUtils.getHeightOfColorAt(mapData, (byte) 30, startroom); unitRoomDimension = new Dimension(width, height); } Vector2d doorDir = null; // determine the gap { Point midStartRoom = new Point(startroom.x + unitRoomDimension.width / 2, startroom.y +unitRoomDimension.height / 2); final int halfWidth = unitRoomDimension.width / 2 + 2; for (Vector2d v:directions) { byte color = MapUtils.getMapColorAt(mapData, (int)(v.x * halfWidth +midStartRoom.x), (int)(v.y *halfWidth +midStartRoom.y)); if (color != 0) { doorDir = v; break; } } if (doorDir == null) { Minecraft.getMinecraft().thePlayer.addChatMessage(new ChatComponentText("§eDungeons Guide §7:: §cCouldn't determine door of the room, disabling mod for this dungeon run")); bugged = true; return; } Point basePoint = new Point(startroom.x, startroom.y); if (doorDir.x > 0) basePoint.x += unitRoomDimension.width; if (doorDir.x < 0) basePoint.x += -1; if (doorDir.y > 0) basePoint.y += unitRoomDimension.height; if (doorDir.y < 0) basePoint.y += -1; int gap = MapUtils.getLengthOfColorExtending(mapData, (byte) 0, basePoint, doorDir); Point pt = MapUtils.findFirstColorWithInNegate(mapData, (byte)0, new Rectangle(basePoint.x, basePoint.y, (int)Math.abs(doorDir.y) * unitRoomDimension.width + 1, (int)Math.abs(doorDir.x) * unitRoomDimension.height + 1)); if (pt == null) { Minecraft.getMinecraft().thePlayer.addChatMessage(new ChatComponentText("§eDungeons Guide §7:: §cCouldn't determine door of the room, disabling mod for this dungeon run")); bugged = true; return; } int doorWidth = MapUtils.getLengthOfColorExtending(mapData, MapUtils.getMapColorAt(mapData, pt.x, pt.y), pt, new Vector2d((int)Math.abs(doorDir.y), (int)Math.abs(doorDir.x))); doorDimension = new Dimension(doorWidth, gap); } // Determine Top Left { int x = startroom.x; int y = startroom.y; while (x >= unitRoomDimension.width + doorDimension.height) x -= unitRoomDimension.width + doorDimension.height; while (y >= unitRoomDimension.height + doorDimension.height) y -= unitRoomDimension.height + doorDimension.height; topLeftMapPoint = new Point(x, y); } // determine door location based on npc, and determine map min from there { DungeonSpecificDataProvider doorFinder = DungeonSpecificDataProviderRegistry.getDoorFinder(DungeonsGuide.getDungeonsGuide().getSkyblockStatus().getDungeonName()); if (doorFinder == null) { Minecraft.getMinecraft().thePlayer.addChatMessage(new ChatComponentText("§eDungeons Guide §7:: §cCouldn't find door processor for "+ DungeonsGuide.getDungeonsGuide().getSkyblockStatus().getDungeonName())); bugged = true; return; } BlockPos door = doorFinder.findDoor(context.getWorld(), DungeonsGuide.getDungeonsGuide().getSkyblockStatus().getDungeonName()); if (door == null) { Minecraft.getMinecraft().thePlayer.addChatMessage(new ChatComponentText("§eDungeons Guide §7:: §cCouldn't determine door of the room, disabling mod for this dungeon run")); bugged = true; return; } DungeonsGuide.sendDebugChat(new ChatComponentText("door Pos:"+door)); Point unitPoint = mapPointToRoomPoint(startroom); unitPoint.translate(unitPoint.x + 1, unitPoint.y + 1); unitPoint.translate((int)doorDir.x, (int)doorDir.y); Vector2d offset = doorFinder.findDoorOffset(context.getWorld(), DungeonsGuide.getDungeonsGuide().getSkyblockStatus().getDungeonName()); axisMatch = doorDir.equals(offset); int worldX = unitPoint.x * 16; int worldY = unitPoint.y * 16; BlockPos worldMin = door.add(-worldX, 0, -worldY); context.setDungeonMin(worldMin); } DungeonsGuide.sendDebugChat(new ChatComponentText("Found Green room:"+startroom)); DungeonsGuide.sendDebugChat(new ChatComponentText("Axis match:"+axisMatch)); DungeonsGuide.sendDebugChat(new ChatComponentText("World Min:"+context.getDungeonMin())); DungeonsGuide.sendDebugChat(new ChatComponentText("Dimension:"+unitRoomDimension)); DungeonsGuide.sendDebugChat(new ChatComponentText("top Left:"+topLeftMapPoint)); DungeonsGuide.sendDebugChat(new ChatComponentText("door dimension:"+doorDimension)); context.createEvent(new DungeonNodataEvent("MAP_PROCESSOR_INIT")); initialized = true; MinecraftForge.EVENT_BUS.post(new DungeonContextInitializationEvent()); } public Point mapPointToRoomPoint(Point mapPoint) { int x = (int)((mapPoint.x - topLeftMapPoint.x) / ((double)unitRoomDimension.width + doorDimension.height)); int y = (int)((mapPoint.y - topLeftMapPoint.y) / ((double)unitRoomDimension.height + doorDimension.height)); return new Point(x,y); } public BlockPos mapPointToWorldPoint(Point mapPoint) { int x = (int)((mapPoint.x - topLeftMapPoint.x) / ((double)unitRoomDimension.width + doorDimension.height) * 32 + context.getDungeonMin().getX()); int y = (int)((mapPoint.y - topLeftMapPoint.y) / ((double)unitRoomDimension.height + doorDimension.height) * 32 + context.getDungeonMin().getZ()); return new BlockPos(x,70,y); } public Point roomPointToMapPoint(Point roomPoint) { return new Point(roomPoint.x * (unitRoomDimension.width +doorDimension.height) + topLeftMapPoint.x, roomPoint.y *(unitRoomDimension.height + doorDimension.height) + topLeftMapPoint.y); } public BlockPos roomPointToWorldPoint(Point roomPoint) { return new BlockPos(context.getDungeonMin().getX() +(roomPoint.x * 32), context.getDungeonMin().getY(), context.getDungeonMin().getZ() +(roomPoint.y *32)); } public Point worldPointToRoomPoint(BlockPos worldPoint) { if (context.getDungeonMin() == null) return null; return new Point((worldPoint.getX() - context.getDungeonMin().getX()) / 32, (worldPoint.getZ() - context.getDungeonMin().getZ()) / 32); } public Point worldPointToMapPoint(Vec3 worldPoint) { if (context.getDungeonMin() == null) return null; return new Point(topLeftMapPoint.x + (int)((worldPoint.xCoord - context.getDungeonMin().getX()) / 32.0f * (unitRoomDimension.width + doorDimension.height)), topLeftMapPoint.y + (int)((worldPoint.zCoord - context.getDungeonMin().getZ()) / 32.0f * (unitRoomDimension.height + doorDimension.height))); } private void processMap(byte[] mapData) { int height = (int)((128.0 - topLeftMapPoint.y) / (unitRoomDimension.height + doorDimension.height)); int width = (int) ((128.0 - topLeftMapPoint.x) / (unitRoomDimension.width + doorDimension.height)); undiscoveredRoom = 0; if (MapUtils.getMapColorAt(mapData,0,0) != 0) return; for (int y = 0; y <= height; y++){ for (int x = 0; x <= width; x++) { Point mapPoint = roomPointToMapPoint(new Point(x,y)); byte color = MapUtils.getMapColorAt(mapData, mapPoint.x, mapPoint.y); MapUtils.record(mapData, mapPoint.x, mapPoint.y, new Color(255,255,0,80)); if (roomsFound.contains(new Point(x,y))) { DungeonRoom dungeonRoom = context.getRoomMapper().get(new Point(x,y)); if (color == 18 && dungeonRoom.getCurrentState() != DungeonRoom.RoomState.FINISHED) { dungeonRoom.setCurrentState(DungeonRoom.RoomState.COMPLETE_WITHOUT_SECRETS); dungeonRoom.setTotalSecrets(0); } else if (color == 30) { dungeonRoom.setCurrentState(DungeonRoom.RoomState.FINISHED); dungeonRoom.setTotalSecrets(0); } else if (dungeonRoom.getCurrentState() != DungeonRoom.RoomState.FINISHED){ byte centerColor = MapUtils.getMapColorAt(mapData, mapPoint.x + unitRoomDimension.width / 2, mapPoint.y + unitRoomDimension.height / 2); MapUtils.record(mapData, mapPoint.x + unitRoomDimension.width / 2, mapPoint.y + unitRoomDimension.height / 2, new Color(0,255,0,80)); if (centerColor == 34) { dungeonRoom.setCurrentState(DungeonRoom.RoomState.COMPLETE_WITHOUT_SECRETS); } else if (centerColor == 30) { dungeonRoom.setCurrentState(DungeonRoom.RoomState.FINISHED); } else if (centerColor == 18) { // red dungeonRoom.setCurrentState(DungeonRoom.RoomState.FAILED); } } if (dungeonRoom.getTotalSecrets() == -1) { if (dungeonRoom.getColor() == 82) dungeonRoom.setTotalSecrets(0); else if (dungeonRoom.getColor() == 74) dungeonRoom.setTotalSecrets(0); MapUtils.record(mapData, mapPoint.x, mapPoint.y +1, new Color(0,255,0,80)); } continue; } if (color != 0 && color != 85) { MapUtils.record(mapData, mapPoint.x, mapPoint.y, new Color(0,255,255,80)); DungeonRoom rooms = buildRoom(mapData, new Point(x,y)); if (rooms == null) continue; context.createEvent(new DungeonRoomDiscoverEvent(rooms.getUnitPoints().get(0), rooms.getRoomMatcher().getRotation(), new SerializableBlockPos(rooms.getMin()), new SerializableBlockPos(rooms.getMax()), rooms.getShape(),rooms.getColor(), rooms.getDungeonRoomInfo().getUuid(), rooms.getDungeonRoomInfo().getName(), rooms.getDungeonRoomInfo().getProcessorId())); DungeonsGuide.sendDebugChat(new ChatComponentText("New Map discovered! shape: "+rooms.getShape()+ " color: "+rooms.getColor()+" unitPos: "+x+","+y)); DungeonsGuide.sendDebugChat(new ChatComponentText("New Map discovered! mapMin: "+rooms.getMin() + " mapMx: "+rooms.getMax())); StringBuilder builder = new StringBuilder(); for (int dy =0;dy<4;dy++) { builder.append("\n"); for (int dx = 0; dx < 4; dx ++) { boolean isSet = ((rooms.getShape() >> (dy * 4 + dx)) & 0x1) != 0; builder.append(isSet ? "O" : "X"); } } DungeonsGuide.sendDebugChat(new ChatComponentText("Shape visual: "+ builder)); context.getDungeonRoomList().add(rooms); for (Point p:rooms.getUnitPoints()) { roomsFound.add(p); context.getRoomMapper().put(p, rooms); } if (rooms.getRoomProcessor() != null && rooms.getRoomProcessor().readGlobalChat()) context.getGlobalRoomProcessors().add(rooms.getRoomProcessor()); } else if (color == 85){ undiscoveredRoom++; } } } } private static final Set door_dirs = Sets.newHashSet(new Vector2d(0,0.5), new Vector2d(0, -0.5), new Vector2d(0.5, 0), new Vector2d(-0.5 , 0)); private DungeonRoom buildRoom(byte[] mapData, Point unitPoint) { Queue toCheck = new LinkedList(); toCheck.add(new Point[] {unitPoint, unitPoint}); // requestor, target Set checked = new HashSet(); List ayConnected = new ArrayList(); int minX = Integer.MAX_VALUE, minY = Integer.MAX_VALUE; int maxX = 0, maxY = 0; while(toCheck.peek() != null) { Point[] check = toCheck.poll(); if (checked.contains(check[1])) continue; checked.add(check[1]); if (checkIfConnected(mapData, check[0], check[1])) { ayConnected.add(check[1]); if (check[1].x < minX) minX = check[1].x; if (check[1].y < minY) minY = check[1].y; if (check[1].x > maxX) maxX = check[1].x; if (check[1].y > maxY) maxY = check[1].y; for (Vector2d dir: directions) { Point newPt = new Point(check[1].x + (int)dir.x, check[1].y +(int)dir.y); toCheck.add(new Point[]{check[1], newPt}); } } } short shape = 0; for (Point p:ayConnected) { int localX = p.x - minX, localY = p.y - minY; shape |= 1 <<(localY *4 + localX); } Set doors = new HashSet<>(); for (Point p: ayConnected) { for (Vector2d v: door_dirs) { Vector2d v2 = new Vector2d(p.x + v.x , p.y + v.y ); if (doors.contains(v2)) doors.remove(v2); else doors.add(v2); } } Point pt2 = roomPointToMapPoint(ayConnected.get(0)); byte unit1 = MapUtils.getMapColorAt(mapData, pt2.x, pt2.y); // 0: none 1: open door door 2. unopen door 3: wither door 4. red door Set> doorsAndStates = new HashSet<>(); final int halfWidth = unitRoomDimension.width + 4; for (Vector2d door : doors) { int floorX = (int)Math.floor(door.x), floorY = (int)Math.floor(door.y); Point mapPt = roomPointToMapPoint(new Point(floorX, floorY)); Point target = new Point(mapPt.x+ unitRoomDimension.width/2 + (int)(halfWidth*(door.x - floorX)), mapPt.y + unitRoomDimension.height/2 + (int)(halfWidth*(door.y - floorY)) ); MapUtils.record(mapData, target.x, target.y, Color.green); byte color = MapUtils.getMapColorAt(mapData, target.x, target.y); Vector2d vector2d = new Vector2d(door.x - minX, door.y - minY); if (color == 0) { doorsAndStates.add(new Tuple<>(vector2d, EDungeonDoorType.NONE)); } else if (color == 85) { doorsAndStates.add(new Tuple<>(vector2d, EDungeonDoorType.UNOPEN)); } else if (color == 119) { doorsAndStates.add(new Tuple<>(vector2d, EDungeonDoorType.WITHER)); } else if (color == 18 && unit1 != 18) { doorsAndStates.add(new Tuple<>(vector2d, EDungeonDoorType.BLOOD)); } else { doorsAndStates.add(new Tuple<>(vector2d, EDungeonDoorType.ENTRANCE)); } } try{ return new DungeonRoom(ayConnected, shape, unit1, roomPointToWorldPoint(new Point(minX, minY)), roomPointToWorldPoint(new Point(maxX+1, maxY+1)).add(-1, 0, -1), context, doorsAndStates); } catch (IllegalStateException ex) { DungeonsGuide.sendDebugChat(new ChatComponentText("Failed to load room, retrying later :: "+ex.getLocalizedMessage())); return null; } } private boolean checkIfConnected(byte[] mapData, Point unitPoint1, Point unitPoint2) { if (unitPoint1 == unitPoint2) return true; if (unitPoint1.equals(unitPoint2)) return true; Point high = (unitPoint2.y > unitPoint1.y) ? unitPoint2 :(unitPoint2.x > unitPoint1.x) ? unitPoint2 : unitPoint1; Point low = high == unitPoint2 ? unitPoint1 : unitPoint2; int xOff = low.x - high.x; int yOff = low.y - high.y; Point pt = roomPointToMapPoint(high); Point pt2 = roomPointToMapPoint(low); byte unit1 = MapUtils.getMapColorAt(mapData, pt.x, pt.y); byte unit2 = MapUtils.getMapColorAt(mapData, pt2.x, pt2.y); pt.translate(xOff, yOff); byte unit3 = MapUtils.getMapColorAt(mapData, pt.x, pt.y); return unit1 == unit2 && unit2 == unit3 && unit1 != 0; } public boolean isThereDifference(byte[] colors1, byte[] colors) { if (colors1 == null || colors == null) return true; for (int i =0; i < colors.length; i++) if (colors[i] != colors1[i]) return true; return false; } private int stabilizationTick = 0; private boolean processed = false; private void processFinishedMap(byte[] mapData) { if (MapUtils.getMapColorAt(mapData, 0, 0) == 0) return; if (processed) return; processed = true; MapUtils.clearMap(); MapUtils.record(mapData, 0, 0, Color.GREEN); int skill = MapUtils.readNumber(mapData, 51, 35, 9); int exp = MapUtils.readNumber(mapData, 51, 54, 9); int time = MapUtils.readNumber(mapData, 51, 73, 9); int bonus = MapUtils.readNumber(mapData, 51, 92, 9); DungeonsGuide.sendDebugChat(new ChatComponentText(("skill: " + skill + " / exp: " + exp + " / time: " + time + " / bonus : " + bonus))); JSONObject payload = new JSONObject().put("timeSB", FeatureRegistry.DUNGEON_SBTIME.getTimeElapsed()) .put("timeR", FeatureRegistry.DUNGEON_REALTIME.getTimeElapsed()) .put("timeScore", time) .put("completed", context.getBossRoomEnterSeconds() != -1) .put("percentage", DungeonsGuide.getDungeonsGuide().getSkyblockStatus().getPercentage() / 100.0) .put("floor", DungeonsGuide.getDungeonsGuide().getSkyblockStatus().getDungeonName()); DungeonsGuide.sendDebugChat(new ChatComponentText(payload.toString())); try { String target = StaticResourceCache.INSTANCE.getResource(StaticResourceCache.DATA_COLLECTION).get().getValue(); if (FeatureRegistry.ETC_COLLECT_SCORE.isEnabled() && !target.contains("falsefalse")) { DungeonsGuide.getDungeonsGuide().getStompConnection().send(new StompPayload().payload(payload.toString().replace("false", "")).header("destination", target.trim())); } } catch (Throwable e) { e.printStackTrace(); } } public void tick() { if (waitCnt < 5) { waitCnt++; return; } if (bugged) return; ItemStack stack = Minecraft.getMinecraft().thePlayer.inventory.getStackInSlot(8); byte[] mapData; if (stack == null || !(stack.getItem() instanceof ItemMap)) { mapData = lastMapData; } else { MapData mapData1 = ((ItemMap)stack.getItem()).getMapData(stack, context.getWorld()); if (mapData1 == null) mapData = lastMapData; else { mapData = mapData1.colors; lastMapData2 = mapData1; if (isThereDifference(lastMapData, mapData)) { stabilizationTick =0; context.createEvent(new DungeonMapUpdateEvent(mapData)); } else { stabilizationTick++; } if (stabilizationTick > 20) { if (doorDimension == null) buildMap(mapData); else processMap(mapData); if (context.isEnded()) { processFinishedMap(mapData); } } lastMapData = mapData; } } if (lastMapData2 != null && mapIconToPlayerMap.size() < context.getPlayers().size() && initialized) { label: for (Map.Entry stringVec4bEntry : lastMapData2.mapDecorations.entrySet()) { if (mapIconToPlayerMap.containsValue(stringVec4bEntry.getKey())) continue; int x = stringVec4bEntry.getValue().func_176112_b() /2 + 64; int y = stringVec4bEntry.getValue().func_176113_c() /2 + 64; BlockPos mapPos = mapPointToWorldPoint(new Point(x, y)); String potentialPlayer = null; for (String player : context.getPlayers()) { // check nearby players if (mapIconToPlayerMap.containsKey(player)) continue; EntityPlayer entityPlayer = Minecraft.getMinecraft().theWorld.getPlayerEntityByName(player); if (entityPlayer == null || entityPlayer.isInvisible()) continue; BlockPos pos = entityPlayer.getPosition(); int dx = mapPos.getX() - pos.getX(); int dz = mapPos.getZ() - pos.getZ(); if (dx * dx + dz * dz < 100) { if (potentialPlayer != null) continue label; potentialPlayer = player; } } if (potentialPlayer == null) continue; for (Map.Entry stringVec4bEntry2 : lastMapData2.mapDecorations.entrySet()) { // check nearby markers if (mapIconToPlayerMap.containsValue(stringVec4bEntry.getKey())) continue; if (stringVec4bEntry.getKey().equals(stringVec4bEntry2.getKey())) continue; int x2 = stringVec4bEntry2.getValue().func_176112_b() /2 + 64; int y2 = stringVec4bEntry2.getValue().func_176113_c() /2 + 64; int dx = x2 - x; int dy = y2 - y; if (dx * dx + dy * dy < 100) { continue label; } } mapIconToPlayerMap.put(potentialPlayer, stringVec4bEntry.getKey()); } } } }