diff --git a/src/main/java/meteordevelopment/meteorclient/systems/modules/combat/BedAura.java b/src/main/java/meteordevelopment/meteorclient/systems/modules/combat/BedAura.java index 366a95cf4c..47d2770558 100644 --- a/src/main/java/meteordevelopment/meteorclient/systems/modules/combat/BedAura.java +++ b/src/main/java/meteordevelopment/meteorclient/systems/modules/combat/BedAura.java @@ -6,24 +6,29 @@ package meteordevelopment.meteorclient.systems.modules.combat; import meteordevelopment.meteorclient.events.render.Render3DEvent; +import meteordevelopment.meteorclient.events.world.BlockUpdateEvent; import meteordevelopment.meteorclient.events.world.TickEvent; +import meteordevelopment.meteorclient.mixin.DirectionAccessor; import meteordevelopment.meteorclient.renderer.ShapeMode; import meteordevelopment.meteorclient.settings.*; import meteordevelopment.meteorclient.systems.modules.Categories; import meteordevelopment.meteorclient.systems.modules.Module; -import meteordevelopment.meteorclient.utils.Utils; +import meteordevelopment.meteorclient.systems.modules.Modules; import meteordevelopment.meteorclient.utils.entity.DamageUtils; import meteordevelopment.meteorclient.utils.entity.EntityUtils; import meteordevelopment.meteorclient.utils.entity.SortPriority; import meteordevelopment.meteorclient.utils.entity.TargetUtils; -import meteordevelopment.meteorclient.utils.player.*; +import meteordevelopment.meteorclient.utils.player.FindItemResult; +import meteordevelopment.meteorclient.utils.player.InvUtils; +import meteordevelopment.meteorclient.utils.player.PlayerUtils; +import meteordevelopment.meteorclient.utils.player.Rotations; import meteordevelopment.meteorclient.utils.render.color.SettingColor; +import meteordevelopment.meteorclient.utils.world.BlockIterator; import meteordevelopment.meteorclient.utils.world.BlockUtils; -import meteordevelopment.meteorclient.utils.world.CardinalDirection; import meteordevelopment.orbit.EventHandler; import net.minecraft.block.BedBlock; -import net.minecraft.block.entity.BedBlockEntity; -import net.minecraft.block.entity.BlockEntity; +import net.minecraft.block.BlockState; +import net.minecraft.block.enums.BedPart; import net.minecraft.entity.player.PlayerEntity; import net.minecraft.item.BedItem; import net.minecraft.util.Hand; @@ -31,22 +36,47 @@ import net.minecraft.util.math.BlockPos; import net.minecraft.util.math.Direction; import net.minecraft.util.math.Vec3d; +import net.minecraft.world.RaycastContext; public class BedAura extends Module { private final SettingGroup sgGeneral = settings.getDefaultGroup(); private final SettingGroup sgTargeting = settings.createGroup("Targeting"); - private final SettingGroup sgAutoMove = settings.createGroup("Inventory"); + private final SettingGroup sgInventory = settings.createGroup("Inventory"); private final SettingGroup sgPause = settings.createGroup("Pause"); private final SettingGroup sgRender = settings.createGroup("Render"); // General - private final Setting delay = sgGeneral.add(new IntSetting.Builder() - .name("delay") - .description("The delay between placing beds in ticks.") - .defaultValue(9) - .min(0) - .sliderMax(20) + private final Setting place = sgGeneral.add(new BoolSetting.Builder() + .name("place") + .description("Allows Bed Aura to place beds.") + .defaultValue(true) + .build() + ); + + private final Setting placeRange = sgGeneral.add(new DoubleSetting.Builder() + .name("place-range") + .description("The range at which beds can be placed.") + .defaultValue(4) + .range(0, 6) + .visible(place::get) + .build() + ); + + private final Setting placeWallsRange = sgGeneral.add(new DoubleSetting.Builder() + .name("walls-range") + .description("Range in which to place beds when behind blocks.") + .defaultValue(4) + .range(0, 6) + .visible(place::get) + .build() + ); + + private final Setting airPlace = sgGeneral.add(new BoolSetting.Builder() + .name("air-place") + .description("Allows Bed Aura to place beds in the air.") + .defaultValue(true) + .visible(place::get) .build() ); @@ -54,32 +84,72 @@ public class BedAura extends Module { .name("strict-direction") .description("Only places beds in the direction you are facing.") .defaultValue(false) + .visible(place::get) .build() ); - // Targeting + private final Setting placeDelay = sgGeneral.add(new IntSetting.Builder() + .name("place-delay") + .description("The delay between placing beds in ticks.") + .defaultValue(10) + .range(0, 20) + .visible(place::get) + .build() + ); - private final Setting targetRange = sgTargeting.add(new DoubleSetting.Builder() - .name("target-range") - .description("The range at which players can be targeted.") - .defaultValue(4) + private final Setting breakDelay = sgGeneral.add(new IntSetting.Builder() + .name("break-delay") + .description("The tick delay between exploding beds.") + .defaultValue(0) .min(0) - .sliderMax(5) + .sliderMax(10) + .build() + ); + + private final Setting spoofPlace = sgGeneral.add(new BoolSetting.Builder() + .name("spoof-place") + .description("Places the entire bed synchronically on your client.") + .defaultValue(true) .build() ); - private final Setting priority = sgTargeting.add(new EnumSetting.Builder() + private final Setting spoofBreak = sgGeneral.add(new BoolSetting.Builder() + .name("spoof-break") + .description("Breaks the entire bed synchronically on your client.") + .defaultValue(true) + .build() + ); + + private final Setting rotate = sgGeneral.add(new BoolSetting.Builder() + .name("rotate") + .description("Rotates server-side towards the beds being broken.") + .defaultValue(true) + .build() + ); + + // Targeting + + private final Setting targetPriority = sgTargeting.add(new EnumSetting.Builder() .name("target-priority") .description("How to filter targets within range.") .defaultValue(SortPriority.LowestHealth) .build() ); + private final Setting targetRange = sgTargeting.add(new DoubleSetting.Builder() + .name("target-range") + .description("Range in which to target players.") + .defaultValue(10) + .min(0) + .sliderMax(16) + .build() + ); + private final Setting minDamage = sgTargeting.add(new DoubleSetting.Builder() .name("min-damage") .description("The minimum damage to inflict on your target.") .defaultValue(7) - .range(0, 36) + .min(0) .sliderMax(36) .build() ); @@ -88,7 +158,7 @@ public class BedAura extends Module { .name("max-self-damage") .description("The maximum damage to inflict on yourself.") .defaultValue(7) - .range(0, 36) + .min(0) .sliderMax(36) .build() ); @@ -100,16 +170,16 @@ public class BedAura extends Module { .build() ); - // Auto move + // Inventory - private final Setting autoMove = sgAutoMove.add(new BoolSetting.Builder() + private final Setting autoMove = sgInventory.add(new BoolSetting.Builder() .name("auto-move") .description("Moves beds into a selected hotbar slot.") .defaultValue(false) .build() ); - private final Setting autoMoveSlot = sgAutoMove.add(new IntSetting.Builder() + private final Setting autoMoveSlot = sgInventory.add(new IntSetting.Builder() .name("auto-move-slot") .description("The slot auto move moves beds to.") .defaultValue(9) @@ -119,32 +189,40 @@ public class BedAura extends Module { .build() ); - private final Setting autoSwitch = sgAutoMove.add(new BoolSetting.Builder() + private final Setting autoSwitch = sgInventory.add(new BoolSetting.Builder() .name("auto-switch") - .description("Switches to and from beds automatically.") + .description("Switches to beds automatically.") .defaultValue(true) .build() ); - // Pause - - private final Setting pauseOnEat = sgPause.add(new BoolSetting.Builder() - .name("pause-on-eat") - .description("Pauses while eating.") + private final Setting swapBack = sgInventory.add(new BoolSetting.Builder() + .name("swap-back") + .description("Switches to your previous slot after using beds.") .defaultValue(true) + .visible(autoSwitch::get) .build() ); - private final Setting pauseOnDrink = sgPause.add(new BoolSetting.Builder() - .name("pause-on-drink") - .description("Pauses while drinking.") + // Pause + + private final Setting pauseOnUse = sgPause.add(new BoolSetting.Builder() + .name("pause-on-use") + .description("Pauses while using an item.") .defaultValue(true) .build() ); private final Setting pauseOnMine = sgPause.add(new BoolSetting.Builder() .name("pause-on-mine") - .description("Pauses while mining.") + .description("Pauses while mining blocks.") + .defaultValue(true) + .build() + ); + + private final Setting pauseOnCA = sgPause.add(new BoolSetting.Builder() + .name("pause-on-CA") + .description("Pauses while Crystal Aura is placing.") .defaultValue(true) .build() ); @@ -169,13 +247,15 @@ public class BedAura extends Module { .name("shape-mode") .description("How the shapes are rendered.") .defaultValue(ShapeMode.Both) + .visible(render::get) .build() ); private final Setting sideColor = sgRender.add(new ColorSetting.Builder() .name("side-color") .description("The side color for positions to be placed.") - .defaultValue(new SettingColor(15, 255, 211,75)) + .defaultValue(new SettingColor(15, 255, 211, 41)) + .visible(() -> render.get() && shapeMode.get().sides()) .build() ); @@ -183,13 +263,21 @@ public class BedAura extends Module { .name("line-color") .description("The line color for positions to be placed.") .defaultValue(new SettingColor(15, 255, 211)) + .visible(() -> render.get() && shapeMode.get().lines()) .build() ); - private CardinalDirection direction; + private double bestPlaceDamage; + private final BlockPos.Mutable bestPlacePos = new BlockPos.Mutable(); + private Direction bestPlaceDirection; + + private double bestBreakDamage; + private final BlockPos.Mutable bestBreakPos = new BlockPos.Mutable(); + + private BlockPos renderBlockPos; + private Direction renderDirection; + private int placeDelayLeft, breakDelayLeft; private PlayerEntity target; - private BlockPos placePos, breakPos; - private int timer; public BedAura() { super(Categories.Combat, "bed-aura", "Automatically places and explodes beds in the Nether and End."); @@ -197,12 +285,19 @@ public BedAura() { @Override public void onActivate() { - timer = delay.get(); - direction = CardinalDirection.North; + renderBlockPos = null; + placeDelayLeft = placeDelay.get(); + breakDelayLeft = 0; + target = null; + } + + @Override + public void onDeactivate() { + renderBlockPos = null; } @EventHandler - private void onTick(TickEvent.Post event) { + private void onTick(TickEvent.Pre event) { // Check if beds can explode here if (mc.world.getDimension().bedWorks()) { error("You can't blow up beds in this dimension, disabling."); @@ -211,138 +306,211 @@ private void onTick(TickEvent.Post event) { } // Pause - if (PlayerUtils.shouldPause(pauseOnMine.get(), pauseOnEat.get(), pauseOnDrink.get())) return; + if (shouldPause()) { + renderBlockPos = null; + return; + } // Find a target - target = TargetUtils.getPlayerTarget(targetRange.get(), priority.get()); - if (target == null) { - placePos = null; - breakPos = null; - return; + if (TargetUtils.isBadTarget(target, targetRange.get())) { + renderBlockPos = null; + target = TargetUtils.getPlayerTarget(targetRange.get(), targetPriority.get()); + if (TargetUtils.isBadTarget(target, targetRange.get())) return; } - // Auto move + // Auto move beds from inventory if (autoMove.get()) { FindItemResult bed = InvUtils.find(itemStack -> itemStack.getItem() instanceof BedItem); - - if (bed.found() && bed.slot() != autoMoveSlot.get() - 1) { - InvUtils.move().from(bed.slot()).toHotbar(autoMoveSlot.get() - 1); + boolean alreadyHasBed = mc.player.getInventory().getStack(autoMoveSlot.get() - 1).getItem() instanceof BedItem; + if (bed.found() && !bed.isHotbar() && !alreadyHasBed) { + InvUtils.quickSwap().fromId(autoMoveSlot.get() - 1).to(bed.slot()); } } - if (breakPos == null) { - placePos = findPlace(target); - } + doBedAura(); + } - // Place bed - if (timer <= 0 && placeBed(placePos)) { - timer = delay.get(); - } - else { - timer--; - } + private void doBedAura() { + bestPlaceDamage = 0; + bestBreakDamage = 0; + + // Find best position to place or break the bed + int iteratorRange = (int) Math.ceil(placeRange.get()); + BlockIterator.register(iteratorRange, iteratorRange, (blockPos, blockState) -> { + if (blockState.getBlock() instanceof BedBlock) { + setBreakInfo(blockPos, blockState); + } else if (place.get()) { + setPlaceInfo(blockPos); + } + }); + + // Break or place + BlockIterator.after(() -> { + renderBlockPos = null; - if (breakPos == null) breakPos = findBreak(); - breakBed(breakPos); + if (bestBreakDamage > 0) { + doBreak(); + } else if (bestPlaceDamage > 0) { + doPlace(); + } + }); } - private BlockPos findPlace(PlayerEntity target) { - if (!InvUtils.find(itemStack -> itemStack.getItem() instanceof BedItem).found()) return null; + private void setPlaceInfo(BlockPos footPos) { + if (!BlockUtils.canPlace(footPos)) return; - for (int index = 0; index < 3; index++) { - int i = index == 0 ? 1 : index == 1 ? 0 : 2; + // Air place check + if (!airPlace.get() && isAirPlace(footPos)) return; - for (CardinalDirection dir : CardinalDirection.values()) { - if (strictDirection.get() - && dir.toDirection() != mc.player.getHorizontalFacing() - && dir.toDirection().getOpposite() != mc.player.getHorizontalFacing()) continue; + // Check raycast and range + if (isOutOfRange(footPos)) return; - BlockPos centerPos = target.getBlockPos().up(i); + Direction rotateDirection = Direction.fromHorizontalDegrees(Rotations.getYaw(footPos.toCenterPos())); - float headSelfDamage = DamageUtils.bedDamage(mc.player, Utils.vec3d(centerPos)); - float offsetSelfDamage = DamageUtils.bedDamage(mc.player, Utils.vec3d(centerPos.offset(dir.toDirection()))); + for (Direction placeDirection : DirectionAccessor.meteor$getHorizontal()) { + BlockPos headPos = footPos.offset(placeDirection); + if (!mc.world.getBlockState(headPos).isReplaceable()) continue; - if (mc.world.getBlockState(centerPos).isReplaceable() - && BlockUtils.canPlace(centerPos.offset(dir.toDirection())) - && DamageUtils.bedDamage(target, Utils.vec3d(centerPos)) >= minDamage.get() - && offsetSelfDamage < maxSelfDamage.get() - && headSelfDamage < maxSelfDamage.get() - && (!antiSuicide.get() || PlayerUtils.getTotalHealth() - headSelfDamage > 0) - && (!antiSuicide.get() || PlayerUtils.getTotalHealth() - offsetSelfDamage > 0)) { - return centerPos.offset((direction = dir).toDirection()); - } + // Match our player's horizontal facing if we are using strict direction + if (strictDirection.get() && placeDirection != rotateDirection) continue; + + float targetDamage = DamageUtils.bedDamage(target, headPos.toCenterPos()); + if (isBestDamage(headPos, targetDamage, bestPlaceDamage)) { + bestPlaceDamage = targetDamage; + bestPlaceDirection = placeDirection; + bestPlacePos.set(footPos); } } - - return null; } - private BlockPos findBreak() { - for (BlockEntity blockEntity : Utils.blockEntities()) { - if (!(blockEntity instanceof BedBlockEntity)) continue; + private void setBreakInfo(BlockPos blockPos, BlockState blockState) { + // Check raycast and range + if (isOutOfRange(blockPos)) return; - BlockPos bedPos = blockEntity.getPos(); - Vec3d bedVec = Utils.vec3d(bedPos); + BlockPos otherPos = blockPos.offset(BedBlock.getOppositePartDirection(blockState)); + BlockPos headPos = blockState.get(BedBlock.PART) == BedPart.HEAD ? blockPos : otherPos; - if (PlayerUtils.isWithinReach(bedVec) - && DamageUtils.bedDamage(target, bedVec) >= minDamage.get() - && DamageUtils.bedDamage(mc.player, bedVec) < maxSelfDamage.get() - && (!antiSuicide.get() || PlayerUtils.getTotalHealth() - DamageUtils.bedDamage(mc.player, bedVec) > 0)) { - return bedPos; - } + float targetDamage = DamageUtils.bedDamage(target, headPos.toCenterPos()); + if (isBestDamage(headPos, targetDamage, bestBreakDamage)) { + bestBreakDamage = targetDamage; + bestBreakPos.set(blockPos); } - - return null; } - private boolean placeBed(BlockPos pos) { - if (pos == null) return false; + private boolean isBestDamage(BlockPos headPos, float targetDamage, double bestDamage) { + float selfDamage = DamageUtils.bedDamage(mc.player, headPos.toCenterPos()); + + // Is the bed optimal? + return targetDamage >= minDamage.get() && targetDamage > bestDamage + && (!antiSuicide.get() || selfDamage <= maxSelfDamage.get()) + && (!antiSuicide.get() || PlayerUtils.getTotalHealth() - selfDamage > 0); + } + private void doPlace() { FindItemResult bed = InvUtils.findInHotbar(itemStack -> itemStack.getItem() instanceof BedItem); - if (bed.getHand() == null && !autoSwitch.get()) return false; - - double yaw = switch (direction) { - case East -> 90; - case South -> 180; - case West -> -90; - default -> 0; - }; - - Rotations.rotate(yaw, Rotations.getPitch(pos), () -> { - BlockUtils.place(pos, bed, false, 0, swing.get(), true); - breakPos = pos; + if (!bed.found()) return; + + // Set render info + renderBlockPos = bestPlacePos; + renderDirection = bestPlaceDirection; + + if (placeDelayLeft++ < placeDelay.get()) return; + + if (autoSwitch.get()) InvUtils.swap(bed.slot(), swapBack.get()); + + if (!mc.player.isHolding(itemStack -> itemStack.getItem() instanceof BedItem)) return; + + // Get rotation to use depending on what direction we are doing + double yaw = Direction.getHorizontalDegreesOrThrow(bestPlaceDirection); + + // Use legit rotation if strict direction is enabled + if (strictDirection.get()) yaw = Rotations.getYaw(bestPlacePos); + + // Place bed! + Rotations.rotate(yaw, Rotations.getPitch(bestPlacePos), () -> { + BlockUtils.place(bestPlacePos, bed, false, 0, swing.get(), true, swapBack.get()); + placeDelayLeft = 0; }); + } + + private void doBreak() { + // Set render info + renderBlockPos = bestBreakPos; + renderDirection = BedBlock.getOppositePartDirection(mc.world.getBlockState(renderBlockPos)); + + if (breakDelayLeft++ < breakDelay.get()) return; + + // Break bed! + if (rotate.get()) { + Rotations.rotate(Rotations.getYaw(bestBreakPos), Rotations.getPitch(bestBreakPos), this::doInteract); + } else { + doInteract(); + } + } + + private void doInteract() { + BlockUtils.interact(new BlockHitResult(bestBreakPos.toCenterPos(), BlockUtils.getDirection(bestBreakPos), bestBreakPos, true), Hand.MAIN_HAND, swing.get()); + breakDelayLeft = 0; + } + + private boolean isOutOfRange(BlockPos blockPos) { + Vec3d pos = blockPos.toCenterPos(); + if (!PlayerUtils.isWithin(pos, placeRange.get())) return true; + RaycastContext raycastContext = new RaycastContext(mc.player.getEyePos(), pos, RaycastContext.ShapeType.COLLIDER, RaycastContext.FluidHandling.NONE, mc.player); + BlockHitResult result = mc.world.raycast(raycastContext); + if (result == null || !result.getBlockPos().equals(blockPos)) + return !PlayerUtils.isWithin(pos, placeWallsRange.get()); + + return false; + } + + private boolean isAirPlace(BlockPos blockPos) { + for (Direction direction : Direction.values()) { + if (!mc.world.getBlockState(blockPos.offset(direction)).isReplaceable()) return false; + } return true; } - private void breakBed(BlockPos pos) { - if (pos == null) return; - breakPos = null; + private boolean shouldPause() { + if (pauseOnUse.get() && mc.player.isUsingItem()) return true; - if (!(mc.world.getBlockState(pos).getBlock() instanceof BedBlock)) return; + if (pauseOnMine.get() && mc.interactionManager.isBreakingBlock()) return true; - boolean wasSneaking = mc.player.isSneaking(); - if (wasSneaking) mc.player.setSneaking(false); + CrystalAura CA = Modules.get().get(CrystalAura.class); + return pauseOnCA.get() && CA.isActive() && CA.kaTimer > 0; + } - mc.interactionManager.interactBlock(mc.player, Hand.OFF_HAND, new BlockHitResult(Vec3d.ofCenter(pos), Direction.UP, pos, false)); + @EventHandler + private void onBlockUpdate(BlockUpdateEvent event) { + // Spoof placement for instantaneous bed sync + if (spoofPlace.get() && event.newState.getBlock() instanceof BedBlock && event.oldState.isReplaceable()) { + BlockPos otherPos = event.pos.offset(BedBlock.getOppositePartDirection(event.newState)); + if (mc.world.getBlockState(otherPos).isReplaceable()) { + mc.world.setBlockState(otherPos, event.newState.with(BedBlock.PART, BedPart.HEAD), 0); + } + } - mc.player.setSneaking(wasSneaking); + // Spoof bed breaking for instantaneous bed sync + if (spoofBreak.get() && event.oldState.getBlock() instanceof BedBlock && event.newState.isReplaceable()) { + BlockPos otherPos = event.pos.offset(BedBlock.getOppositePartDirection(event.oldState)); + if (mc.world.getBlockState(otherPos).getBlock() instanceof BedBlock) { + mc.world.setBlockState(otherPos, mc.world.getFluidState(otherPos).getBlockState(), 0); + } + } } @EventHandler private void onRender(Render3DEvent event) { - if (render.get() && placePos != null && breakPos == null) { - int x = placePos.getX(); - int y = placePos.getY(); - int z = placePos.getZ(); - - switch (direction) { - case North -> event.renderer.box(x, y, z, x + 1, y + 0.6, z + 2, sideColor.get(), lineColor.get(), shapeMode.get(), 0); - case South -> event.renderer.box(x, y, z - 1, x + 1, y + 0.6, z + 1, sideColor.get(), lineColor.get(), shapeMode.get(), 0); - case East -> event.renderer.box(x - 1, y, z, x + 1, y + 0.6, z + 1, sideColor.get(), lineColor.get(), shapeMode.get(), 0); - case West -> event.renderer.box(x, y, z, x + 2, y + 0.6, z + 1, sideColor.get(), lineColor.get(), shapeMode.get(), 0); - } + if (!render.get() || renderBlockPos == null) return; + + int x = renderBlockPos.getX(), y = renderBlockPos.getY(), z = renderBlockPos.getZ(); + + switch (renderDirection) { + case Direction.SOUTH -> event.renderer.box(x, y, z, x + 1, y + 0.6, z + 2, sideColor.get(), lineColor.get(), shapeMode.get(), 0); + case Direction.NORTH -> event.renderer.box(x, y, z - 1, x + 1, y + 0.6, z + 1, sideColor.get(), lineColor.get(), shapeMode.get(), 0); + case Direction.WEST -> event.renderer.box(x - 1, y, z, x + 1, y + 0.6, z + 1, sideColor.get(), lineColor.get(), shapeMode.get(), 0); + case Direction.EAST -> event.renderer.box(x, y, z, x + 2, y + 0.6, z + 1, sideColor.get(), lineColor.get(), shapeMode.get(), 0); } }