From ece7936967834fc663b6e0dc14ea286ff23b1700 Mon Sep 17 00:00:00 2001 From: embeddedt <42941056+embeddedt@users.noreply.github.com> Date: Sat, 13 Apr 2024 19:00:06 -0400 Subject: [PATCH] Fix fluid renderer occluding when it shouldn't There are two bugs fixed by this commit: * scratchPos gets clobbered which causes isFaceSturdy to be invoked with the wrong position. Fixed by refactoring the method body. * It is not enough to check isFaceSturdy to determine that a fluid is occluded, the occlusion shape must be checked too --- .../chunk/compile/pipeline/FluidRenderer.java | 27 ++++++++++++++++--- 1 file changed, 23 insertions(+), 4 deletions(-) diff --git a/src/main/java/me/jellysquid/mods/sodium/client/render/chunk/compile/pipeline/FluidRenderer.java b/src/main/java/me/jellysquid/mods/sodium/client/render/chunk/compile/pipeline/FluidRenderer.java index 98665545a..36d0a5886 100644 --- a/src/main/java/me/jellysquid/mods/sodium/client/render/chunk/compile/pipeline/FluidRenderer.java +++ b/src/main/java/me/jellysquid/mods/sodium/client/render/chunk/compile/pipeline/FluidRenderer.java @@ -32,6 +32,7 @@ import net.minecraft.tags.FluidTags; import net.minecraft.util.Mth; import net.minecraft.world.level.BlockAndTintGetter; +import net.minecraft.world.level.block.Block; import net.minecraft.world.level.block.SupportType; import net.minecraft.world.level.block.state.BlockState; import net.minecraft.world.level.material.Fluid; @@ -75,14 +76,32 @@ public FluidRenderer(ColorProviderRegistry colorProviderRegistry, LightPipelineP } private boolean isFluidOccluded(BlockAndTintGetter world, int x, int y, int z, Direction dir, Fluid fluid) { + // Check if the fluid adjacent to us in the given direction is the same + if (world.getFluidState(this.scratchPos.set(x + dir.getStepX(), y + dir.getStepY(), z + dir.getStepZ())).getType().isSame(fluid)) { + return true; + } + + // Stricter than vanilla: check whether the containing block can occlude, has a sturdy face on the given side, + // and has a solid occlusion shape. If so, assume the fluid inside is not visible on that side. + // This avoids rendering the top face of water inside an upper waterlogged slab, for instance. BlockPos pos = this.scratchPos.set(x, y, z); BlockState blockState = world.getBlockState(pos); - BlockPos adjPos = this.scratchPos.set(x + dir.getStepX(), y + dir.getStepY(), z + dir.getStepZ()); - if (blockState.canOcclude()) { - return world.getFluidState(adjPos).getType().isSame(fluid) || blockState.isFaceSturdy(world, pos, dir, SupportType.FULL); + if (!blockState.canOcclude() || !blockState.isFaceSturdy(world, pos, dir, SupportType.FULL)) { + return false; + } + + VoxelShape sideShape = blockState.getFaceOcclusionShape(world, pos, dir); + if (sideShape == Shapes.block()) { + // The face fills the 1x1 area, so the fluid is occluded + return true; + } else if (sideShape == Shapes.empty()) { + // The face does not exist, so the fluid is not occluded + return false; + } else { + // Check if the face fills the 1x1 area + return Block.isShapeFullBlock(sideShape); } - return world.getFluidState(adjPos).getType().isSame(fluid); } private boolean isSideExposed(BlockAndTintGetter world, int x, int y, int z, Direction dir, float height) {