mirror of
https://github.com/Jozufozu/Flywheel.git
synced 2025-01-22 10:57:55 +01:00
Fix rendering far from world origin
- Make BatchingEngine use a shifting render origin - Make frustum checks relative to the render origin - Improve AbstractEntityVisual#isVisible to be closer to vanilla - Move render origin implementation code to AbstractEngine
This commit is contained in:
parent
05458d01d2
commit
2847ad28ed
13 changed files with 124 additions and 130 deletions
|
@ -1,9 +1,5 @@
|
||||||
package com.jozufozu.flywheel.api.event;
|
package com.jozufozu.flywheel.api.event;
|
||||||
|
|
||||||
import org.jetbrains.annotations.NotNull;
|
|
||||||
import org.joml.FrustumIntersection;
|
|
||||||
|
|
||||||
import com.jozufozu.flywheel.lib.math.MatrixUtil;
|
|
||||||
import com.mojang.blaze3d.vertex.PoseStack;
|
import com.mojang.blaze3d.vertex.PoseStack;
|
||||||
import com.mojang.math.Matrix4f;
|
import com.mojang.math.Matrix4f;
|
||||||
|
|
||||||
|
@ -12,20 +8,12 @@ import net.minecraft.client.multiplayer.ClientLevel;
|
||||||
import net.minecraft.client.renderer.LevelRenderer;
|
import net.minecraft.client.renderer.LevelRenderer;
|
||||||
import net.minecraft.client.renderer.RenderBuffers;
|
import net.minecraft.client.renderer.RenderBuffers;
|
||||||
|
|
||||||
public record RenderContext(LevelRenderer renderer, ClientLevel level, PoseStack stack, Matrix4f viewProjection,
|
public record RenderContext(LevelRenderer renderer, ClientLevel level, RenderBuffers buffers, PoseStack stack,
|
||||||
Matrix4f projection, RenderBuffers buffers, Camera camera, FrustumIntersection culler) {
|
Matrix4f projection, Matrix4f viewProjection, Camera camera) {
|
||||||
@NotNull
|
public static RenderContext create(LevelRenderer renderer, ClientLevel level, RenderBuffers buffers, PoseStack stack, Matrix4f projection, Camera camera) {
|
||||||
public static Matrix4f createViewProjection(PoseStack view, Matrix4f projection) {
|
Matrix4f viewProjection = projection.copy();
|
||||||
var viewProjection = projection.copy();
|
viewProjection.multiply(stack.last().pose());
|
||||||
viewProjection.multiply(view.last().pose());
|
|
||||||
return viewProjection;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static FrustumIntersection createCuller(Matrix4f viewProjection, float camX, float camY, float camZ) {
|
return new RenderContext(renderer, level, buffers, stack, projection, viewProjection, camera);
|
||||||
org.joml.Matrix4f proj = MatrixUtil.toJoml(viewProjection);
|
|
||||||
|
|
||||||
proj.translate(camX, camY, camZ);
|
|
||||||
|
|
||||||
return new FrustumIntersection(proj);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -19,7 +19,7 @@ public class Backends {
|
||||||
*/
|
*/
|
||||||
public static final Backend BATCHING = SimpleBackend.builder()
|
public static final Backend BATCHING = SimpleBackend.builder()
|
||||||
.engineMessage(new TextComponent("Using Batching Engine").withStyle(ChatFormatting.GREEN))
|
.engineMessage(new TextComponent("Using Batching Engine").withStyle(ChatFormatting.GREEN))
|
||||||
.engineFactory(level -> new BatchingEngine())
|
.engineFactory(level -> new BatchingEngine(256))
|
||||||
.supported(() -> !ShadersModHandler.isShaderPackInUse())
|
.supported(() -> !ShadersModHandler.isShaderPackInUse())
|
||||||
.register(Flywheel.rl("batching"));
|
.register(Flywheel.rl("batching"));
|
||||||
|
|
||||||
|
@ -28,7 +28,7 @@ public class Backends {
|
||||||
*/
|
*/
|
||||||
public static final Backend INSTANCING = SimpleBackend.builder()
|
public static final Backend INSTANCING = SimpleBackend.builder()
|
||||||
.engineMessage(new TextComponent("Using Instancing Engine").withStyle(ChatFormatting.GREEN))
|
.engineMessage(new TextComponent("Using Instancing Engine").withStyle(ChatFormatting.GREEN))
|
||||||
.engineFactory(level -> new InstancingEngine(Contexts.WORLD, 100))
|
.engineFactory(level -> new InstancingEngine(256, Contexts.WORLD))
|
||||||
.fallback(() -> Backends.BATCHING)
|
.fallback(() -> Backends.BATCHING)
|
||||||
.supported(() -> !ShadersModHandler.isShaderPackInUse() && GlCompat.getInstance()
|
.supported(() -> !ShadersModHandler.isShaderPackInUse() && GlCompat.getInstance()
|
||||||
.instancedArraysSupported())
|
.instancedArraysSupported())
|
||||||
|
@ -40,7 +40,7 @@ public class Backends {
|
||||||
*/
|
*/
|
||||||
public static final Backend INDIRECT = SimpleBackend.builder()
|
public static final Backend INDIRECT = SimpleBackend.builder()
|
||||||
.engineMessage(new TextComponent("Using Indirect Engine").withStyle(ChatFormatting.GREEN))
|
.engineMessage(new TextComponent("Using Indirect Engine").withStyle(ChatFormatting.GREEN))
|
||||||
.engineFactory(level -> new IndirectEngine(100))
|
.engineFactory(level -> new IndirectEngine(256))
|
||||||
.fallback(() -> Backends.INSTANCING)
|
.fallback(() -> Backends.INSTANCING)
|
||||||
.supported(() -> !ShadersModHandler.isShaderPackInUse() && GlCompat.getInstance()
|
.supported(() -> !ShadersModHandler.isShaderPackInUse() && GlCompat.getInstance()
|
||||||
.supportsIndirect())
|
.supportsIndirect())
|
||||||
|
|
|
@ -112,7 +112,7 @@ public class FlwCompiler {
|
||||||
var details = errors.stream()
|
var details = errors.stream()
|
||||||
.map(FailedCompilation::getMessage)
|
.map(FailedCompilation::getMessage)
|
||||||
.collect(Collectors.joining("\n"));
|
.collect(Collectors.joining("\n"));
|
||||||
throw new ShaderLoadingException("Compilation failed.\n" + details);
|
// throw new ShaderLoadingException("Compilation failed.\n" + details);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,42 @@
|
||||||
|
package com.jozufozu.flywheel.backend.engine;
|
||||||
|
|
||||||
|
import com.jozufozu.flywheel.api.backend.Engine;
|
||||||
|
|
||||||
|
import net.minecraft.client.Camera;
|
||||||
|
import net.minecraft.core.BlockPos;
|
||||||
|
import net.minecraft.core.Vec3i;
|
||||||
|
import net.minecraft.world.phys.Vec3;
|
||||||
|
|
||||||
|
public abstract class AbstractEngine implements Engine {
|
||||||
|
protected final int sqrMaxOriginDistance;
|
||||||
|
protected BlockPos renderOrigin = BlockPos.ZERO;
|
||||||
|
|
||||||
|
public AbstractEngine(int maxOriginDistance) {
|
||||||
|
sqrMaxOriginDistance = maxOriginDistance * maxOriginDistance;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean updateRenderOrigin(Camera camera) {
|
||||||
|
Vec3 cameraPos = camera.getPosition();
|
||||||
|
double dx = renderOrigin.getX() - cameraPos.x;
|
||||||
|
double dy = renderOrigin.getY() - cameraPos.y;
|
||||||
|
double dz = renderOrigin.getZ() - cameraPos.z;
|
||||||
|
double distanceSqr = dx * dx + dy * dy + dz * dz;
|
||||||
|
|
||||||
|
if (distanceSqr <= sqrMaxOriginDistance) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
renderOrigin = new BlockPos(cameraPos);
|
||||||
|
onRenderOriginChanged();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Vec3i renderOrigin() {
|
||||||
|
return renderOrigin;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void onRenderOriginChanged() {
|
||||||
|
}
|
||||||
|
}
|
|
@ -2,7 +2,6 @@ package com.jozufozu.flywheel.backend.engine.batching;
|
||||||
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
import com.jozufozu.flywheel.api.backend.Engine;
|
|
||||||
import com.jozufozu.flywheel.api.event.RenderContext;
|
import com.jozufozu.flywheel.api.event.RenderContext;
|
||||||
import com.jozufozu.flywheel.api.event.RenderStage;
|
import com.jozufozu.flywheel.api.event.RenderStage;
|
||||||
import com.jozufozu.flywheel.api.instance.Instance;
|
import com.jozufozu.flywheel.api.instance.Instance;
|
||||||
|
@ -10,19 +9,21 @@ import com.jozufozu.flywheel.api.instance.InstanceType;
|
||||||
import com.jozufozu.flywheel.api.instance.Instancer;
|
import com.jozufozu.flywheel.api.instance.Instancer;
|
||||||
import com.jozufozu.flywheel.api.model.Model;
|
import com.jozufozu.flywheel.api.model.Model;
|
||||||
import com.jozufozu.flywheel.api.task.TaskExecutor;
|
import com.jozufozu.flywheel.api.task.TaskExecutor;
|
||||||
|
import com.jozufozu.flywheel.backend.engine.AbstractEngine;
|
||||||
import com.jozufozu.flywheel.util.FlwUtil;
|
import com.jozufozu.flywheel.util.FlwUtil;
|
||||||
import com.mojang.blaze3d.vertex.PoseStack;
|
import com.mojang.blaze3d.vertex.PoseStack;
|
||||||
|
|
||||||
import net.minecraft.client.Camera;
|
|
||||||
import net.minecraft.client.multiplayer.ClientLevel;
|
import net.minecraft.client.multiplayer.ClientLevel;
|
||||||
import net.minecraft.core.BlockPos;
|
|
||||||
import net.minecraft.core.Vec3i;
|
|
||||||
import net.minecraft.world.phys.Vec3;
|
import net.minecraft.world.phys.Vec3;
|
||||||
|
|
||||||
public class BatchingEngine implements Engine {
|
public class BatchingEngine extends AbstractEngine {
|
||||||
private final BatchingTransformManager transformManager = new BatchingTransformManager();
|
private final BatchingTransformManager transformManager = new BatchingTransformManager();
|
||||||
private final BatchingDrawTracker drawTracker = new BatchingDrawTracker();
|
private final BatchingDrawTracker drawTracker = new BatchingDrawTracker();
|
||||||
|
|
||||||
|
public BatchingEngine(int maxOriginDistance) {
|
||||||
|
super(maxOriginDistance);
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public <I extends Instance> Instancer<I> instancer(InstanceType<I> type, Model model, RenderStage stage) {
|
public <I extends Instance> Instancer<I> instancer(InstanceType<I> type, Model model, RenderStage stage) {
|
||||||
return transformManager.getInstancer(type, model, stage);
|
return transformManager.getInstancer(type, model, stage);
|
||||||
|
@ -32,9 +33,9 @@ public class BatchingEngine implements Engine {
|
||||||
public void beginFrame(TaskExecutor executor, RenderContext context) {
|
public void beginFrame(TaskExecutor executor, RenderContext context) {
|
||||||
transformManager.flush();
|
transformManager.flush();
|
||||||
|
|
||||||
Vec3 cameraPos = context.camera().getPosition();
|
|
||||||
var stack = FlwUtil.copyPoseStack(context.stack());
|
var stack = FlwUtil.copyPoseStack(context.stack());
|
||||||
stack.translate(-cameraPos.x, -cameraPos.y, -cameraPos.z);
|
Vec3 cameraPos = context.camera().getPosition();
|
||||||
|
stack.translate(renderOrigin.getX() - cameraPos.x, renderOrigin.getY() - cameraPos.y, renderOrigin.getZ() - cameraPos.z);
|
||||||
|
|
||||||
// TODO: async task executor barriers
|
// TODO: async task executor barriers
|
||||||
executor.syncPoint();
|
executor.syncPoint();
|
||||||
|
@ -78,13 +79,8 @@ public class BatchingEngine implements Engine {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean updateRenderOrigin(Camera camera) {
|
protected void onRenderOriginChanged() {
|
||||||
return false;
|
transformManager.clearInstancers();
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public Vec3i renderOrigin() {
|
|
||||||
return BlockPos.ZERO;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -95,5 +91,6 @@ public class BatchingEngine implements Engine {
|
||||||
@Override
|
@Override
|
||||||
public void addDebugInfo(List<String> info) {
|
public void addDebugInfo(List<String> info) {
|
||||||
info.add("Batching");
|
info.add("Batching");
|
||||||
|
info.add("Origin: " + renderOrigin.getX() + ", " + renderOrigin.getY() + ", " + renderOrigin.getZ());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,7 +4,6 @@ import java.util.List;
|
||||||
|
|
||||||
import org.lwjgl.opengl.GL32;
|
import org.lwjgl.opengl.GL32;
|
||||||
|
|
||||||
import com.jozufozu.flywheel.api.backend.Engine;
|
|
||||||
import com.jozufozu.flywheel.api.event.RenderContext;
|
import com.jozufozu.flywheel.api.event.RenderContext;
|
||||||
import com.jozufozu.flywheel.api.event.RenderStage;
|
import com.jozufozu.flywheel.api.event.RenderStage;
|
||||||
import com.jozufozu.flywheel.api.instance.Instance;
|
import com.jozufozu.flywheel.api.instance.Instance;
|
||||||
|
@ -12,25 +11,18 @@ import com.jozufozu.flywheel.api.instance.InstanceType;
|
||||||
import com.jozufozu.flywheel.api.instance.Instancer;
|
import com.jozufozu.flywheel.api.instance.Instancer;
|
||||||
import com.jozufozu.flywheel.api.model.Model;
|
import com.jozufozu.flywheel.api.model.Model;
|
||||||
import com.jozufozu.flywheel.api.task.TaskExecutor;
|
import com.jozufozu.flywheel.api.task.TaskExecutor;
|
||||||
|
import com.jozufozu.flywheel.backend.engine.AbstractEngine;
|
||||||
import com.jozufozu.flywheel.gl.GlStateTracker;
|
import com.jozufozu.flywheel.gl.GlStateTracker;
|
||||||
import com.jozufozu.flywheel.gl.GlTextureUnit;
|
import com.jozufozu.flywheel.gl.GlTextureUnit;
|
||||||
import com.mojang.blaze3d.systems.RenderSystem;
|
import com.mojang.blaze3d.systems.RenderSystem;
|
||||||
|
|
||||||
import net.minecraft.client.Camera;
|
|
||||||
import net.minecraft.client.Minecraft;
|
import net.minecraft.client.Minecraft;
|
||||||
import net.minecraft.core.BlockPos;
|
|
||||||
import net.minecraft.core.Vec3i;
|
|
||||||
import net.minecraft.world.phys.Vec3;
|
|
||||||
|
|
||||||
public class IndirectEngine implements Engine {
|
|
||||||
private final int sqrMaxOriginDistance;
|
|
||||||
|
|
||||||
|
public class IndirectEngine extends AbstractEngine {
|
||||||
private final IndirectDrawManager drawManager = new IndirectDrawManager();
|
private final IndirectDrawManager drawManager = new IndirectDrawManager();
|
||||||
|
|
||||||
private BlockPos renderOrigin = BlockPos.ZERO;
|
|
||||||
|
|
||||||
public IndirectEngine(int maxOriginDistance) {
|
public IndirectEngine(int maxOriginDistance) {
|
||||||
this.sqrMaxOriginDistance = maxOriginDistance * maxOriginDistance;
|
super(maxOriginDistance);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -68,25 +60,8 @@ public class IndirectEngine implements Engine {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean updateRenderOrigin(Camera camera) {
|
protected void onRenderOriginChanged() {
|
||||||
Vec3 cameraPos = camera.getPosition();
|
|
||||||
double dx = renderOrigin.getX() - cameraPos.x;
|
|
||||||
double dy = renderOrigin.getY() - cameraPos.y;
|
|
||||||
double dz = renderOrigin.getZ() - cameraPos.z;
|
|
||||||
double distanceSqr = dx * dx + dy * dy + dz * dz;
|
|
||||||
|
|
||||||
if (distanceSqr <= sqrMaxOriginDistance) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
renderOrigin = new BlockPos(cameraPos);
|
|
||||||
drawManager.clearInstancers();
|
drawManager.clearInstancers();
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public Vec3i renderOrigin() {
|
|
||||||
return renderOrigin;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
|
@ -4,7 +4,6 @@ import java.util.List;
|
||||||
|
|
||||||
import org.lwjgl.opengl.GL32;
|
import org.lwjgl.opengl.GL32;
|
||||||
|
|
||||||
import com.jozufozu.flywheel.api.backend.Engine;
|
|
||||||
import com.jozufozu.flywheel.api.context.Context;
|
import com.jozufozu.flywheel.api.context.Context;
|
||||||
import com.jozufozu.flywheel.api.event.RenderContext;
|
import com.jozufozu.flywheel.api.event.RenderContext;
|
||||||
import com.jozufozu.flywheel.api.event.RenderStage;
|
import com.jozufozu.flywheel.api.event.RenderStage;
|
||||||
|
@ -15,29 +14,22 @@ import com.jozufozu.flywheel.api.model.Model;
|
||||||
import com.jozufozu.flywheel.api.task.TaskExecutor;
|
import com.jozufozu.flywheel.api.task.TaskExecutor;
|
||||||
import com.jozufozu.flywheel.backend.Pipelines;
|
import com.jozufozu.flywheel.backend.Pipelines;
|
||||||
import com.jozufozu.flywheel.backend.compile.FlwCompiler;
|
import com.jozufozu.flywheel.backend.compile.FlwCompiler;
|
||||||
|
import com.jozufozu.flywheel.backend.engine.AbstractEngine;
|
||||||
import com.jozufozu.flywheel.backend.engine.UniformBuffer;
|
import com.jozufozu.flywheel.backend.engine.UniformBuffer;
|
||||||
import com.jozufozu.flywheel.gl.GlStateTracker;
|
import com.jozufozu.flywheel.gl.GlStateTracker;
|
||||||
import com.jozufozu.flywheel.gl.GlTextureUnit;
|
import com.jozufozu.flywheel.gl.GlTextureUnit;
|
||||||
import com.jozufozu.flywheel.lib.material.MaterialIndices;
|
import com.jozufozu.flywheel.lib.material.MaterialIndices;
|
||||||
import com.mojang.blaze3d.systems.RenderSystem;
|
import com.mojang.blaze3d.systems.RenderSystem;
|
||||||
|
|
||||||
import net.minecraft.client.Camera;
|
|
||||||
import net.minecraft.client.Minecraft;
|
import net.minecraft.client.Minecraft;
|
||||||
import net.minecraft.core.BlockPos;
|
|
||||||
import net.minecraft.core.Vec3i;
|
|
||||||
import net.minecraft.world.phys.Vec3;
|
|
||||||
|
|
||||||
public class InstancingEngine implements Engine {
|
public class InstancingEngine extends AbstractEngine {
|
||||||
private final Context context;
|
private final Context context;
|
||||||
private final int sqrMaxOriginDistance;
|
|
||||||
|
|
||||||
private final InstancingDrawManager drawManager = new InstancingDrawManager();
|
private final InstancingDrawManager drawManager = new InstancingDrawManager();
|
||||||
|
|
||||||
private BlockPos renderOrigin = BlockPos.ZERO;
|
public InstancingEngine(int maxOriginDistance, Context context) {
|
||||||
|
super(maxOriginDistance);
|
||||||
public InstancingEngine(Context context, int maxOriginDistance) {
|
|
||||||
this.context = context;
|
this.context = context;
|
||||||
this.sqrMaxOriginDistance = maxOriginDistance * maxOriginDistance;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -116,25 +108,8 @@ public class InstancingEngine implements Engine {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean updateRenderOrigin(Camera camera) {
|
protected void onRenderOriginChanged() {
|
||||||
Vec3 cameraPos = camera.getPosition();
|
|
||||||
double dx = renderOrigin.getX() - cameraPos.x;
|
|
||||||
double dy = renderOrigin.getY() - cameraPos.y;
|
|
||||||
double dz = renderOrigin.getZ() - cameraPos.z;
|
|
||||||
double distanceSqr = dx * dx + dy * dy + dz * dz;
|
|
||||||
|
|
||||||
if (distanceSqr <= sqrMaxOriginDistance) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
renderOrigin = new BlockPos(cameraPos);
|
|
||||||
drawManager.clearInstancers();
|
drawManager.clearInstancers();
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public Vec3i renderOrigin() {
|
|
||||||
return renderOrigin;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
|
@ -19,7 +19,9 @@ import com.jozufozu.flywheel.impl.visualization.manager.BlockEntityVisualManager
|
||||||
import com.jozufozu.flywheel.impl.visualization.manager.EffectVisualManager;
|
import com.jozufozu.flywheel.impl.visualization.manager.EffectVisualManager;
|
||||||
import com.jozufozu.flywheel.impl.visualization.manager.EntityVisualManager;
|
import com.jozufozu.flywheel.impl.visualization.manager.EntityVisualManager;
|
||||||
import com.jozufozu.flywheel.impl.visualization.manager.VisualManager;
|
import com.jozufozu.flywheel.impl.visualization.manager.VisualManager;
|
||||||
|
import com.jozufozu.flywheel.lib.math.MatrixUtil;
|
||||||
|
|
||||||
|
import net.minecraft.core.Vec3i;
|
||||||
import net.minecraft.world.entity.Entity;
|
import net.minecraft.world.entity.Entity;
|
||||||
import net.minecraft.world.level.LevelAccessor;
|
import net.minecraft.world.level.LevelAccessor;
|
||||||
import net.minecraft.world.level.block.entity.BlockEntity;
|
import net.minecraft.world.level.block.entity.BlockEntity;
|
||||||
|
@ -93,16 +95,20 @@ public class VisualWorld implements AutoCloseable {
|
||||||
taskExecutor.syncPoint();
|
taskExecutor.syncPoint();
|
||||||
|
|
||||||
if (!originChanged) {
|
if (!originChanged) {
|
||||||
|
Vec3i renderOrigin = engine.renderOrigin();
|
||||||
var cameraPos = context.camera()
|
var cameraPos = context.camera()
|
||||||
.getPosition();
|
.getPosition();
|
||||||
double cameraX = cameraPos.x;
|
double cameraX = cameraPos.x;
|
||||||
double cameraY = cameraPos.y;
|
double cameraY = cameraPos.y;
|
||||||
double cameraZ = cameraPos.z;
|
double cameraZ = cameraPos.z;
|
||||||
FrustumIntersection culler = context.culler();
|
|
||||||
|
|
||||||
blockEntities.beginFrame(taskExecutor, cameraX, cameraY, cameraZ, culler);
|
org.joml.Matrix4f proj = MatrixUtil.toJoml(context.viewProjection());
|
||||||
entities.beginFrame(taskExecutor, cameraX, cameraY, cameraZ, culler);
|
proj.translate((float) (renderOrigin.getX() - cameraX), (float) (renderOrigin.getY() - cameraY), (float) (renderOrigin.getZ() - cameraZ));
|
||||||
effects.beginFrame(taskExecutor, cameraX, cameraY, cameraZ, culler);
|
FrustumIntersection frustum = new FrustumIntersection(proj);
|
||||||
|
|
||||||
|
blockEntities.beginFrame(taskExecutor, cameraX, cameraY, cameraZ, frustum);
|
||||||
|
entities.beginFrame(taskExecutor, cameraX, cameraY, cameraZ, frustum);
|
||||||
|
effects.beginFrame(taskExecutor, cameraX, cameraY, cameraZ, frustum);
|
||||||
}
|
}
|
||||||
|
|
||||||
engine.beginFrame(taskExecutor, context);
|
engine.beginFrame(taskExecutor, context);
|
||||||
|
|
|
@ -69,10 +69,11 @@ public abstract class AbstractBlockEntityVisual<T extends BlockEntity> extends A
|
||||||
* represents should be rendered at to appear in the correct location.
|
* represents should be rendered at to appear in the correct location.
|
||||||
*/
|
*/
|
||||||
public BlockPos getVisualPosition() {
|
public BlockPos getVisualPosition() {
|
||||||
return pos.subtract(renderOrigin);
|
return visualPos;
|
||||||
}
|
}
|
||||||
|
|
||||||
public boolean isVisible(FrustumIntersection frustum) {
|
public boolean isVisible(FrustumIntersection frustum) {
|
||||||
return frustum.testAab(pos.getX(), pos.getY(), pos.getZ(), pos.getX() + 1, pos.getY() + 1, pos.getZ() + 1);
|
return frustum.testAab(visualPos.getX(), visualPos.getY(), visualPos.getZ(),
|
||||||
|
visualPos.getX() + 1, visualPos.getY() + 1, visualPos.getZ() + 1);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -76,7 +76,9 @@ public abstract class AbstractEntityVisual<T extends Entity> extends AbstractVis
|
||||||
*/
|
*/
|
||||||
public Vector3f getVisualPosition() {
|
public Vector3f getVisualPosition() {
|
||||||
Vec3 pos = entity.position();
|
Vec3 pos = entity.position();
|
||||||
return new Vector3f((float) (pos.x - renderOrigin.getX()), (float) (pos.y - renderOrigin.getY()), (float) (pos.z - renderOrigin.getZ()));
|
return new Vector3f((float) (pos.x - renderOrigin.getX()),
|
||||||
|
(float) (pos.y - renderOrigin.getY()),
|
||||||
|
(float) (pos.z - renderOrigin.getZ()));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -88,11 +90,22 @@ public abstract class AbstractEntityVisual<T extends Entity> extends AbstractVis
|
||||||
*/
|
*/
|
||||||
public Vector3f getVisualPosition(float partialTicks) {
|
public Vector3f getVisualPosition(float partialTicks) {
|
||||||
Vec3 pos = entity.position();
|
Vec3 pos = entity.position();
|
||||||
return new Vector3f((float) (Mth.lerp(partialTicks, entity.xOld, pos.x) - renderOrigin.getX()), (float) (Mth.lerp(partialTicks, entity.yOld, pos.y) - renderOrigin.getY()), (float) (Mth.lerp(partialTicks, entity.zOld, pos.z) - renderOrigin.getZ()));
|
return new Vector3f((float) (Mth.lerp(partialTicks, entity.xOld, pos.x) - renderOrigin.getX()),
|
||||||
|
(float) (Mth.lerp(partialTicks, entity.yOld, pos.y) - renderOrigin.getY()),
|
||||||
|
(float) (Mth.lerp(partialTicks, entity.zOld, pos.z) - renderOrigin.getZ()));
|
||||||
}
|
}
|
||||||
|
|
||||||
public boolean isVisible(FrustumIntersection frustum) {
|
public boolean isVisible(FrustumIntersection frustum) {
|
||||||
AABB aabb = entity.getBoundingBox();
|
if (entity.noCulling) {
|
||||||
return frustum.testAab((float) aabb.minX, (float) aabb.minY, (float) aabb.minZ, (float) aabb.maxX, (float) aabb.maxY, (float) aabb.maxZ);
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
AABB aabb = entity.getBoundingBoxForCulling();
|
||||||
|
return frustum.testAab((float) (aabb.minX - renderOrigin.getX()) - 0.5f,
|
||||||
|
(float) (aabb.minY - renderOrigin.getY()) - 0.5f,
|
||||||
|
(float) (aabb.minZ - renderOrigin.getZ()) - 0.5f,
|
||||||
|
(float) (aabb.maxX - renderOrigin.getX()) + 0.5f,
|
||||||
|
(float) (aabb.maxY - renderOrigin.getY()) + 0.5f,
|
||||||
|
(float) (aabb.maxZ - renderOrigin.getZ()) + 0.5f);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -40,10 +40,7 @@ public class LevelRendererMixin {
|
||||||
|
|
||||||
@Inject(at = @At("HEAD"), method = "renderLevel")
|
@Inject(at = @At("HEAD"), method = "renderLevel")
|
||||||
private void flywheel$beginRender(PoseStack pPoseStack, float pPartialTick, long pFinishNanoTime, boolean pRenderBlockOutline, Camera pCamera, GameRenderer pGameRenderer, LightTexture pLightTexture, Matrix4f pProjectionMatrix, CallbackInfo ci) {
|
private void flywheel$beginRender(PoseStack pPoseStack, float pPartialTick, long pFinishNanoTime, boolean pRenderBlockOutline, Camera pCamera, GameRenderer pGameRenderer, LightTexture pLightTexture, Matrix4f pProjectionMatrix, CallbackInfo ci) {
|
||||||
var viewProjection = RenderContext.createViewProjection(pPoseStack, pProjectionMatrix);
|
flywheel$renderContext = RenderContext.create((LevelRenderer) (Object) this, level, renderBuffers, pPoseStack, pProjectionMatrix, pCamera);
|
||||||
var cameraPos = pCamera.getPosition();
|
|
||||||
var culler = RenderContext.createCuller(viewProjection, (float) -cameraPos.x, (float) -cameraPos.y, (float) -cameraPos.z);
|
|
||||||
flywheel$renderContext = new RenderContext((LevelRenderer) (Object) this, level, pPoseStack, viewProjection, pProjectionMatrix, renderBuffers, pCamera, culler);
|
|
||||||
|
|
||||||
MinecraftForge.EVENT_BUS.post(new BeginFrameEvent(flywheel$renderContext));
|
MinecraftForge.EVENT_BUS.post(new BeginFrameEvent(flywheel$renderContext));
|
||||||
}
|
}
|
||||||
|
|
|
@ -24,8 +24,8 @@ import net.minecraft.world.level.block.RenderShape;
|
||||||
import net.minecraft.world.level.block.state.BlockState;
|
import net.minecraft.world.level.block.state.BlockState;
|
||||||
import net.minecraft.world.phys.Vec3;
|
import net.minecraft.world.phys.Vec3;
|
||||||
|
|
||||||
public class MinecartVisual<T extends AbstractMinecart> extends AbstractEntityVisual<T> implements DynamicVisual, TickableVisual {
|
public class MinecartVisual<T extends AbstractMinecart> extends AbstractEntityVisual<T> implements TickableVisual, DynamicVisual {
|
||||||
private static final SimpleLazyModel BODY_MODEL = new SimpleLazyModel(MinecartVisual::getBodyMesh, Materials.MINECART);
|
private static final SimpleLazyModel BODY_MODEL = new SimpleLazyModel(MinecartVisual::createBodyMesh, Materials.MINECART);
|
||||||
|
|
||||||
private final PoseStack stack = new PoseStack();
|
private final PoseStack stack = new PoseStack();
|
||||||
|
|
||||||
|
@ -47,11 +47,6 @@ public class MinecartVisual<T extends AbstractMinecart> extends AbstractEntityVi
|
||||||
super.init();
|
super.init();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean decreaseFramerateWithDistance() {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void tick() {
|
public void tick() {
|
||||||
BlockState displayBlockState = entity.getDisplayBlockState();
|
BlockState displayBlockState = entity.getDisplayBlockState();
|
||||||
|
@ -139,6 +134,11 @@ public class MinecartVisual<T extends AbstractMinecart> extends AbstractEntityVi
|
||||||
body.setTransform(stack);
|
body.setTransform(stack);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean decreaseFramerateWithDistance() {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void updateLight() {
|
public void updateLight() {
|
||||||
if (contents == null) {
|
if (contents == null) {
|
||||||
|
@ -156,6 +156,11 @@ public class MinecartVisual<T extends AbstractMinecart> extends AbstractEntityVi
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private TransformedInstance createBodyInstance() {
|
||||||
|
return instancerProvider.instancer(InstanceTypes.TRANSFORMED, BODY_MODEL, RenderStage.AFTER_ENTITIES)
|
||||||
|
.createInstance();
|
||||||
|
}
|
||||||
|
|
||||||
private TransformedInstance createContentsInstance() {
|
private TransformedInstance createContentsInstance() {
|
||||||
RenderShape shape = blockState.getRenderShape();
|
RenderShape shape = blockState.getRenderShape();
|
||||||
|
|
||||||
|
@ -174,13 +179,8 @@ public class MinecartVisual<T extends AbstractMinecart> extends AbstractEntityVi
|
||||||
.createInstance();
|
.createInstance();
|
||||||
}
|
}
|
||||||
|
|
||||||
private TransformedInstance createBodyInstance() {
|
|
||||||
return instancerProvider.instancer(InstanceTypes.TRANSFORMED, BODY_MODEL, RenderStage.AFTER_ENTITIES)
|
|
||||||
.createInstance();
|
|
||||||
}
|
|
||||||
|
|
||||||
@NotNull
|
@NotNull
|
||||||
private static ModelPart getBodyMesh() {
|
private static ModelPart createBodyMesh() {
|
||||||
int y = -3;
|
int y = -3;
|
||||||
return ModelPart.builder("minecart", 64, 32)
|
return ModelPart.builder("minecart", 64, 32)
|
||||||
.cuboid().invertYZ().start(-10, -8, -y).size(20, 16, 2).textureOffset(0, 10).rotateZ((float) Math.PI).rotateX(((float)Math.PI / 2F)).endCuboid()
|
.cuboid().invertYZ().start(-10, -8, -y).size(20, 16, 2).textureOffset(0, 10).rotateZ((float) Math.PI).rotateX(((float)Math.PI / 2F)).endCuboid()
|
||||||
|
|
|
@ -72,6 +72,14 @@ public class ShulkerBoxVisual extends AbstractBlockEntityVisual<ShulkerBoxBlockE
|
||||||
super.init();
|
super.init();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private Direction getDirection() {
|
||||||
|
if (blockState.getBlock() instanceof ShulkerBoxBlock) {
|
||||||
|
return blockState.getValue(ShulkerBoxBlock.FACING);
|
||||||
|
}
|
||||||
|
|
||||||
|
return Direction.UP;
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void beginFrame() {
|
public void beginFrame() {
|
||||||
float progress = blockEntity.getProgress(AnimationTickHolder.getPartialTicks());
|
float progress = blockEntity.getProgress(AnimationTickHolder.getPartialTicks());
|
||||||
|
@ -139,12 +147,4 @@ public class ShulkerBoxVisual extends AbstractBlockEntityVisual<ShulkerBoxBlockE
|
||||||
.endCuboid()
|
.endCuboid()
|
||||||
.build();
|
.build();
|
||||||
}
|
}
|
||||||
|
|
||||||
private Direction getDirection() {
|
|
||||||
if (blockState.getBlock() instanceof ShulkerBoxBlock) {
|
|
||||||
return blockState.getValue(ShulkerBoxBlock.FACING);
|
|
||||||
}
|
|
||||||
|
|
||||||
return Direction.UP;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue