package net.minecraft.client; import java.util.Arrays; import net.fabricmc.api.EnvType; import net.fabricmc.api.Environment; import net.minecraft.core.BlockPos; import net.minecraft.core.Direction; import net.minecraft.tags.FluidTags; import net.minecraft.util.Mth; import net.minecraft.world.entity.Entity; import net.minecraft.world.entity.LivingEntity; import net.minecraft.world.entity.vehicle.Minecart; import net.minecraft.world.entity.vehicle.NewMinecartBehavior; import net.minecraft.world.level.BlockGetter; import net.minecraft.world.level.ClipContext; import net.minecraft.world.level.ClipContext.Block; import net.minecraft.world.level.ClipContext.Fluid; import net.minecraft.world.level.block.Blocks; import net.minecraft.world.level.block.state.BlockState; import net.minecraft.world.level.material.FluidState; import net.minecraft.world.level.material.FogType; import net.minecraft.world.phys.HitResult; import net.minecraft.world.phys.Vec3; import net.minecraft.world.phys.HitResult.Type; import org.joml.Quaternionf; import org.joml.Vector3f; @Environment(EnvType.CLIENT) public class Camera { private static final float DEFAULT_CAMERA_DISTANCE = 4.0F; private static final Vector3f FORWARDS = new Vector3f(0.0F, 0.0F, -1.0F); private static final Vector3f UP = new Vector3f(0.0F, 1.0F, 0.0F); private static final Vector3f LEFT = new Vector3f(-1.0F, 0.0F, 0.0F); private boolean initialized; private BlockGetter level; private Entity entity; private Vec3 position = Vec3.ZERO; private final BlockPos.MutableBlockPos blockPosition = new BlockPos.MutableBlockPos(); private final Vector3f forwards = new Vector3f(FORWARDS); private final Vector3f up = new Vector3f(UP); private final Vector3f left = new Vector3f(LEFT); private float xRot; private float yRot; private final Quaternionf rotation = new Quaternionf(); private boolean detached; private float eyeHeight; private float eyeHeightOld; private float partialTickTime; public static final float FOG_DISTANCE_SCALE = 0.083333336F; public void setup(BlockGetter level, Entity entity, boolean detached, boolean thirdPersonReverse, float partialTick) { this.initialized = true; this.level = level; this.entity = entity; this.detached = detached; this.partialTickTime = partialTick; if (entity.isPassenger() && entity.getVehicle() instanceof Minecart minecart && minecart.getBehavior() instanceof NewMinecartBehavior newMinecartBehavior && newMinecartBehavior.cartHasPosRotLerp()) { Vec3 vec3 = minecart.getPassengerRidingPosition(entity) .subtract(minecart.position()) .subtract(entity.getVehicleAttachmentPoint(minecart)) .add(new Vec3(0.0, Mth.lerp(partialTick, this.eyeHeightOld, this.eyeHeight), 0.0)); this.setRotation(entity.getViewYRot(partialTick), entity.getViewXRot(partialTick)); this.setPosition(newMinecartBehavior.getCartLerpPosition(partialTick).add(vec3)); } else { this.setRotation(entity.getViewYRot(partialTick), entity.getViewXRot(partialTick)); this.setPosition( Mth.lerp((double)partialTick, entity.xo, entity.getX()), Mth.lerp((double)partialTick, entity.yo, entity.getY()) + Mth.lerp(partialTick, this.eyeHeightOld, this.eyeHeight), Mth.lerp((double)partialTick, entity.zo, entity.getZ()) ); } if (detached) { if (thirdPersonReverse) { this.setRotation(this.yRot + 180.0F, -this.xRot); } float f = entity instanceof LivingEntity livingEntity ? livingEntity.getScale() : 1.0F; this.move(-this.getMaxZoom(4.0F * f), 0.0F, 0.0F); } else if (entity instanceof LivingEntity && ((LivingEntity)entity).isSleeping()) { Direction direction = ((LivingEntity)entity).getBedOrientation(); this.setRotation(direction != null ? direction.toYRot() - 180.0F : 0.0F, 0.0F); this.move(0.0F, 0.3F, 0.0F); } } public void tick() { if (this.entity != null) { this.eyeHeightOld = this.eyeHeight; this.eyeHeight = this.eyeHeight + (this.entity.getEyeHeight() - this.eyeHeight) * 0.5F; } } private float getMaxZoom(float maxZoom) { float f = 0.1F; for (int i = 0; i < 8; i++) { float g = (i & 1) * 2 - 1; float h = (i >> 1 & 1) * 2 - 1; float j = (i >> 2 & 1) * 2 - 1; Vec3 vec3 = this.position.add(g * 0.1F, h * 0.1F, j * 0.1F); Vec3 vec32 = vec3.add(new Vec3(this.forwards).scale(-maxZoom)); HitResult hitResult = this.level.clip(new ClipContext(vec3, vec32, Block.VISUAL, Fluid.NONE, this.entity)); if (hitResult.getType() != Type.MISS) { float k = (float)hitResult.getLocation().distanceToSqr(this.position); if (k < Mth.square(maxZoom)) { maxZoom = Mth.sqrt(k); } } } return maxZoom; } protected void move(float zoom, float dy, float dx) { Vector3f vector3f = new Vector3f(dx, dy, -zoom).rotate(this.rotation); this.setPosition(new Vec3(this.position.x + vector3f.x, this.position.y + vector3f.y, this.position.z + vector3f.z)); } protected void setRotation(float yRot, float xRot) { this.xRot = xRot; this.yRot = yRot; this.rotation.rotationYXZ((float) Math.PI - yRot * (float) (Math.PI / 180.0), -xRot * (float) (Math.PI / 180.0), 0.0F); FORWARDS.rotate(this.rotation, this.forwards); UP.rotate(this.rotation, this.up); LEFT.rotate(this.rotation, this.left); } /** * Sets the position and blockpos of the active render */ protected void setPosition(double x, double y, double z) { this.setPosition(new Vec3(x, y, z)); } protected void setPosition(Vec3 pos) { this.position = pos; this.blockPosition.set(pos.x, pos.y, pos.z); } public Vec3 getPosition() { return this.position; } public BlockPos getBlockPosition() { return this.blockPosition; } public float getXRot() { return this.xRot; } public float getYRot() { return this.yRot; } public Quaternionf rotation() { return this.rotation; } public Entity getEntity() { return this.entity; } public boolean isInitialized() { return this.initialized; } public boolean isDetached() { return this.detached; } public Camera.NearPlane getNearPlane() { Minecraft minecraft = Minecraft.getInstance(); double d = (double)minecraft.getWindow().getWidth() / minecraft.getWindow().getHeight(); double e = Math.tan(minecraft.options.fov().get().intValue() * (float) (Math.PI / 180.0) / 2.0) * 0.05F; double f = e * d; Vec3 vec3 = new Vec3(this.forwards).scale(0.05F); Vec3 vec32 = new Vec3(this.left).scale(f); Vec3 vec33 = new Vec3(this.up).scale(e); return new Camera.NearPlane(vec3, vec32, vec33); } public FogType getFluidInCamera() { if (!this.initialized) { return FogType.NONE; } else { FluidState fluidState = this.level.getFluidState(this.blockPosition); if (fluidState.is(FluidTags.WATER) && this.position.y < this.blockPosition.getY() + fluidState.getHeight(this.level, this.blockPosition)) { return FogType.WATER; } else { Camera.NearPlane nearPlane = this.getNearPlane(); for (Vec3 vec3 : Arrays.asList(nearPlane.forward, nearPlane.getTopLeft(), nearPlane.getTopRight(), nearPlane.getBottomLeft(), nearPlane.getBottomRight())) { Vec3 vec32 = this.position.add(vec3); BlockPos blockPos = BlockPos.containing(vec32); FluidState fluidState2 = this.level.getFluidState(blockPos); if (fluidState2.is(FluidTags.LAVA)) { if (vec32.y <= fluidState2.getHeight(this.level, blockPos) + blockPos.getY()) { return FogType.LAVA; } } else { BlockState blockState = this.level.getBlockState(blockPos); if (blockState.is(Blocks.POWDER_SNOW)) { return FogType.POWDER_SNOW; } } } return FogType.NONE; } } } public final Vector3f getLookVector() { return this.forwards; } public final Vector3f getUpVector() { return this.up; } public final Vector3f getLeftVector() { return this.left; } public void reset() { this.level = null; this.entity = null; this.initialized = false; } public float getPartialTickTime() { return this.partialTickTime; } @Environment(EnvType.CLIENT) public static class NearPlane { final Vec3 forward; private final Vec3 left; private final Vec3 up; NearPlane(Vec3 forward, Vec3 left, Vec3 up) { this.forward = forward; this.left = left; this.up = up; } public Vec3 getTopLeft() { return this.forward.add(this.up).add(this.left); } public Vec3 getTopRight() { return this.forward.add(this.up).subtract(this.left); } public Vec3 getBottomLeft() { return this.forward.subtract(this.up).add(this.left); } public Vec3 getBottomRight() { return this.forward.subtract(this.up).subtract(this.left); } public Vec3 getPointOnPlane(float leftScale, float upScale) { return this.forward.add(this.up.scale(upScale)).subtract(this.left.scale(leftScale)); } } }