From b0bc3d3145f5b82dbc666507341d2d3fe430c2ad Mon Sep 17 00:00:00 2001 From: Jozufozu Date: Thu, 1 Feb 2024 17:40:36 -0800 Subject: [PATCH] Flywheel ECS - Add SimpleEntityVisual which is composed of many EntityComponents. - Would be nice to expand on this system, particularly to add components when registering visualizers. - Misc. doc updates. --- .../flywheel/api/visual/LitVisual.java | 30 ++++++++-------- .../jozufozu/flywheel/api/visual/Visual.java | 10 ++++-- .../lib/visual/AbstractEntityVisual.java | 13 ------- .../flywheel/lib/visual/EntityComponent.java | 9 +++++ .../lib/visual/SimpleEntityVisual.java | 36 +++++++++++++++++++ .../BoundingBoxComponent.java | 11 ++++-- .../{ => components}/FireComponent.java | 8 +++-- .../{ => components}/ShadowComponent.java | 14 +++++--- .../flywheel/vanilla/MinecartVisual.java | 17 +++++---- 9 files changed, 102 insertions(+), 46 deletions(-) create mode 100644 src/main/java/com/jozufozu/flywheel/lib/visual/EntityComponent.java create mode 100644 src/main/java/com/jozufozu/flywheel/lib/visual/SimpleEntityVisual.java rename src/main/java/com/jozufozu/flywheel/lib/visual/{ => components}/BoundingBoxComponent.java (94%) rename src/main/java/com/jozufozu/flywheel/lib/visual/{ => components}/FireComponent.java (95%) rename src/main/java/com/jozufozu/flywheel/lib/visual/{ => components}/ShadowComponent.java (95%) diff --git a/src/main/java/com/jozufozu/flywheel/api/visual/LitVisual.java b/src/main/java/com/jozufozu/flywheel/api/visual/LitVisual.java index 12c4d7212..b4f1b0a32 100644 --- a/src/main/java/com/jozufozu/flywheel/api/visual/LitVisual.java +++ b/src/main/java/com/jozufozu/flywheel/api/visual/LitVisual.java @@ -5,28 +5,30 @@ import java.util.function.LongConsumer; import net.minecraft.core.SectionPos; /** - * A non-moving visual that listens to light updates. - *
- * If your visual moves around in the world at all, you should use {@link TickableVisual} or {@link DynamicVisual}, - * and poll for light yourself rather than listening for updates. + * A visual that listens to light updates. + * + *

If your visual moves around in the world at all, you should use {@link TickableVisual} or {@link DynamicVisual}, + * and poll for light yourself along with listening for updates. When your visual moves to a different section, call + * {@link Notifier#notifySectionsChanged}.

*/ public interface LitVisual extends Visual { /** * Called when a section this visual is contained in receives a light update. - *
- * Even if multiple sections are updated at the same time, this method will only be called once. - *
- * The implementation is free to parallelize calls to this method, as well as call into + * + *

Even if multiple sections are updated at the same time, this method will only be called once.

+ * + *

The implementation is free to parallelize calls to this method, as well as call into * {@link DynamicVisual#beginFrame} simultaneously. It is safe to query/update light here, * but you must ensure proper synchronization if you want to mutate anything outside this - * visual or anything that is also mutated by {@link DynamicVisual#beginFrame}. + * visual or anything that is also mutated by {@link DynamicVisual#beginFrame}.

*/ void updateLight(); /** * Collect the sections that this visual is contained in. - *
- * This method is called upon visual creation, and the frame after {@link Notifier#notifySectionsChanged} is called. + * + *

This method is called upon visual creation, and the frame after + * {@link Notifier#notifySectionsChanged} is called.

* * @param consumer The consumer to provide the sections to. * @see SectionPos#asLong @@ -35,9 +37,9 @@ public interface LitVisual extends Visual { /** * Set the notifier object. - *
- * This method is only called once, upon visual creation, - * after {@link #init} and before {@link #collectLightSections}. + * + *

This method is only called once, upon visual creation, + * after {@link #init} and before {@link #collectLightSections}.

* * @param notifier The notifier. */ diff --git a/src/main/java/com/jozufozu/flywheel/api/visual/Visual.java b/src/main/java/com/jozufozu/flywheel/api/visual/Visual.java index 5421d4397..7dac8dbf3 100644 --- a/src/main/java/com/jozufozu/flywheel/api/visual/Visual.java +++ b/src/main/java/com/jozufozu/flywheel/api/visual/Visual.java @@ -11,13 +11,17 @@ package com.jozufozu.flywheel.api.visual; public interface Visual { /** * Initialize instances here. + * + *

This method will be called exactly once upon visual creation.

*/ void init(float partialTick); /** - * Update instances here. Good for when instances don't change very often and when animations are GPU based. - *
- *
If your animations are complex or more CPU driven, see {@link DynamicVisual} or {@link TickableVisual}. + * Update instances here. + * + *

Good for when instances don't change very often and when animations are GPU based. + * + *
If your animations are complex or more CPU driven, see {@link DynamicVisual} or {@link TickableVisual}.

*/ void update(float partialTick); diff --git a/src/main/java/com/jozufozu/flywheel/lib/visual/AbstractEntityVisual.java b/src/main/java/com/jozufozu/flywheel/lib/visual/AbstractEntityVisual.java index 957608290..ebba08ccb 100644 --- a/src/main/java/com/jozufozu/flywheel/lib/visual/AbstractEntityVisual.java +++ b/src/main/java/com/jozufozu/flywheel/lib/visual/AbstractEntityVisual.java @@ -33,17 +33,11 @@ import net.minecraft.world.phys.Vec3; public abstract class AbstractEntityVisual extends AbstractVisual implements EntityVisual { protected final T entity; protected final EntityVisibilityTester visibilityTester; - protected final ShadowComponent shadow; - protected final FireComponent fire; - protected final BoundingBoxComponent boundingBox; public AbstractEntityVisual(VisualizationContext ctx, T entity) { super(ctx, entity.level()); this.entity = entity; visibilityTester = new EntityVisibilityTester(entity, ctx.renderOrigin(), 1.5f); - shadow = new ShadowComponent(ctx, entity); - fire = new FireComponent(ctx, entity); - boundingBox = new BoundingBoxComponent(ctx, entity); } @Override @@ -93,11 +87,4 @@ public abstract class AbstractEntityVisual extends AbstractVis public boolean isVisible(FrustumIntersection frustum) { return entity.noCulling || visibilityTester.check(frustum); } - - @Override - protected void _delete() { - shadow.delete(); - fire.delete(); - boundingBox.delete(); - } } diff --git a/src/main/java/com/jozufozu/flywheel/lib/visual/EntityComponent.java b/src/main/java/com/jozufozu/flywheel/lib/visual/EntityComponent.java new file mode 100644 index 000000000..d47ca3145 --- /dev/null +++ b/src/main/java/com/jozufozu/flywheel/lib/visual/EntityComponent.java @@ -0,0 +1,9 @@ +package com.jozufozu.flywheel.lib.visual; + +import com.jozufozu.flywheel.api.visual.VisualFrameContext; + +public interface EntityComponent { + void beginFrame(VisualFrameContext context); + + void delete(); +} diff --git a/src/main/java/com/jozufozu/flywheel/lib/visual/SimpleEntityVisual.java b/src/main/java/com/jozufozu/flywheel/lib/visual/SimpleEntityVisual.java new file mode 100644 index 000000000..5b810787b --- /dev/null +++ b/src/main/java/com/jozufozu/flywheel/lib/visual/SimpleEntityVisual.java @@ -0,0 +1,36 @@ +package com.jozufozu.flywheel.lib.visual; + +import java.util.ArrayList; +import java.util.List; + +import com.jozufozu.flywheel.api.visual.DynamicVisual; +import com.jozufozu.flywheel.api.visual.VisualFrameContext; +import com.jozufozu.flywheel.api.visualization.VisualizationContext; + +import net.minecraft.world.entity.Entity; + +public class SimpleEntityVisual extends AbstractEntityVisual implements DynamicVisual { + protected final List components = new ArrayList<>(); + + public SimpleEntityVisual(VisualizationContext ctx, T entity) { + super(ctx, entity); + } + + public void addComponent(EntityComponent component) { + components.add(component); + } + + @Override + public void beginFrame(VisualFrameContext ctx) { + for (EntityComponent component : components) { + component.beginFrame(ctx); + } + } + + @Override + protected void _delete() { + for (EntityComponent component : components) { + component.delete(); + } + } +} diff --git a/src/main/java/com/jozufozu/flywheel/lib/visual/BoundingBoxComponent.java b/src/main/java/com/jozufozu/flywheel/lib/visual/components/BoundingBoxComponent.java similarity index 94% rename from src/main/java/com/jozufozu/flywheel/lib/visual/BoundingBoxComponent.java rename to src/main/java/com/jozufozu/flywheel/lib/visual/components/BoundingBoxComponent.java index d64b0b78c..72a61f0a6 100644 --- a/src/main/java/com/jozufozu/flywheel/lib/visual/BoundingBoxComponent.java +++ b/src/main/java/com/jozufozu/flywheel/lib/visual/components/BoundingBoxComponent.java @@ -1,4 +1,4 @@ -package com.jozufozu.flywheel.lib.visual; +package com.jozufozu.flywheel.lib.visual.components; import org.joml.Vector4f; import org.joml.Vector4fc; @@ -15,6 +15,8 @@ import com.jozufozu.flywheel.lib.material.StandardMaterialShaders; import com.jozufozu.flywheel.lib.math.MoreMath; import com.jozufozu.flywheel.lib.model.QuadMesh; import com.jozufozu.flywheel.lib.model.SingleMeshModel; +import com.jozufozu.flywheel.lib.visual.EntityComponent; +import com.jozufozu.flywheel.lib.visual.InstanceRecycler; import net.minecraft.client.Minecraft; import net.minecraft.client.renderer.LightTexture; @@ -22,7 +24,7 @@ import net.minecraft.util.Mth; import net.minecraft.world.entity.Entity; import net.minecraft.world.entity.LivingEntity; -public class BoundingBoxComponent { +public class BoundingBoxComponent implements EntityComponent { private static final Material MATERIAL = SimpleMaterial.builder() .shaders(StandardMaterialShaders.WIREFRAME) .backfaceCulling(false) @@ -53,10 +55,12 @@ public class BoundingBoxComponent { return instance; } - public void showEyeBox(boolean renderEyeBox) { + public BoundingBoxComponent showEyeBox(boolean renderEyeBox) { this.showEyeBox = renderEyeBox; + return this; } + @Override public void beginFrame(VisualFrameContext context) { recycler.resetCount(); @@ -93,6 +97,7 @@ public class BoundingBoxComponent { recycler.discardExtra(); } + @Override public void delete() { recycler.delete(); } diff --git a/src/main/java/com/jozufozu/flywheel/lib/visual/FireComponent.java b/src/main/java/com/jozufozu/flywheel/lib/visual/components/FireComponent.java similarity index 95% rename from src/main/java/com/jozufozu/flywheel/lib/visual/FireComponent.java rename to src/main/java/com/jozufozu/flywheel/lib/visual/components/FireComponent.java index 15ce01bdc..3b4a04bca 100644 --- a/src/main/java/com/jozufozu/flywheel/lib/visual/FireComponent.java +++ b/src/main/java/com/jozufozu/flywheel/lib/visual/components/FireComponent.java @@ -1,4 +1,4 @@ -package com.jozufozu.flywheel.lib.visual; +package com.jozufozu.flywheel.lib.visual.components; import org.joml.Vector4f; import org.joml.Vector4fc; @@ -14,6 +14,8 @@ import com.jozufozu.flywheel.lib.material.SimpleMaterial; import com.jozufozu.flywheel.lib.model.ModelCache; import com.jozufozu.flywheel.lib.model.QuadMesh; import com.jozufozu.flywheel.lib.model.SingleMeshModel; +import com.jozufozu.flywheel.lib.visual.EntityComponent; +import com.jozufozu.flywheel.lib.visual.InstanceRecycler; import com.mojang.blaze3d.vertex.PoseStack; import com.mojang.math.Axis; @@ -26,7 +28,7 @@ import net.minecraft.world.entity.Entity; /** * A component that uses instances to render the fire animation on an entity. */ -public class FireComponent { +public class FireComponent implements EntityComponent { private static final Material FIRE_MATERIAL = SimpleMaterial.builderOf(Materials.CHUNK_CUTOUT_UNSHADED) .backfaceCulling(false) // Disable backface because we want to be able to flip the model. .build(); @@ -66,6 +68,7 @@ public class FireComponent { * * @param context The frame context. */ + @Override public void beginFrame(VisualFrameContext context) { fire0.resetCount(); fire1.resetCount(); @@ -118,6 +121,7 @@ public class FireComponent { } } + @Override public void delete() { fire0.delete(); fire1.delete(); diff --git a/src/main/java/com/jozufozu/flywheel/lib/visual/ShadowComponent.java b/src/main/java/com/jozufozu/flywheel/lib/visual/components/ShadowComponent.java similarity index 95% rename from src/main/java/com/jozufozu/flywheel/lib/visual/ShadowComponent.java rename to src/main/java/com/jozufozu/flywheel/lib/visual/components/ShadowComponent.java index b4d8930c2..572b7e353 100644 --- a/src/main/java/com/jozufozu/flywheel/lib/visual/ShadowComponent.java +++ b/src/main/java/com/jozufozu/flywheel/lib/visual/components/ShadowComponent.java @@ -1,4 +1,4 @@ -package com.jozufozu.flywheel.lib.visual; +package com.jozufozu.flywheel.lib.visual.components; import org.jetbrains.annotations.Nullable; import org.joml.Vector4f; @@ -16,6 +16,8 @@ import com.jozufozu.flywheel.lib.instance.ShadowInstance; import com.jozufozu.flywheel.lib.material.SimpleMaterial; import com.jozufozu.flywheel.lib.model.QuadMesh; import com.jozufozu.flywheel.lib.model.SingleMeshModel; +import com.jozufozu.flywheel.lib.visual.EntityComponent; +import com.jozufozu.flywheel.lib.visual.InstanceRecycler; import net.minecraft.client.Minecraft; import net.minecraft.client.renderer.LightTexture; @@ -40,7 +42,7 @@ import net.minecraft.world.phys.shapes.VoxelShape; *
* The shadow will be cast on blocks at most {@code min(radius, 2 * strength)} blocks below the entity.

*/ -public class ShadowComponent { +public class ShadowComponent implements EntityComponent { private static final Material SHADOW_MATERIAL = SimpleMaterial.builder() .texture(new ResourceLocation("textures/misc/shadow.png")) .mipmap(false) @@ -88,8 +90,9 @@ public class ShadowComponent { * * @param radius The radius of the shadow, in blocks. */ - public void radius(float radius) { + public ShadowComponent radius(float radius) { this.radius = Math.min(radius, 32); + return this; } /** @@ -97,8 +100,9 @@ public class ShadowComponent { * * @param strength The strength of the shadow. */ - public void strength(float strength) { + public ShadowComponent strength(float strength) { this.strength = strength; + return this; } /** @@ -107,6 +111,7 @@ public class ShadowComponent { * * @param context The frame context. */ + @Override public void beginFrame(VisualFrameContext context) { instances.resetCount(); @@ -210,6 +215,7 @@ public class ShadowComponent { return shape; } + @Override public void delete() { instances.delete(); } diff --git a/src/main/java/com/jozufozu/flywheel/vanilla/MinecartVisual.java b/src/main/java/com/jozufozu/flywheel/vanilla/MinecartVisual.java index 2fb19d7e5..5b3a64c95 100644 --- a/src/main/java/com/jozufozu/flywheel/vanilla/MinecartVisual.java +++ b/src/main/java/com/jozufozu/flywheel/vanilla/MinecartVisual.java @@ -14,7 +14,10 @@ import com.jozufozu.flywheel.lib.model.ModelHolder; import com.jozufozu.flywheel.lib.model.Models; import com.jozufozu.flywheel.lib.model.SingleMeshModel; import com.jozufozu.flywheel.lib.model.part.ModelPartConverter; -import com.jozufozu.flywheel.lib.visual.AbstractEntityVisual; +import com.jozufozu.flywheel.lib.visual.SimpleEntityVisual; +import com.jozufozu.flywheel.lib.visual.components.BoundingBoxComponent; +import com.jozufozu.flywheel.lib.visual.components.FireComponent; +import com.jozufozu.flywheel.lib.visual.components.ShadowComponent; import com.mojang.blaze3d.vertex.PoseStack; import com.mojang.math.Axis; @@ -26,7 +29,7 @@ import net.minecraft.world.level.block.RenderShape; import net.minecraft.world.level.block.state.BlockState; import net.minecraft.world.phys.Vec3; -public class MinecartVisual extends AbstractEntityVisual implements TickableVisual, DynamicVisual { +public class MinecartVisual extends SimpleEntityVisual implements TickableVisual, DynamicVisual { public static final ModelHolder CHEST_BODY_MODEL = createBodyModelHolder(ModelLayers.CHEST_MINECART); public static final ModelHolder COMMAND_BLOCK_BODY_MODEL = createBodyModelHolder(ModelLayers.COMMAND_BLOCK_MINECART); public static final ModelHolder FURNACE_BODY_MODEL = createBodyModelHolder(ModelLayers.FURNACE_MINECART); @@ -48,7 +51,6 @@ public class MinecartVisual extends AbstractEntityVi public MinecartVisual(VisualizationContext ctx, T entity, ModelHolder bodyModel) { super(ctx, entity); this.bodyModel = bodyModel; - shadow.radius(0.7f); } private static ModelHolder createBodyModelHolder(ModelLayerLocation layer) { @@ -59,6 +61,10 @@ public class MinecartVisual extends AbstractEntityVi @Override public void init(float partialTick) { + addComponent(new ShadowComponent(visualizationContext, entity).radius(0.7f)); + addComponent(new FireComponent(visualizationContext, entity)); + addComponent(new BoundingBoxComponent(visualizationContext, entity)); + body = createBodyInstance(); blockState = entity.getDisplayBlockState(); contents = createContentsInstance(); @@ -109,9 +115,7 @@ public class MinecartVisual extends AbstractEntityVi @Override public void beginFrame(VisualFrameContext context) { - shadow.beginFrame(context); - fire.beginFrame(context); - boundingBox.beginFrame(context); + super.beginFrame(context); if (!isVisible(context.frustum())) { return; @@ -210,7 +214,6 @@ public class MinecartVisual extends AbstractEntityVi if (contents != null) { contents.delete(); } - super._delete(); } public static boolean shouldSkipRender(AbstractMinecart minecart) {