/* * 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.roomfinder; import com.google.common.collect.Sets; import kr.syeyoung.dungeonsguide.DungeonsGuide; import kr.syeyoung.dungeonsguide.dungeon.DungeonContext; import kr.syeyoung.dungeonsguide.dungeon.MapProcessor; import kr.syeyoung.dungeonsguide.dungeon.data.DungeonRoomInfo; import kr.syeyoung.dungeonsguide.dungeon.doorfinder.DungeonDoor; import kr.syeyoung.dungeonsguide.dungeon.doorfinder.EDungeonDoorType; import kr.syeyoung.dungeonsguide.dungeon.events.DungeonStateChangeEvent; import kr.syeyoung.dungeonsguide.dungeon.mechanics.DungeonMechanic; import kr.syeyoung.dungeonsguide.dungeon.mechanics.DungeonRoomDoor; import kr.syeyoung.dungeonsguide.features.FeatureRegistry; import kr.syeyoung.dungeonsguide.pathfinding.CachedWorld; import kr.syeyoung.dungeonsguide.pathfinding.JPSPathfinder; import kr.syeyoung.dungeonsguide.pathfinding.NodeProcessorDungeonRoom; import kr.syeyoung.dungeonsguide.roomedit.EditingContext; import kr.syeyoung.dungeonsguide.roomprocessor.ProcessorFactory; import kr.syeyoung.dungeonsguide.roomprocessor.RoomProcessor; import kr.syeyoung.dungeonsguide.roomprocessor.RoomProcessorGenerator; import lombok.AllArgsConstructor; import lombok.Getter; import lombok.Setter; import net.minecraft.block.Block; import net.minecraft.block.state.IBlockState; import net.minecraft.entity.Entity; import net.minecraft.pathfinding.PathEntity; import net.minecraft.pathfinding.PathFinder; import net.minecraft.pathfinding.PathPoint; import net.minecraft.util.*; import net.minecraft.world.ChunkCache; import net.minecraft.world.IBlockAccess; import net.minecraft.world.World; import javax.vecmath.Vector2d; import java.awt.*; import java.util.*; import java.util.List; import java.util.concurrent.Executors; import java.util.concurrent.ScheduledExecutorService; import java.util.concurrent.ScheduledFuture; import java.util.concurrent.TimeUnit; @Getter public class DungeonRoom { private final List unitPoints; private final short shape; private final byte color; private final BlockPos min; private final BlockPos max; private final Point minRoomPt; private final DungeonContext context; private final List doors = new ArrayList(); private DungeonRoomInfo dungeonRoomInfo; private final int unitWidth; // X private final int unitHeight; // Z @Setter private int totalSecrets = -1; private RoomState currentState = RoomState.DISCOVERED; private Map cached = null; @Getter private World cachedWorld; public Map getMechanics() { if (cached == null || EditingContext.getEditingContext() != null) { cached = new HashMap(dungeonRoomInfo.getMechanics()); int index = 0; for (DungeonDoor door : doors) { if (door.getType().isExist()) cached.put((door.getType().getName())+"-"+(++index), new DungeonRoomDoor(this, door)); } } return cached; } public void setCurrentState(RoomState currentState) { context.createEvent(new DungeonStateChangeEvent(unitPoints.get(0), dungeonRoomInfo.getName(), this.currentState, currentState)); this.currentState = currentState; } public ScheduledFuture> createEntityPathTo(IBlockAccess blockaccess, Entity entityIn, BlockPos targetPos, float dist, int timeout) { if (FeatureRegistry.SECRET_PATHFIND_STRATEGY.isEnabled()) { ScheduledFuture> sf = asyncPathFinder.schedule(() -> { BlockPos min = new BlockPos(getMin().getX(), 0, getMin().getZ()); BlockPos max= new BlockPos(getMax().getX(), 255, getMax().getZ()); JPSPathfinder pathFinder = new JPSPathfinder(this); pathFinder.pathfind(entityIn.getPositionVector(), new Vec3(targetPos).addVector(0.5, 0.5, 0.5), 1.5f,timeout); return pathFinder.getRoute(); }, 0, TimeUnit.MILLISECONDS); return sf; } else { ScheduledFuture> sf = asyncPathFinder.schedule(() -> { PathFinder pathFinder = new PathFinder(nodeProcessorDungeonRoom); PathEntity latest = pathFinder.createEntityPathTo(blockaccess, entityIn, targetPos, dist); if (latest != null) { List poses = new ArrayList<>(); for (int i = 0; i < latest.getCurrentPathLength(); i++) { PathPoint pathPoint = latest.getPathPointFromIndex(i); poses.add(new Vec3(getMin().add(pathPoint.xCoord, pathPoint.yCoord, pathPoint.zCoord)).addVector(0.5,0.5,0.5)); } return poses; } return new ArrayList<>(); }, 0, TimeUnit.MILLISECONDS); return sf; } } private static final ScheduledExecutorService asyncPathFinder = Executors.newScheduledThreadPool(4); @Getter private final NodeProcessorDungeonRoom nodeProcessorDungeonRoom; @Getter private final Map roomContext = new HashMap(); @AllArgsConstructor @Getter public enum RoomState { DISCOVERED(0), COMPLETE_WITHOUT_SECRETS(0), FINISHED(0), FAILED(-14); private final int scoreModifier; } private RoomProcessor roomProcessor; public DungeonRoom(List points, short shape, byte color, BlockPos min, BlockPos max, DungeonContext context, Set> doorsAndStates) { this.unitPoints = points; this.shape = shape; this.color = color; this.min = min; this.max = max; this.context = context; minRoomPt = new Point(Integer.MAX_VALUE, Integer.MAX_VALUE); for (Point pt : unitPoints) { if (pt.x < minRoomPt.x) minRoomPt.x = pt.x; if (pt.y < minRoomPt.y) minRoomPt.y = pt.y; } unitWidth = (int) Math.ceil(max.getX() - min.getX() / 32.0); unitHeight = (int) Math.ceil(max.getZ() - min.getZ() / 32.0); ChunkCache chunkCache = new ChunkCache(getContext().getWorld(), min.add(-3, 0, -3), max.add(3,0,3), 0); this.cachedWorld = new CachedWorld(chunkCache); minx = min.getX() * 2; miny = 0; minz = min.getZ() * 2; maxx = max.getX() * 2 + 2; maxy = 255 * 2 + 2; maxz = max.getZ() * 2 + 2; lenx = maxx - minx; leny = maxy - miny; lenz = maxz - minz; arr = new long[lenx *leny * lenz * 2 / 8];; buildDoors(doorsAndStates); buildRoom(); nodeProcessorDungeonRoom = new NodeProcessorDungeonRoom(this); updateRoomProcessor(); } private static final Set directions = Sets.newHashSet(new Vector2d(0,16), new Vector2d(0, -16), new Vector2d(16, 0), new Vector2d(-16 , 0)); private void buildDoors(Set> doorsAndStates) { Set> positions = new HashSet<>(); BlockPos pos = context.getMapProcessor().roomPointToWorldPoint(minRoomPt).add(16,0,16); for (Tuple doorsAndState : doorsAndStates) { Vector2d vector2d = doorsAndState.getFirst(); BlockPos neu = pos.add(vector2d.x * 32, 0, vector2d.y * 32); positions.add(new Tuple<>(neu, doorsAndState.getSecond())); } for (Tuple door : positions) { doors.add(new DungeonDoor(context.getWorld(), door.getFirst(), door.getSecond())); } } private RoomMatcher roomMatcher = null; private void buildRoom() { if (roomMatcher == null) roomMatcher = new RoomMatcher(this); DungeonRoomInfo dungeonRoomInfo = roomMatcher.match(); if (dungeonRoomInfo == null) { dungeonRoomInfo = roomMatcher.createNew(); if (color == 18) dungeonRoomInfo.setProcessorId("bossroom"); } this.dungeonRoomInfo = dungeonRoomInfo; totalSecrets = dungeonRoomInfo.getTotalSecrets(); } public void updateRoomProcessor() { RoomProcessorGenerator roomProcessorGenerator = ProcessorFactory.getRoomProcessorGenerator(dungeonRoomInfo.getProcessorId()); if (roomProcessorGenerator == null) this.roomProcessor = null; else this.roomProcessor = roomProcessorGenerator.createNew(this); } public Block getAbsoluteBlockAt(int x, int y, int z) { // validate x y z's BlockPos pos = new BlockPos(x,y,z); if (canAccessAbsolute(pos)) { return cachedWorld.getBlockState(pos).getBlock(); } return null; } public Block getRelativeBlockAt(int x, int y, int z) { // validate x y z's if (canAccessRelative(x,z)) { BlockPos pos = new BlockPos(x,y,z).add(min.getX(),min.getY(),min.getZ()); return cachedWorld.getBlockState(pos).getBlock(); } return null; } public BlockPos getRelativeBlockPosAt(int x, int y, int z) { BlockPos pos = new BlockPos(x,y,z).add(min.getX(),min.getY(),min.getZ()); return pos; } public int getRelativeBlockDataAt(int x, int y, int z) { // validate x y z's if (canAccessRelative(x,z)) { BlockPos pos = new BlockPos(x,y,z).add(min.getX(),min.getY(),min.getZ()); IBlockState iBlockState = cachedWorld.getBlockState(pos); return iBlockState.getBlock().getMetaFromState(iBlockState); } return -1; } public int getAbsoluteBlockDataAt(int x, int y, int z) { // validate x y z's BlockPos pos = new BlockPos(x,y,z); if (canAccessAbsolute(pos)) { IBlockState iBlockState = cachedWorld.getBlockState(pos); return iBlockState.getBlock().getMetaFromState(iBlockState); } return -1; } public boolean canAccessAbsolute(BlockPos pos) { MapProcessor mapProcessor = this.context.getMapProcessor(); Point roomPt = mapProcessor.worldPointToRoomPoint(pos); roomPt.translate(-minRoomPt.x, -minRoomPt.y); return (shape >>(roomPt.y *4 +roomPt.x) & 0x1) > 0; } public boolean canAccessRelative(int x, int z) { return x>= 0 && z >= 0 && (shape >>((z/32) *4 +(x/32)) & 0x1) > 0; } long arr[]; private final int minx, miny, minz, maxx, maxy, maxz; private final int lenx, leny, lenz; private static final float playerWidth = 0.3f; public boolean isBlocked(int x,int y, int z) { if (x < minx || z < minz || x >= maxx || z >= maxz || y < miny || y >= maxy) return true; int dx = x - minx, dy = y - miny, dz = z - minz; int bitIdx = dx * leny * lenz + dy * lenz + dz; int location = bitIdx / 4; int bitStart = (2 * (bitIdx % 4)); long theBit = arr[location]; if (((theBit >> bitStart) & 0x2) > 0) return ((theBit >> bitStart) & 1) > 0; float wX = x / 2.0f, wY = y / 2.0f, wZ = z / 2.0f; AxisAlignedBB bb = AxisAlignedBB.fromBounds(wX - playerWidth, wY, wZ - playerWidth, wX + playerWidth, wY + 1.9f, wZ + playerWidth); int i = MathHelper.floor_double(bb.minX); int j = MathHelper.floor_double(bb.maxX + 1.0D); int k = MathHelper.floor_double(bb.minY); int l = MathHelper.floor_double(bb.maxY + 1.0D); int i1 = MathHelper.floor_double(bb.minZ); int j1 = MathHelper.floor_double(bb.maxZ + 1.0D); BlockPos.MutableBlockPos blockPos = new BlockPos.MutableBlockPos(); List list = new ArrayList<>(); for (int k1 = i; k1 < j; ++k1) { for (int l1 = i1; l1 < j1; ++l1) { for (int i2 = k - 1; i2 < l; ++i2) { blockPos.set(k1, i2, l1); IBlockState iblockstate1 = cachedWorld.getBlockState(blockPos); Block b = iblockstate1.getBlock(); if (!b.getMaterial().blocksMovement())continue; if (b.isFullCube() && i2 == k-1) continue; if (iblockstate1.equals( NodeProcessorDungeonRoom.preBuilt)) continue; if (b.isFullCube()) { theBit |= (3L << bitStart); arr[location] = theBit; return true; } try { b.addCollisionBoxesToList(cachedWorld, blockPos, iblockstate1, bb, list, null); } catch (Exception e) { return true; } if (list.size() > 0) { theBit |= (3L << bitStart); arr[location] = theBit; return true; } } } } theBit |= 2L << bitStart; arr[location] = theBit; return false; } public void resetBlock(BlockPos pos) { for (int x = -1; x <= 1; x++) { for (int y = -1; y <= 1; y++) { for (int z = -1; z <= 1; z++) { resetBlock(pos.getX()*2 + x, pos.getY()*2 + y, pos.getZ()*2 + z); } } } } private void resetBlock(int x, int y, int z) { if (x < minx || z < minz || x >= maxx || z >= maxz || y < miny || y >= maxy) return; int dx = x - minx, dy = y - miny, dz = z - minz; int bitIdx = dx * leny * lenz + dy * lenz + dz; int location = bitIdx / 4; arr[location] = 0; } }