diff --git a/src/main/java/com/jozufozu/flywheel/api/backend/Engine.java b/src/main/java/com/jozufozu/flywheel/api/backend/Engine.java index bdce50a09..4f0c75bac 100644 --- a/src/main/java/com/jozufozu/flywheel/api/backend/Engine.java +++ b/src/main/java/com/jozufozu/flywheel/api/backend/Engine.java @@ -2,6 +2,7 @@ package com.jozufozu.flywheel.api.backend; import com.jozufozu.flywheel.api.event.RenderContext; import com.jozufozu.flywheel.api.event.RenderStage; +import com.jozufozu.flywheel.api.instance.Instance; import com.jozufozu.flywheel.api.instance.InstancerProvider; import com.jozufozu.flywheel.api.task.Plan; import com.jozufozu.flywheel.api.task.TaskExecutor; @@ -14,6 +15,8 @@ public interface Engine extends InstancerProvider { void renderStage(TaskExecutor executor, RenderContext context, RenderStage stage); + void renderCrumblingInstance(TaskExecutor taskExecutor, RenderContext context, Instance instance, int progress); + /** * Maintain the render origin to be within a certain distance from the camera in all directions, * preventing floating point precision issues at high coordinates. @@ -24,7 +27,5 @@ public interface Engine extends InstancerProvider { Vec3i renderOrigin(); - // TODO: "delete" implies that the object cannot be used afterwards, but all current implementations - // support the "invalidate" contract as well, meaning they can be reused after this call. Rename? - void delete(); + void invalidate(); } diff --git a/src/main/java/com/jozufozu/flywheel/api/instance/Instance.java b/src/main/java/com/jozufozu/flywheel/api/instance/Instance.java index 36fec9ee9..1ba692c7d 100644 --- a/src/main/java/com/jozufozu/flywheel/api/instance/Instance.java +++ b/src/main/java/com/jozufozu/flywheel/api/instance/Instance.java @@ -2,4 +2,6 @@ package com.jozufozu.flywheel.api.instance; public interface Instance { InstanceType type(); + + InstanceHandle handle(); } diff --git a/src/main/java/com/jozufozu/flywheel/api/task/Plan.java b/src/main/java/com/jozufozu/flywheel/api/task/Plan.java index f7f1fe31c..43b3e7937 100644 --- a/src/main/java/com/jozufozu/flywheel/api/task/Plan.java +++ b/src/main/java/com/jozufozu/flywheel/api/task/Plan.java @@ -33,16 +33,6 @@ public interface Plan { */ Plan then(Plan plan); - /** - * Create a new plan that executes this plan, then transforms the context to execute the given plan. - * - * @param map A function that transforms the plan context. - * @param plan The plan to execute after this plan with the transformed context. - * @param The type of the transformed context. - * @return The composed plan. - */ - Plan thenMap(Function map, Plan plan); - /** * Create a new plan that executes this plan and the given plan in parallel. * @@ -51,16 +41,6 @@ public interface Plan { */ Plan and(Plan plan); - /** - * Create a new plan that executes this plan and the given plan in parallel. - * - * @param map A function that transforms the plan context. - * @param plan The plan to execute in parallel with this plan than accepts the transformed context. - * @param The type of the transformed context. - * @return The composed plan. - */ - Plan andMap(Function map, Plan plan); - /** * If possible, create a new plan that accomplishes everything * this plan does but with a simpler execution schedule. diff --git a/src/main/java/com/jozufozu/flywheel/backend/engine/InstanceHandleImpl.java b/src/main/java/com/jozufozu/flywheel/backend/engine/InstanceHandleImpl.java index 2cc8f059e..9e22fc28c 100644 --- a/src/main/java/com/jozufozu/flywheel/backend/engine/InstanceHandleImpl.java +++ b/src/main/java/com/jozufozu/flywheel/backend/engine/InstanceHandleImpl.java @@ -3,7 +3,7 @@ package com.jozufozu.flywheel.backend.engine; import com.jozufozu.flywheel.api.instance.InstanceHandle; public class InstanceHandleImpl implements InstanceHandle { - private final AbstractInstancer instancer; + public final AbstractInstancer instancer; private int index; public InstanceHandleImpl(AbstractInstancer instancer, int index) { diff --git a/src/main/java/com/jozufozu/flywheel/backend/engine/batching/BatchingEngine.java b/src/main/java/com/jozufozu/flywheel/backend/engine/batching/BatchingEngine.java index 3c35bfa3b..428bf5f29 100644 --- a/src/main/java/com/jozufozu/flywheel/backend/engine/batching/BatchingEngine.java +++ b/src/main/java/com/jozufozu/flywheel/backend/engine/batching/BatchingEngine.java @@ -93,13 +93,18 @@ public class BatchingEngine extends AbstractEngine implements SimplyComposedPlan drawTracker.draw(stage); } + @Override + public void renderCrumblingInstance(TaskExecutor taskExecutor, RenderContext context, Instance instance, int progress) { + // TODO: implement + } + @Override protected void onRenderOriginChanged() { initializedInstancers.forEach(BatchedInstancer::clear); } @Override - public void delete() { + public void invalidate() { instancers.clear(); meshPools.values() diff --git a/src/main/java/com/jozufozu/flywheel/backend/engine/indirect/IndirectEngine.java b/src/main/java/com/jozufozu/flywheel/backend/engine/indirect/IndirectEngine.java index 8f7ae86a6..5ccf5308a 100644 --- a/src/main/java/com/jozufozu/flywheel/backend/engine/indirect/IndirectEngine.java +++ b/src/main/java/com/jozufozu/flywheel/backend/engine/indirect/IndirectEngine.java @@ -68,6 +68,11 @@ public class IndirectEngine extends AbstractEngine { } } + @Override + public void renderCrumblingInstance(TaskExecutor taskExecutor, RenderContext context, Instance instance, int progress) { + // TODO: implement + } + private void setup() { GlTextureUnit.T2.makeActive(); Minecraft.getInstance().gameRenderer.lightTexture().turnOnLightLayer(); @@ -85,7 +90,7 @@ public class IndirectEngine extends AbstractEngine { } @Override - public void delete() { + public void invalidate() { drawManager.invalidate(); } } diff --git a/src/main/java/com/jozufozu/flywheel/backend/engine/instancing/InstancingEngine.java b/src/main/java/com/jozufozu/flywheel/backend/engine/instancing/InstancingEngine.java index 3e4716780..54951a074 100644 --- a/src/main/java/com/jozufozu/flywheel/backend/engine/instancing/InstancingEngine.java +++ b/src/main/java/com/jozufozu/flywheel/backend/engine/instancing/InstancingEngine.java @@ -75,6 +75,11 @@ public class InstancingEngine extends AbstractEngine { } } + @Override + public void renderCrumblingInstance(TaskExecutor taskExecutor, RenderContext context, Instance instance, int progress) { + // TODO: implement + } + private void setup() { GlTextureUnit.T2.makeActive(); Minecraft.getInstance().gameRenderer.lightTexture().turnOnLightLayer(); @@ -97,7 +102,7 @@ public class InstancingEngine extends AbstractEngine { continue; } - setup(shader); + setup(shader, context); shader.material().setup(); @@ -109,7 +114,7 @@ public class InstancingEngine extends AbstractEngine { } } - private void setup(ShaderState desc) { + private void setup(ShaderState desc, Context context) { var material = desc.material(); var vertexType = desc.vertexType(); var instanceType = desc.instanceType(); @@ -130,7 +135,7 @@ public class InstancingEngine extends AbstractEngine { } @Override - public void delete() { + public void invalidate() { drawManager.invalidate(); } } diff --git a/src/main/java/com/jozufozu/flywheel/impl/visualization/FrameContext.java b/src/main/java/com/jozufozu/flywheel/impl/visualization/FrameContext.java index 66b49578b..bf8945e4e 100644 --- a/src/main/java/com/jozufozu/flywheel/impl/visualization/FrameContext.java +++ b/src/main/java/com/jozufozu/flywheel/impl/visualization/FrameContext.java @@ -10,7 +10,7 @@ import net.minecraft.core.Vec3i; public record FrameContext(double cameraX, double cameraY, double cameraZ, FrustumIntersection frustum, float partialTick) { @NotNull - public static FrameContext create(RenderContext context, Vec3i renderOrigin, float partialTick) { + public static FrameContext create(RenderContext context, Vec3i renderOrigin) { var cameraPos = context.camera() .getPosition(); double cameraX = cameraPos.x; @@ -21,6 +21,6 @@ public record FrameContext(double cameraX, double cameraY, double cameraZ, Frust viewProjection.translate((float) (renderOrigin.getX() - cameraX), (float) (renderOrigin.getY() - cameraY), (float) (renderOrigin.getZ() - cameraZ)); FrustumIntersection frustum = new FrustumIntersection(viewProjection); - return new FrameContext(cameraX, cameraY, cameraZ, frustum, partialTick); + return new FrameContext(cameraX, cameraY, cameraZ, frustum, context.partialTick()); } } diff --git a/src/main/java/com/jozufozu/flywheel/impl/visualization/VisualizationManagerImpl.java b/src/main/java/com/jozufozu/flywheel/impl/visualization/VisualizationManagerImpl.java index 2f69307d2..ec4ede546 100644 --- a/src/main/java/com/jozufozu/flywheel/impl/visualization/VisualizationManagerImpl.java +++ b/src/main/java/com/jozufozu/flywheel/impl/visualization/VisualizationManagerImpl.java @@ -1,11 +1,14 @@ package com.jozufozu.flywheel.impl.visualization; +import java.util.SortedSet; + import org.jetbrains.annotations.Nullable; import com.jozufozu.flywheel.api.backend.BackendManager; import com.jozufozu.flywheel.api.backend.Engine; import com.jozufozu.flywheel.api.event.RenderContext; import com.jozufozu.flywheel.api.event.RenderStage; +import com.jozufozu.flywheel.api.instance.Instance; import com.jozufozu.flywheel.api.task.Plan; import com.jozufozu.flywheel.api.task.TaskExecutor; import com.jozufozu.flywheel.api.visual.DynamicVisual; @@ -20,15 +23,17 @@ import com.jozufozu.flywheel.impl.visualization.manager.BlockEntityVisualManager import com.jozufozu.flywheel.impl.visualization.manager.EffectVisualManager; import com.jozufozu.flywheel.impl.visualization.manager.EntityVisualManager; import com.jozufozu.flywheel.lib.task.Flag; +import com.jozufozu.flywheel.lib.task.MapContextPlan; import com.jozufozu.flywheel.lib.task.NamedFlag; -import com.jozufozu.flywheel.lib.task.NestedPlan; import com.jozufozu.flywheel.lib.task.RaisePlan; -import com.jozufozu.flywheel.lib.task.SimplyComposedPlan; +import com.jozufozu.flywheel.lib.task.IfElsePlan; import com.jozufozu.flywheel.lib.util.LevelAttached; +import it.unimi.dsi.fastutil.longs.Long2ObjectMap; import net.minecraft.client.Minecraft; import net.minecraft.client.multiplayer.ClientLevel; import net.minecraft.core.Vec3i; +import net.minecraft.server.level.BlockDestructionProgress; import net.minecraft.world.entity.Entity; import net.minecraft.world.level.LevelAccessor; import net.minecraft.world.level.block.entity.BlockEntity; @@ -50,7 +55,8 @@ public class VisualizationManagerImpl implements VisualizationManager { private final Plan framePlan; private final Flag tickFlag = new NamedFlag("tick"); - private final Flag frameFlag = new NamedFlag("frame"); + private final Flag frameVisualsFlag = new NamedFlag("frameVisualUpdates"); + private final Flag frameFlag = new NamedFlag("frameComplete"); private VisualizationManagerImpl(LevelAccessor level) { engine = BackendManager.getBackend() @@ -66,7 +72,21 @@ public class VisualizationManagerImpl implements VisualizationManager { .and(effects.createTickPlan()) .then(RaisePlan.raise(tickFlag)) .simplify(); - framePlan = new FramePlan().then(RaisePlan.raise(frameFlag)); + + framePlan = IfElsePlan.on((RenderContext ctx) -> engine.updateRenderOrigin(ctx.camera())) + .ifTrue(MapContextPlan.map(RenderContext::partialTick) + .to(blockEntities.createRecreationPlan() + .and(entities.createRecreationPlan()) + .and(effects.createRecreationPlan()))) + .ifFalse(MapContextPlan.map((RenderContext ctx) -> FrameContext.create(ctx, engine.renderOrigin())) + .to(blockEntities.createFramePlan() + .and(entities.createFramePlan()) + .and(effects.createFramePlan()))) + .plan() + .then(RaisePlan.raise(frameVisualsFlag)) + .then(engine.createFramePlan()) + .then(RaisePlan.raise(frameFlag)) + .simplify(); } public static boolean supportsVisualization(@Nullable LevelAccessor level) { @@ -171,6 +191,8 @@ public class VisualizationManagerImpl implements VisualizationManager { // Note we don't lower here because many frames may happen per tick. taskExecutor.syncUntil(tickFlag::isRaised); + frameVisualsFlag.lower(); + frameFlag.lower(); framePlan.execute(taskExecutor, context); } @@ -181,6 +203,41 @@ public class VisualizationManagerImpl implements VisualizationManager { engine.renderStage(taskExecutor, context, stage); } + public void renderCrumbling(RenderContext context, Long2ObjectMap> destructionProgress) { + taskExecutor.syncUntil(frameVisualsFlag::isRaised); + + for (var entry : destructionProgress.long2ObjectEntrySet()) { + var set = entry.getValue(); + if (set == null || set.isEmpty()) { + // Nothing to do if there's no crumbling. + continue; + } + + var visual = blockEntities.visualAtPos(entry.getLongKey()); + + if (visual == null) { + // The block doesn't have a visual, this is probably the common case. + continue; + } + + var instanceList = visual.getCrumblingInstances(); + + if (instanceList.isEmpty()) { + // The visual doesn't want to render anything crumbling. + continue; + } + + // now for the fun part + + int progress = set.last() + .getProgress(); + + for (Instance instance : instanceList) { + engine.renderCrumblingInstance(taskExecutor, context, instance, progress); + } + } + } + /** * Free all acquired resources and delete this manager. */ @@ -192,29 +249,6 @@ public class VisualizationManagerImpl implements VisualizationManager { blockEntities.invalidate(); entities.invalidate(); effects.invalidate(); - engine.delete(); - } - - private class FramePlan implements SimplyComposedPlan { - private final Plan recreationPlan = NestedPlan.of(blockEntities.createRecreationPlan(), entities.createRecreationPlan(), effects.createRecreationPlan()); - private final Plan normalPlan = blockEntities.createFramePlan() - .and(entities.createFramePlan()) - .and(effects.createFramePlan()); - - private final Plan enginePlan = engine.createFramePlan(); - - @Override - public void execute(TaskExecutor taskExecutor, RenderContext context, Runnable onCompletion) { - Runnable then = () -> enginePlan.execute(taskExecutor, context, onCompletion); - float partialTick = context.partialTick(); - - if (engine.updateRenderOrigin(context.camera())) { - recreationPlan.execute(taskExecutor, partialTick, then); - } else { - var frameContext = FrameContext.create(context, engine.renderOrigin(), partialTick); - - normalPlan.execute(taskExecutor, frameContext, then); - } - } + engine.invalidate(); } } diff --git a/src/main/java/com/jozufozu/flywheel/impl/visualization/manager/AbstractVisualManager.java b/src/main/java/com/jozufozu/flywheel/impl/visualization/manager/AbstractVisualManager.java index 1b0bbf5e7..9093e9fae 100644 --- a/src/main/java/com/jozufozu/flywheel/impl/visualization/manager/AbstractVisualManager.java +++ b/src/main/java/com/jozufozu/flywheel/impl/visualization/manager/AbstractVisualManager.java @@ -15,6 +15,7 @@ import com.jozufozu.flywheel.impl.visualization.ratelimit.DistanceUpdateLimiterI import com.jozufozu.flywheel.impl.visualization.ratelimit.NonLimiter; import com.jozufozu.flywheel.impl.visualization.storage.Storage; import com.jozufozu.flywheel.impl.visualization.storage.Transaction; +import com.jozufozu.flywheel.lib.task.MapContextPlan; import com.jozufozu.flywheel.lib.task.SimplePlan; public abstract class AbstractVisualManager implements VisualManager { @@ -88,7 +89,8 @@ public abstract class AbstractVisualManager implements VisualManager { tickLimiter.tick(); processQueue(0); }) - .thenMap(this::createVisualTickContext, getStorage().getTickPlan()); + .then(MapContextPlan.map(this::createVisualTickContext) + .to(getStorage().getTickPlan())); } public Plan createFramePlan() { @@ -96,7 +98,8 @@ public abstract class AbstractVisualManager implements VisualManager { frameLimiter.tick(); processQueue(context.partialTick()); }) - .thenMap(this::createVisualContext, getStorage().getFramePlan()); + .then(MapContextPlan.map(this::createVisualContext) + .to(getStorage().getFramePlan())); } private VisualFrameContext createVisualContext(FrameContext ctx) { diff --git a/src/main/java/com/jozufozu/flywheel/impl/visualization/manager/BlockEntityVisualManager.java b/src/main/java/com/jozufozu/flywheel/impl/visualization/manager/BlockEntityVisualManager.java index 91cc4f795..428046f18 100644 --- a/src/main/java/com/jozufozu/flywheel/impl/visualization/manager/BlockEntityVisualManager.java +++ b/src/main/java/com/jozufozu/flywheel/impl/visualization/manager/BlockEntityVisualManager.java @@ -1,7 +1,5 @@ package com.jozufozu.flywheel.impl.visualization.manager; -import java.util.List; - import org.jetbrains.annotations.Nullable; import com.jozufozu.flywheel.api.backend.Engine; @@ -30,11 +28,8 @@ public class BlockEntityVisualManager extends AbstractVisualManager return storage; } - public void getCrumblingVisuals(long pos, List> visuals) { - BlockEntityVisual visual = storage.posLookup.get(pos); - if (visual != null) { - visuals.add(visual); - } + public BlockEntityVisual visualAtPos(long pos) { + return storage.posLookup.get(pos); } private static class BlockEntityStorage extends Storage { diff --git a/src/main/java/com/jozufozu/flywheel/lib/instance/AbstractInstance.java b/src/main/java/com/jozufozu/flywheel/lib/instance/AbstractInstance.java index a4d17b1ec..33d790d12 100644 --- a/src/main/java/com/jozufozu/flywheel/lib/instance/AbstractInstance.java +++ b/src/main/java/com/jozufozu/flywheel/lib/instance/AbstractInstance.java @@ -18,6 +18,11 @@ public abstract class AbstractInstance implements Instance { return type; } + @Override + public InstanceHandle handle() { + return handle; + } + public final void setChanged() { handle.setChanged(); } diff --git a/src/main/java/com/jozufozu/flywheel/lib/task/IfElsePlan.java b/src/main/java/com/jozufozu/flywheel/lib/task/IfElsePlan.java new file mode 100644 index 000000000..e801e32d0 --- /dev/null +++ b/src/main/java/com/jozufozu/flywheel/lib/task/IfElsePlan.java @@ -0,0 +1,65 @@ +package com.jozufozu.flywheel.lib.task; + +import java.util.function.Predicate; + +import com.jozufozu.flywheel.api.task.Plan; +import com.jozufozu.flywheel.api.task.TaskExecutor; + +/** + * Executes one plan or another, depending on a dynamically evaluated condition. + * @param condition The condition to branch on. + * @param onTrue The plan to execute if the condition is true. + * @param onFalse The plan to execute if the condition is false. + * @param The type of the context object. + */ +public record IfElsePlan(Predicate condition, Plan onTrue, Plan onFalse) implements SimplyComposedPlan { + public static Builder on(Predicate condition) { + return new Builder<>(condition); + } + + @Override + public void execute(TaskExecutor taskExecutor, C context, Runnable onCompletion) { + if (condition.test(context)) { + onTrue.execute(taskExecutor, context, onCompletion); + } else { + onFalse.execute(taskExecutor, context, onCompletion); + } + } + + @Override + public Plan simplify() { + var maybeSimplifiedTrue = onTrue.simplify(); + var maybeSimplifiedFalse = onFalse.simplify(); + + if (maybeSimplifiedTrue instanceof UnitPlan && maybeSimplifiedFalse instanceof UnitPlan) { + // The condition may have side effects that still need to be evaluated. + return SimplePlan.of(condition::test); + } + + return new IfElsePlan<>(condition, maybeSimplifiedTrue, maybeSimplifiedFalse); + } + + public static class Builder { + private final Predicate condition; + private Plan onTrue = UnitPlan.of(); + private Plan onFalse = UnitPlan.of(); + + public Builder(Predicate condition) { + this.condition = condition; + } + + public Builder ifTrue(Plan onTrue) { + this.onTrue = onTrue; + return this; + } + + public Builder ifFalse(Plan onFalse) { + this.onFalse = onFalse; + return this; + } + + public IfElsePlan plan() { + return new IfElsePlan<>(condition, onTrue, onFalse); + } + } +} diff --git a/src/main/java/com/jozufozu/flywheel/lib/task/MapContextPlan.java b/src/main/java/com/jozufozu/flywheel/lib/task/MapContextPlan.java index e06f8fc9d..442f1fd10 100644 --- a/src/main/java/com/jozufozu/flywheel/lib/task/MapContextPlan.java +++ b/src/main/java/com/jozufozu/flywheel/lib/task/MapContextPlan.java @@ -6,6 +6,10 @@ import com.jozufozu.flywheel.api.task.Plan; import com.jozufozu.flywheel.api.task.TaskExecutor; public record MapContextPlan(Function map, Plan plan) implements SimplyComposedPlan { + public static Builder map(Function map) { + return new Builder<>(map); + } + @Override public void execute(TaskExecutor taskExecutor, C context, Runnable onCompletion) { D newContext = map.apply(context); @@ -22,4 +26,20 @@ public record MapContextPlan(Function map, Plan plan) implements return new MapContextPlan<>(map, maybeSimplified); } + + public static class Builder { + private final Function map; + + public Builder(Function map) { + this.map = map; + } + + public MapContextPlan to(Plan plan) { + return new MapContextPlan<>(map, plan); + } + + public MapContextPlan plan() { + return new MapContextPlan<>(map, UnitPlan.of()); + } + } } diff --git a/src/main/java/com/jozufozu/flywheel/lib/task/SimplyComposedPlan.java b/src/main/java/com/jozufozu/flywheel/lib/task/SimplyComposedPlan.java index 0747ddbbe..d3ffa68c6 100644 --- a/src/main/java/com/jozufozu/flywheel/lib/task/SimplyComposedPlan.java +++ b/src/main/java/com/jozufozu/flywheel/lib/task/SimplyComposedPlan.java @@ -10,21 +10,11 @@ public interface SimplyComposedPlan extends Plan { return new BarrierPlan<>(this, plan); } - @Override - default Plan thenMap(Function map, Plan plan) { - return then(new MapContextPlan<>(map, plan)); - } - @Override default Plan and(Plan plan) { return NestedPlan.of(this, plan); } - @Override - default Plan andMap(Function map, Plan plan) { - return and(new MapContextPlan<>(map, plan)); - } - @Override default Plan simplify() { return this; diff --git a/src/main/java/com/jozufozu/flywheel/lib/task/UnitPlan.java b/src/main/java/com/jozufozu/flywheel/lib/task/UnitPlan.java index 6723d97d1..2ad6fa8a8 100644 --- a/src/main/java/com/jozufozu/flywheel/lib/task/UnitPlan.java +++ b/src/main/java/com/jozufozu/flywheel/lib/task/UnitPlan.java @@ -26,21 +26,11 @@ public class UnitPlan implements Plan { return plan; } - @Override - public Plan thenMap(Function map, Plan plan) { - return new MapContextPlan<>(map, plan); - } - @Override public Plan and(Plan plan) { return plan; } - @Override - public Plan andMap(Function map, Plan plan) { - return new MapContextPlan<>(map, plan); - } - @Override public Plan simplify() { return this; diff --git a/src/main/java/com/jozufozu/flywheel/mixin/LevelRendererAccessor.java b/src/main/java/com/jozufozu/flywheel/mixin/LevelRendererAccessor.java deleted file mode 100644 index f306c68cb..000000000 --- a/src/main/java/com/jozufozu/flywheel/mixin/LevelRendererAccessor.java +++ /dev/null @@ -1,16 +0,0 @@ -package com.jozufozu.flywheel.mixin; - -import java.util.SortedSet; - -import org.spongepowered.asm.mixin.Mixin; -import org.spongepowered.asm.mixin.gen.Accessor; - -import it.unimi.dsi.fastutil.longs.Long2ObjectMap; -import net.minecraft.client.renderer.LevelRenderer; -import net.minecraft.server.level.BlockDestructionProgress; - -@Mixin(LevelRenderer.class) -public interface LevelRendererAccessor { - @Accessor("destructionProgress") - Long2ObjectMap> flywheel$getDestructionProgress(); -} diff --git a/src/main/java/com/jozufozu/flywheel/mixin/LevelRendererMixin.java b/src/main/java/com/jozufozu/flywheel/mixin/LevelRendererMixin.java index 12419bcad..e2d6d9a53 100644 --- a/src/main/java/com/jozufozu/flywheel/mixin/LevelRendererMixin.java +++ b/src/main/java/com/jozufozu/flywheel/mixin/LevelRendererMixin.java @@ -1,5 +1,7 @@ package com.jozufozu.flywheel.mixin; +import java.util.SortedSet; + import org.joml.Matrix4f; import org.objectweb.asm.Opcodes; import org.spongepowered.asm.mixin.Final; @@ -16,14 +18,17 @@ import com.jozufozu.flywheel.api.event.ReloadRenderersEvent; import com.jozufozu.flywheel.api.event.RenderContext; import com.jozufozu.flywheel.api.event.RenderStage; import com.jozufozu.flywheel.api.event.RenderStageEvent; +import com.jozufozu.flywheel.impl.visualization.VisualizationManagerImpl; import com.mojang.blaze3d.vertex.PoseStack; +import it.unimi.dsi.fastutil.longs.Long2ObjectMap; import net.minecraft.client.Camera; import net.minecraft.client.multiplayer.ClientLevel; import net.minecraft.client.renderer.GameRenderer; import net.minecraft.client.renderer.LevelRenderer; import net.minecraft.client.renderer.LightTexture; import net.minecraft.client.renderer.RenderBuffers; +import net.minecraft.server.level.BlockDestructionProgress; import net.minecraftforge.common.MinecraftForge; @Mixin(value = LevelRenderer.class, priority = 1001) // Higher priority to go after Sodium @@ -35,6 +40,10 @@ public class LevelRendererMixin { @Final private RenderBuffers renderBuffers; + @Shadow + @Final + private Long2ObjectMap> destructionProgress; + @Unique private RenderContext flywheel$renderContext; @@ -142,4 +151,12 @@ public class LevelRendererMixin { private void flywheel$onStage$afterWeather(CallbackInfo ci) { flywheel$dispatch(RenderStage.AFTER_WEATHER); } + + @Inject(method = "renderLevel", at = @At(value = "INVOKE_STRING", target = "Lnet/minecraft/util/profiling/ProfilerFiller;popPush(Ljava/lang/String;)V", args = "ldc=destroyProgress")) + private void flywheel$crumbling(CallbackInfo ci) { + var vm = VisualizationManagerImpl.get(level); + if (vm != null) { + vm.renderCrumbling(flywheel$renderContext, destructionProgress); + } + } } diff --git a/src/main/resources/flywheel.mixins.json b/src/main/resources/flywheel.mixins.json index c8e875c47..fbaa3ae9e 100644 --- a/src/main/resources/flywheel.mixins.json +++ b/src/main/resources/flywheel.mixins.json @@ -11,7 +11,6 @@ "EntityTypeMixin", "FogUpdateMixin", "GlStateManagerMixin", - "LevelRendererAccessor", "LevelRendererMixin", "LightUpdateMixin", "RenderTypeMixin",