Explosive changes

- Fix TNT minecarts' contents not receiving overlay and scaling
- Fix all types of minecarts using the body model of the standard
minecart
- Add overlay to ColoredLitInstance
- Make default chunk materials not disable overlay
- Rename *Visualizer.shouldSkipRender -> skipVanillaRender
- Rename Simple*Visualizer.configure -> builder
- Rename Simple*Visualizer.*Config -> Builder
- Make Simple*Visualizer.Builder always skip vanilla render by default
- Remove DiffuseLightCalculator and associated methods in RenderMath
- Add BackendImplemented annotation to Backend
- Mark PoseTransformStack as Internal at the class level
This commit is contained in:
PepperCode1 2024-01-25 13:14:39 -08:00
parent 7cfaf06f36
commit 72c755ae13
18 changed files with 284 additions and 226 deletions

View file

@ -1,11 +1,13 @@
package com.jozufozu.flywheel.api.backend; package com.jozufozu.flywheel.api.backend;
import com.jozufozu.flywheel.api.BackendImplemented;
import com.jozufozu.flywheel.api.registry.IdRegistry; import com.jozufozu.flywheel.api.registry.IdRegistry;
import com.jozufozu.flywheel.impl.IdRegistryImpl; import com.jozufozu.flywheel.impl.IdRegistryImpl;
import net.minecraft.network.chat.Component; import net.minecraft.network.chat.Component;
import net.minecraft.world.level.LevelAccessor; import net.minecraft.world.level.LevelAccessor;
@BackendImplemented
public interface Backend { public interface Backend {
static IdRegistry<Backend> REGISTRY = IdRegistryImpl.create(); static IdRegistry<Backend> REGISTRY = IdRegistryImpl.create();

View file

@ -2,6 +2,7 @@ package com.jozufozu.flywheel.api.visualization;
import com.jozufozu.flywheel.api.visual.BlockEntityVisual; import com.jozufozu.flywheel.api.visual.BlockEntityVisual;
import net.minecraft.client.renderer.blockentity.BlockEntityRenderer;
import net.minecraft.world.level.block.entity.BlockEntity; import net.minecraft.world.level.block.entity.BlockEntity;
/** /**
@ -19,9 +20,9 @@ public interface BlockEntityVisualizer<T extends BlockEntity> {
BlockEntityVisual<? super T> createVisual(VisualizationContext ctx, T blockEntity); BlockEntityVisual<? super T> createVisual(VisualizationContext ctx, T blockEntity);
/** /**
* Checks if the given block entity should not be rendered normally. * Checks if the given block entity should not be rendered with the vanilla {@link BlockEntityRenderer}.
* @param blockEntity The block entity to check. * @param blockEntity The block entity to check.
* @return {@code true} if the block entity should not be rendered normally, {@code false} if it should. * @return {@code true} if the block entity should not be rendered with the vanilla {@link BlockEntityRenderer}, {@code false} if it should.
*/ */
boolean shouldSkipRender(T blockEntity); boolean skipVanillaRender(T blockEntity);
} }

View file

@ -2,6 +2,7 @@ package com.jozufozu.flywheel.api.visualization;
import com.jozufozu.flywheel.api.visual.EntityVisual; import com.jozufozu.flywheel.api.visual.EntityVisual;
import net.minecraft.client.renderer.entity.EntityRenderer;
import net.minecraft.world.entity.Entity; import net.minecraft.world.entity.Entity;
/** /**
@ -19,9 +20,9 @@ public interface EntityVisualizer<T extends Entity> {
EntityVisual<? super T> createVisual(VisualizationContext ctx, T entity); EntityVisual<? super T> createVisual(VisualizationContext ctx, T entity);
/** /**
* Checks if the given entity should not render normally. * Checks if the given entity should not render with the vanilla {@link EntityRenderer}.
* @param entity The entity to check. * @param entity The entity to check.
* @return {@code true} if the entity should not render normally, {@code false} if it should. * @return {@code true} if the entity should not render with the vanilla {@link EntityRenderer}, {@code false} if it should.
*/ */
boolean shouldSkipRender(T entity); boolean skipVanillaRender(T entity);
} }

View file

@ -60,7 +60,7 @@ public final class VisualizationHelper {
if (visualizer == null) { if (visualizer == null) {
return false; return false;
} }
return visualizer.shouldSkipRender(blockEntity); return visualizer.skipVanillaRender(blockEntity);
} }
/** /**
@ -74,7 +74,7 @@ public final class VisualizationHelper {
if (visualizer == null) { if (visualizer == null) {
return false; return false;
} }
return visualizer.shouldSkipRender(entity); return visualizer.skipVanillaRender(entity);
} }
public static <T extends BlockEntity> boolean tryAddBlockEntity(T blockEntity) { public static <T extends BlockEntity> boolean tryAddBlockEntity(T blockEntity) {
@ -91,6 +91,6 @@ public final class VisualizationHelper {
manager.getBlockEntities().queueAdd(blockEntity); manager.getBlockEntities().queueAdd(blockEntity);
return visualizer.shouldSkipRender(blockEntity); return visualizer.skipVanillaRender(blockEntity);
} }
} }

View file

@ -4,37 +4,23 @@ import com.jozufozu.flywheel.api.instance.InstanceHandle;
import com.jozufozu.flywheel.api.instance.InstanceType; import com.jozufozu.flywheel.api.instance.InstanceType;
import net.minecraft.client.renderer.LightTexture; import net.minecraft.client.renderer.LightTexture;
import net.minecraft.client.renderer.texture.OverlayTexture;
public abstract class ColoredLitInstance extends AbstractInstance implements FlatLit { public abstract class ColoredLitInstance extends AbstractInstance implements FlatLit {
public byte blockLight;
public byte skyLight;
public byte r = (byte) 0xFF; public byte r = (byte) 0xFF;
public byte g = (byte) 0xFF; public byte g = (byte) 0xFF;
public byte b = (byte) 0xFF; public byte b = (byte) 0xFF;
public byte a = (byte) 0xFF; public byte a = (byte) 0xFF;
public byte blockLight;
public byte skyLight;
public int overlay = OverlayTexture.NO_OVERLAY;
public ColoredLitInstance(InstanceType<? extends ColoredLitInstance> type, InstanceHandle handle) { public ColoredLitInstance(InstanceType<? extends ColoredLitInstance> type, InstanceHandle handle) {
super(type, handle); super(type, handle);
} }
@Override
public ColoredLitInstance setBlockLight(int blockLight) {
this.blockLight = (byte) blockLight;
return this;
}
@Override
public ColoredLitInstance setSkyLight(int skyLight) {
this.skyLight = (byte) skyLight;
return this;
}
@Override
public int getPackedLight() {
return LightTexture.pack(blockLight, skyLight);
}
public ColoredLitInstance setColor(int color) { public ColoredLitInstance setColor(int color) {
return setColor(color, false); return setColor(color, false);
} }
@ -70,4 +56,26 @@ public abstract class ColoredLitInstance extends AbstractInstance implements Fla
this.a = a; this.a = a;
return this; return this;
} }
@Override
public ColoredLitInstance setBlockLight(int blockLight) {
this.blockLight = (byte) blockLight;
return this;
}
@Override
public ColoredLitInstance setSkyLight(int skyLight) {
this.skyLight = (byte) skyLight;
return this;
}
@Override
public int getPackedLight() {
return LightTexture.pack(blockLight, skyLight);
}
public ColoredLitInstance setOverlay(int overlay) {
this.overlay = overlay;
return this;
}
} }

View file

@ -13,20 +13,22 @@ import com.jozufozu.flywheel.lib.math.MatrixMath;
public final class InstanceTypes { public final class InstanceTypes {
public static final InstanceType<TransformedInstance> TRANSFORMED = SimpleInstanceType.builder(TransformedInstance::new) public static final InstanceType<TransformedInstance> TRANSFORMED = SimpleInstanceType.builder(TransformedInstance::new)
.layout(LayoutBuilder.create() .layout(LayoutBuilder.create()
.vector("light", IntegerRepr.SHORT, 2)
.vector("color", FloatRepr.NORMALIZED_UNSIGNED_BYTE, 4) .vector("color", FloatRepr.NORMALIZED_UNSIGNED_BYTE, 4)
.vector("light", IntegerRepr.SHORT, 2)
.vector("overlay", IntegerRepr.SHORT, 2)
.matrix("pose", FloatRepr.FLOAT, 4) .matrix("pose", FloatRepr.FLOAT, 4)
.matrix("normal", FloatRepr.FLOAT, 3) .matrix("normal", FloatRepr.FLOAT, 3)
.build()) .build())
.writer((ptr, instance) -> { .writer((ptr, instance) -> {
MemoryUtil.memPutShort(ptr, instance.blockLight); MemoryUtil.memPutByte(ptr, instance.r);
MemoryUtil.memPutShort(ptr + 2, instance.skyLight); MemoryUtil.memPutByte(ptr + 1, instance.g);
MemoryUtil.memPutByte(ptr + 4, instance.r); MemoryUtil.memPutByte(ptr + 2, instance.b);
MemoryUtil.memPutByte(ptr + 5, instance.g); MemoryUtil.memPutByte(ptr + 3, instance.a);
MemoryUtil.memPutByte(ptr + 6, instance.b); MemoryUtil.memPutShort(ptr + 4, instance.blockLight);
MemoryUtil.memPutByte(ptr + 7, instance.a); MemoryUtil.memPutShort(ptr + 6, instance.skyLight);
MatrixMath.writeUnsafe(instance.model, ptr + 8); MemoryUtil.memPutInt(ptr + 8, instance.overlay);
MatrixMath.writeUnsafe(instance.normal, ptr + 72); MatrixMath.writeUnsafe(instance.model, ptr + 12);
MatrixMath.writeUnsafe(instance.normal, ptr + 76);
}) })
.vertexShader(Flywheel.rl("instance/transformed.vert")) .vertexShader(Flywheel.rl("instance/transformed.vert"))
.cullShader(Flywheel.rl("instance/cull/transformed.glsl")) .cullShader(Flywheel.rl("instance/cull/transformed.glsl"))
@ -34,29 +36,31 @@ public final class InstanceTypes {
public static final InstanceType<OrientedInstance> ORIENTED = SimpleInstanceType.builder(OrientedInstance::new) public static final InstanceType<OrientedInstance> ORIENTED = SimpleInstanceType.builder(OrientedInstance::new)
.layout(LayoutBuilder.create() .layout(LayoutBuilder.create()
.vector("light", IntegerRepr.SHORT, 2)
.vector("color", FloatRepr.NORMALIZED_UNSIGNED_BYTE, 4) .vector("color", FloatRepr.NORMALIZED_UNSIGNED_BYTE, 4)
.vector("light", IntegerRepr.SHORT, 2)
.vector("overlay", IntegerRepr.SHORT, 2)
.vector("position", FloatRepr.FLOAT, 3) .vector("position", FloatRepr.FLOAT, 3)
.vector("pivot", FloatRepr.FLOAT, 3) .vector("pivot", FloatRepr.FLOAT, 3)
.vector("rotation", FloatRepr.FLOAT, 4) .vector("rotation", FloatRepr.FLOAT, 4)
.build()) .build())
.writer((ptr, instance) -> { .writer((ptr, instance) -> {
MemoryUtil.memPutShort(ptr, instance.blockLight); MemoryUtil.memPutByte(ptr, instance.r);
MemoryUtil.memPutShort(ptr + 2, instance.skyLight); MemoryUtil.memPutByte(ptr + 1, instance.g);
MemoryUtil.memPutByte(ptr + 4, instance.r); MemoryUtil.memPutByte(ptr + 2, instance.b);
MemoryUtil.memPutByte(ptr + 5, instance.g); MemoryUtil.memPutByte(ptr + 3, instance.a);
MemoryUtil.memPutByte(ptr + 6, instance.b); MemoryUtil.memPutShort(ptr + 4, instance.blockLight);
MemoryUtil.memPutByte(ptr + 7, instance.a); MemoryUtil.memPutShort(ptr + 6, instance.skyLight);
MemoryUtil.memPutFloat(ptr + 8, instance.posX); MemoryUtil.memPutInt(ptr + 8, instance.overlay);
MemoryUtil.memPutFloat(ptr + 12, instance.posY); MemoryUtil.memPutFloat(ptr + 12, instance.posX);
MemoryUtil.memPutFloat(ptr + 16, instance.posZ); MemoryUtil.memPutFloat(ptr + 16, instance.posY);
MemoryUtil.memPutFloat(ptr + 20, instance.pivotX); MemoryUtil.memPutFloat(ptr + 20, instance.posZ);
MemoryUtil.memPutFloat(ptr + 24, instance.pivotY); MemoryUtil.memPutFloat(ptr + 24, instance.pivotX);
MemoryUtil.memPutFloat(ptr + 28, instance.pivotZ); MemoryUtil.memPutFloat(ptr + 28, instance.pivotY);
MemoryUtil.memPutFloat(ptr + 32, instance.rotation.x); MemoryUtil.memPutFloat(ptr + 32, instance.pivotZ);
MemoryUtil.memPutFloat(ptr + 36, instance.rotation.y); MemoryUtil.memPutFloat(ptr + 36, instance.rotation.x);
MemoryUtil.memPutFloat(ptr + 40, instance.rotation.z); MemoryUtil.memPutFloat(ptr + 40, instance.rotation.y);
MemoryUtil.memPutFloat(ptr + 44, instance.rotation.w); MemoryUtil.memPutFloat(ptr + 44, instance.rotation.z);
MemoryUtil.memPutFloat(ptr + 48, instance.rotation.w);
}) })
.vertexShader(Flywheel.rl("instance/oriented.vert")) .vertexShader(Flywheel.rl("instance/oriented.vert"))
.cullShader(Flywheel.rl("instance/cull/oriented.glsl")) .cullShader(Flywheel.rl("instance/cull/oriented.glsl"))

View file

@ -11,61 +11,18 @@ import net.minecraft.core.Vec3i;
import net.minecraft.world.phys.Vec3; import net.minecraft.world.phys.Vec3;
public class OrientedInstance extends ColoredLitInstance implements Rotate<OrientedInstance> { public class OrientedInstance extends ColoredLitInstance implements Rotate<OrientedInstance> {
public final Quaternionf rotation = new Quaternionf();
public float pivotX = 0.5f;
public float pivotY = 0.5f;
public float pivotZ = 0.5f;
public float posX; public float posX;
public float posY; public float posY;
public float posZ; public float posZ;
public float pivotX = 0.5f;
public float pivotY = 0.5f;
public float pivotZ = 0.5f;
public final Quaternionf rotation = new Quaternionf();
public OrientedInstance(InstanceType<? extends OrientedInstance> type, InstanceHandle handle) { public OrientedInstance(InstanceType<? extends OrientedInstance> type, InstanceHandle handle) {
super(type, handle); super(type, handle);
} }
@Override
public OrientedInstance rotate(Quaternionf quaternion) {
rotation.mul(quaternion);
return this;
}
public OrientedInstance setRotation(Quaternionf q) {
rotation.set(q);
return this;
}
public OrientedInstance setRotation(float x, float y, float z, float w) {
rotation.set(x, y, z, w);
return this;
}
public OrientedInstance resetRotation() {
rotation.identity();
return this;
}
public OrientedInstance setPivot(float x, float y, float z) {
pivotX = x;
pivotY = y;
pivotZ = z;
return this;
}
public OrientedInstance setPivot(Vector3f pos) {
return setPivot(pos.x(), pos.y(), pos.z());
}
public OrientedInstance setPivot(Vec3 pos) {
return setPivot((float) pos.x(), (float) pos.y(), (float) pos.z());
}
public OrientedInstance nudgePosition(float x, float y, float z) {
posX += x;
posY += y;
posZ += z;
return this;
}
public OrientedInstance setPosition(float x, float y, float z) { public OrientedInstance setPosition(float x, float y, float z) {
posX = x; posX = x;
posY = y; posY = y;
@ -81,7 +38,58 @@ public class OrientedInstance extends ColoredLitInstance implements Rotate<Orien
return setPosition(pos.getX(), pos.getY(), pos.getZ()); return setPosition(pos.getX(), pos.getY(), pos.getZ());
} }
public OrientedInstance setPosition(Vec3 pos) {
return setPosition((float) pos.x(), (float) pos.y(), (float) pos.z());
}
public OrientedInstance resetPosition() { public OrientedInstance resetPosition() {
return setPosition(0, 0, 0); return setPosition(0, 0, 0);
} }
public OrientedInstance nudgePosition(float x, float y, float z) {
posX += x;
posY += y;
posZ += z;
return this;
}
public OrientedInstance setPivot(float x, float y, float z) {
pivotX = x;
pivotY = y;
pivotZ = z;
return this;
}
public OrientedInstance setPivot(Vector3f pos) {
return setPivot(pos.x(), pos.y(), pos.z());
}
public OrientedInstance setPivot(Vec3i pos) {
return setPivot(pos.getX(), pos.getY(), pos.getZ());
}
public OrientedInstance setPivot(Vec3 pos) {
return setPivot((float) pos.x(), (float) pos.y(), (float) pos.z());
}
public OrientedInstance setRotation(Quaternionf q) {
rotation.set(q);
return this;
}
public OrientedInstance setRotation(float x, float y, float z, float w) {
rotation.set(x, y, z, w);
return this;
}
public OrientedInstance resetRotation() {
rotation.identity();
return this;
}
@Override
public OrientedInstance rotate(Quaternionf quaternion) {
rotation.mul(quaternion);
return this;
}
} }

View file

@ -10,54 +10,44 @@ public final class Materials {
private static final ResourceLocation MINECART_LOCATION = new ResourceLocation("textures/entity/minecart.png"); private static final ResourceLocation MINECART_LOCATION = new ResourceLocation("textures/entity/minecart.png");
public static final Material CHUNK_SOLID_SHADED = SimpleMaterial.builder() public static final Material CHUNK_SOLID_SHADED = SimpleMaterial.builder()
.useOverlay(false)
.build(); .build();
public static final Material CHUNK_SOLID_UNSHADED = SimpleMaterial.builder() public static final Material CHUNK_SOLID_UNSHADED = SimpleMaterial.builder()
.useOverlay(false)
.diffuse(false) .diffuse(false)
.build(); .build();
public static final Material CHUNK_CUTOUT_MIPPED_SHADED = SimpleMaterial.builder() public static final Material CHUNK_CUTOUT_MIPPED_SHADED = SimpleMaterial.builder()
.cutout(CutoutShaders.HALF) .cutout(CutoutShaders.HALF)
.useOverlay(false)
.build(); .build();
public static final Material CHUNK_CUTOUT_MIPPED_UNSHADED = SimpleMaterial.builder() public static final Material CHUNK_CUTOUT_MIPPED_UNSHADED = SimpleMaterial.builder()
.cutout(CutoutShaders.HALF) .cutout(CutoutShaders.HALF)
.useOverlay(false)
.diffuse(false) .diffuse(false)
.build(); .build();
public static final Material CHUNK_CUTOUT_SHADED = SimpleMaterial.builder() public static final Material CHUNK_CUTOUT_SHADED = SimpleMaterial.builder()
.cutout(CutoutShaders.ONE_TENTH) .cutout(CutoutShaders.ONE_TENTH)
.mipmap(false) .mipmap(false)
.useOverlay(false)
.build(); .build();
public static final Material CHUNK_CUTOUT_UNSHADED = SimpleMaterial.builder() public static final Material CHUNK_CUTOUT_UNSHADED = SimpleMaterial.builder()
.cutout(CutoutShaders.ONE_TENTH) .cutout(CutoutShaders.ONE_TENTH)
.mipmap(false) .mipmap(false)
.useOverlay(false)
.diffuse(false) .diffuse(false)
.build(); .build();
public static final Material CHUNK_TRANSLUCENT_SHADED = SimpleMaterial.builder() public static final Material CHUNK_TRANSLUCENT_SHADED = SimpleMaterial.builder()
.transparency(Transparency.TRANSLUCENT) .transparency(Transparency.TRANSLUCENT)
.useOverlay(false)
.build(); .build();
public static final Material CHUNK_TRANSLUCENT_UNSHADED = SimpleMaterial.builder() public static final Material CHUNK_TRANSLUCENT_UNSHADED = SimpleMaterial.builder()
.transparency(Transparency.TRANSLUCENT) .transparency(Transparency.TRANSLUCENT)
.useOverlay(false)
.diffuse(false) .diffuse(false)
.build(); .build();
public static final Material CHUNK_TRIPWIRE_SHADED = SimpleMaterial.builder() public static final Material CHUNK_TRIPWIRE_SHADED = SimpleMaterial.builder()
.cutout(CutoutShaders.ONE_TENTH) .cutout(CutoutShaders.ONE_TENTH)
.transparency(Transparency.TRANSLUCENT) .transparency(Transparency.TRANSLUCENT)
.useOverlay(false)
.build(); .build();
public static final Material CHUNK_TRIPWIRE_UNSHADED = SimpleMaterial.builder() public static final Material CHUNK_TRIPWIRE_UNSHADED = SimpleMaterial.builder()
.cutout(CutoutShaders.ONE_TENTH) .cutout(CutoutShaders.ONE_TENTH)
.transparency(Transparency.TRANSLUCENT) .transparency(Transparency.TRANSLUCENT)
.useOverlay(false)
.diffuse(false) .diffuse(false)
.build(); .build();

View file

@ -1,14 +0,0 @@
package com.jozufozu.flywheel.lib.math;
import net.minecraft.client.multiplayer.ClientLevel;
public interface DiffuseLightCalculator {
DiffuseLightCalculator DEFAULT = RenderMath::diffuseLight;
DiffuseLightCalculator NETHER = RenderMath::diffuseLightNether;
static DiffuseLightCalculator forLevel(ClientLevel level) {
return level.effects().constantAmbientLight() ? NETHER : DEFAULT;
}
float getDiffuse(float normalX, float normalY, float normalZ, boolean shaded);
}

View file

@ -1,7 +1,5 @@
package com.jozufozu.flywheel.lib.math; package com.jozufozu.flywheel.lib.math;
import net.minecraftforge.client.model.lighting.QuadLighter;
public final class RenderMath { public final class RenderMath {
private RenderMath() { private RenderMath() {
} }
@ -33,18 +31,4 @@ public final class RenderMath {
public static byte unb(float f) { public static byte unb(float f) {
return (byte) (int) (f * 255); return (byte) (int) (f * 255);
} }
public static float diffuseLight(float x, float y, float z, boolean shaded) {
if (!shaded) {
return 1f;
}
return QuadLighter.calculateShade(x, y, z, false);
}
public static float diffuseLightNether(float x, float y, float z, boolean shaded) {
if (!shaded) {
return 0.9f;
}
return QuadLighter.calculateShade(x, y, z, true);
}
} }

View file

@ -12,10 +12,10 @@ import com.mojang.blaze3d.vertex.PoseStack;
* <br> * <br>
* Only one instance of this class should exist per {@link PoseStack}. * Only one instance of this class should exist per {@link PoseStack}.
*/ */
@ApiStatus.Internal
public class PoseTransformStack implements TransformStack<PoseTransformStack> { public class PoseTransformStack implements TransformStack<PoseTransformStack> {
private final PoseStack stack; private final PoseStack stack;
@ApiStatus.Internal
public PoseTransformStack(PoseStack stack) { public PoseTransformStack(PoseStack stack) {
this.stack = stack; this.stack = stack;
} }

View file

@ -8,16 +8,17 @@ import com.jozufozu.flywheel.api.visualization.BlockEntityVisualizer;
import com.jozufozu.flywheel.api.visualization.VisualizationContext; import com.jozufozu.flywheel.api.visualization.VisualizationContext;
import com.jozufozu.flywheel.api.visualization.VisualizerRegistry; import com.jozufozu.flywheel.api.visualization.VisualizerRegistry;
import net.minecraft.client.renderer.blockentity.BlockEntityRenderer;
import net.minecraft.world.level.block.entity.BlockEntity; import net.minecraft.world.level.block.entity.BlockEntity;
import net.minecraft.world.level.block.entity.BlockEntityType; import net.minecraft.world.level.block.entity.BlockEntityType;
public class SimpleBlockEntityVisualizer<T extends BlockEntity> implements BlockEntityVisualizer<T> { public class SimpleBlockEntityVisualizer<T extends BlockEntity> implements BlockEntityVisualizer<T> {
protected Factory<T> visualFactory; protected Factory<T> visualFactory;
protected Predicate<T> skipRender; protected Predicate<T> skipVanillaRender;
public SimpleBlockEntityVisualizer(Factory<T> visualFactory, Predicate<T> skipRender) { public SimpleBlockEntityVisualizer(Factory<T> visualFactory, Predicate<T> skipVanillaRender) {
this.visualFactory = visualFactory; this.visualFactory = visualFactory;
this.skipRender = skipRender; this.skipVanillaRender = skipVanillaRender;
} }
@Override @Override
@ -26,8 +27,8 @@ public class SimpleBlockEntityVisualizer<T extends BlockEntity> implements Block
} }
@Override @Override
public boolean shouldSkipRender(T blockEntity) { public boolean skipVanillaRender(T blockEntity) {
return skipRender.test(blockEntity); return skipVanillaRender.test(blockEntity);
} }
/** /**
@ -37,8 +38,8 @@ public class SimpleBlockEntityVisualizer<T extends BlockEntity> implements Block
* @param <T> The type of the block entity. * @param <T> The type of the block entity.
* @return The configuration object. * @return The configuration object.
*/ */
public static <T extends BlockEntity> BlockEntityConfig<T> configure(BlockEntityType<T> type) { public static <T extends BlockEntity> Builder<T> builder(BlockEntityType<T> type) {
return new BlockEntityConfig<>(type); return new Builder<>(type);
} }
@FunctionalInterface @FunctionalInterface
@ -51,12 +52,12 @@ public class SimpleBlockEntityVisualizer<T extends BlockEntity> implements Block
* *
* @param <T> The type of the block entity. * @param <T> The type of the block entity.
*/ */
public static class BlockEntityConfig<T extends BlockEntity> { public static class Builder<T extends BlockEntity> {
protected BlockEntityType<T> type; protected BlockEntityType<T> type;
protected Factory<T> visualFactory; protected Factory<T> visualFactory;
protected Predicate<T> skipRender; protected Predicate<T> skipVanillaRender;
public BlockEntityConfig(BlockEntityType<T> type) { public Builder(BlockEntityType<T> type) {
this.type = type; this.type = type;
} }
@ -66,40 +67,44 @@ public class SimpleBlockEntityVisualizer<T extends BlockEntity> implements Block
* @param visualFactory The visual factory. * @param visualFactory The visual factory.
* @return {@code this} * @return {@code this}
*/ */
public BlockEntityConfig<T> factory(Factory<T> visualFactory) { public Builder<T> factory(Factory<T> visualFactory) {
this.visualFactory = visualFactory; this.visualFactory = visualFactory;
return this; return this;
} }
/** /**
* Sets a predicate to determine whether to skip rendering a block entity. * Sets a predicate to determine whether to skip rendering with the vanilla {@link BlockEntityRenderer}.
* @param skipRender The predicate. *
* @param skipVanillaRender The predicate.
* @return {@code this} * @return {@code this}
*/ */
public BlockEntityConfig<T> skipRender(Predicate<T> skipRender) { public Builder<T> skipVanillaRender(Predicate<T> skipVanillaRender) {
this.skipRender = skipRender; this.skipVanillaRender = skipVanillaRender;
return this; return this;
} }
/** /**
* Sets a predicate to always skip rendering for block entities of this type. * Sets a predicate to never skip rendering with the vanilla {@link BlockEntityRenderer}.
*
* @return {@code this} * @return {@code this}
*/ */
public BlockEntityConfig<T> alwaysSkipRender() { public Builder<T> neverSkipVanillaRender() {
this.skipRender = be -> true; this.skipVanillaRender = blockEntity -> false;
return this; return this;
} }
/** /**
* Constructs the block entity visualizer, and sets it for the block entity type. * Constructs the block entity visualizer and sets it for the block entity type.
*
* @return The block entity visualizer. * @return The block entity visualizer.
*/ */
public SimpleBlockEntityVisualizer<T> apply() { public SimpleBlockEntityVisualizer<T> apply() {
Objects.requireNonNull(visualFactory, "Visual factory cannot be null!"); Objects.requireNonNull(visualFactory, "Visual factory cannot be null!");
if (skipRender == null) { if (skipVanillaRender == null) {
skipRender = be -> false; skipVanillaRender = blockEntity -> true;
} }
SimpleBlockEntityVisualizer<T> visualizer = new SimpleBlockEntityVisualizer<>(visualFactory, skipRender);
SimpleBlockEntityVisualizer<T> visualizer = new SimpleBlockEntityVisualizer<>(visualFactory, skipVanillaRender);
VisualizerRegistry.setVisualizer(type, visualizer); VisualizerRegistry.setVisualizer(type, visualizer);
return visualizer; return visualizer;
} }

View file

@ -8,16 +8,17 @@ import com.jozufozu.flywheel.api.visualization.EntityVisualizer;
import com.jozufozu.flywheel.api.visualization.VisualizationContext; import com.jozufozu.flywheel.api.visualization.VisualizationContext;
import com.jozufozu.flywheel.api.visualization.VisualizerRegistry; import com.jozufozu.flywheel.api.visualization.VisualizerRegistry;
import net.minecraft.client.renderer.entity.EntityRenderer;
import net.minecraft.world.entity.Entity; import net.minecraft.world.entity.Entity;
import net.minecraft.world.entity.EntityType; import net.minecraft.world.entity.EntityType;
public class SimpleEntityVisualizer<T extends Entity> implements EntityVisualizer<T> { public class SimpleEntityVisualizer<T extends Entity> implements EntityVisualizer<T> {
protected Factory<T> visualFactory; protected Factory<T> visualFactory;
protected Predicate<T> skipRender; protected Predicate<T> skipVanillaRender;
public SimpleEntityVisualizer(Factory<T> visualFactory, Predicate<T> skipRender) { public SimpleEntityVisualizer(Factory<T> visualFactory, Predicate<T> skipVanillaRender) {
this.visualFactory = visualFactory; this.visualFactory = visualFactory;
this.skipRender = skipRender; this.skipVanillaRender = skipVanillaRender;
} }
@Override @Override
@ -26,8 +27,8 @@ public class SimpleEntityVisualizer<T extends Entity> implements EntityVisualize
} }
@Override @Override
public boolean shouldSkipRender(T entity) { public boolean skipVanillaRender(T entity) {
return skipRender.test(entity); return skipVanillaRender.test(entity);
} }
/** /**
@ -37,8 +38,8 @@ public class SimpleEntityVisualizer<T extends Entity> implements EntityVisualize
* @param <T> The type of the entity. * @param <T> The type of the entity.
* @return The configuration object. * @return The configuration object.
*/ */
public static <T extends Entity> EntityConfig<T> configure(EntityType<T> type) { public static <T extends Entity> Builder<T> builder(EntityType<T> type) {
return new EntityConfig<>(type); return new Builder<>(type);
} }
@FunctionalInterface @FunctionalInterface
@ -51,12 +52,12 @@ public class SimpleEntityVisualizer<T extends Entity> implements EntityVisualize
* *
* @param <T> The type of the entity. * @param <T> The type of the entity.
*/ */
public static class EntityConfig<T extends Entity> { public static class Builder<T extends Entity> {
protected EntityType<T> type; protected EntityType<T> type;
protected Factory<T> visualFactory; protected Factory<T> visualFactory;
protected Predicate<T> skipRender; protected Predicate<T> skipVanillaRender;
public EntityConfig(EntityType<T> type) { public Builder(EntityType<T> type) {
this.type = type; this.type = type;
} }
@ -66,40 +67,44 @@ public class SimpleEntityVisualizer<T extends Entity> implements EntityVisualize
* @param visualFactory The visual factory. * @param visualFactory The visual factory.
* @return {@code this} * @return {@code this}
*/ */
public EntityConfig<T> factory(Factory<T> visualFactory) { public Builder<T> factory(Factory<T> visualFactory) {
this.visualFactory = visualFactory; this.visualFactory = visualFactory;
return this; return this;
} }
/** /**
* Sets a predicate to determine whether to skip rendering an entity. * Sets a predicate to determine whether to skip rendering with the vanilla {@link EntityRenderer}.
* @param skipRender The predicate. *
* @param skipVanillaRender The predicate.
* @return {@code this} * @return {@code this}
*/ */
public EntityConfig<T> skipRender(Predicate<T> skipRender) { public Builder<T> skipVanillaRender(Predicate<T> skipVanillaRender) {
this.skipRender = skipRender; this.skipVanillaRender = skipVanillaRender;
return this; return this;
} }
/** /**
* Sets a predicate to always skip rendering for entities of this type. * Sets a predicate to always skip rendering with the vanilla {@link EntityRenderer}.
*
* @return {@code this} * @return {@code this}
*/ */
public EntityConfig<T> alwaysSkipRender() { public Builder<T> neverSkipVanillaRender() {
this.skipRender = entity -> true; this.skipVanillaRender = entity -> false;
return this; return this;
} }
/** /**
* Constructs the entity visualizer, and sets it for the entity type. * Constructs the entity visualizer and sets it for the entity type.
*
* @return The entity visualizer. * @return The entity visualizer.
*/ */
public SimpleEntityVisualizer<T> apply() { public SimpleEntityVisualizer<T> apply() {
Objects.requireNonNull(visualFactory, "Visual factory cannot be null!"); Objects.requireNonNull(visualFactory, "Visual factory cannot be null!");
if (skipRender == null) { if (skipVanillaRender == null) {
skipRender = entity -> false; skipVanillaRender = entity -> true;
} }
SimpleEntityVisualizer<T> visualizer = new SimpleEntityVisualizer<>(visualFactory, skipRender);
SimpleEntityVisualizer<T> visualizer = new SimpleEntityVisualizer<>(visualFactory, skipVanillaRender);
VisualizerRegistry.setVisualizer(type, visualizer); VisualizerRegistry.setVisualizer(type, visualizer);
return visualizer; return visualizer;
} }

View file

@ -17,6 +17,7 @@ import com.jozufozu.flywheel.lib.visual.AbstractEntityVisual;
import com.mojang.blaze3d.vertex.PoseStack; import com.mojang.blaze3d.vertex.PoseStack;
import com.mojang.math.Axis; import com.mojang.math.Axis;
import net.minecraft.client.model.geom.ModelLayerLocation;
import net.minecraft.client.model.geom.ModelLayers; import net.minecraft.client.model.geom.ModelLayers;
import net.minecraft.util.Mth; import net.minecraft.util.Mth;
import net.minecraft.world.entity.vehicle.AbstractMinecart; import net.minecraft.world.entity.vehicle.AbstractMinecart;
@ -25,9 +26,15 @@ 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 TickableVisual, DynamicVisual { public class MinecartVisual<T extends AbstractMinecart> extends AbstractEntityVisual<T> implements TickableVisual, DynamicVisual {
private static final ModelHolder BODY_MODEL = new ModelHolder(() -> { public static final ModelHolder CHEST_BODY_MODEL = createBodyModelHolder(ModelLayers.CHEST_MINECART);
return new SingleMeshModel(ModelPartConverter.convert(ModelLayers.MINECART), Materials.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);
public static final ModelHolder HOPPER_BODY_MODEL = createBodyModelHolder(ModelLayers.HOPPER_MINECART);
public static final ModelHolder STANDARD_BODY_MODEL = createBodyModelHolder(ModelLayers.MINECART);
public static final ModelHolder SPAWNER_BODY_MODEL = createBodyModelHolder(ModelLayers.SPAWNER_MINECART);
public static final ModelHolder TNT_BODY_MODEL = createBodyModelHolder(ModelLayers.TNT_MINECART);
private final ModelHolder bodyModel;
private TransformedInstance body; private TransformedInstance body;
private TransformedInstance contents; private TransformedInstance contents;
@ -36,8 +43,15 @@ public class MinecartVisual<T extends AbstractMinecart> extends AbstractEntityVi
private final PoseStack stack = new PoseStack(); private final PoseStack stack = new PoseStack();
public MinecartVisual(VisualizationContext ctx, T entity) { public MinecartVisual(VisualizationContext ctx, T entity, ModelHolder bodyModel) {
super(ctx, entity); super(ctx, entity);
this.bodyModel = bodyModel;
}
private static ModelHolder createBodyModelHolder(ModelLayerLocation layer) {
return new ModelHolder(() -> {
return new SingleMeshModel(ModelPartConverter.convert(layer), Materials.MINECART);
});
} }
@Override @Override
@ -46,13 +60,13 @@ public class MinecartVisual<T extends AbstractMinecart> extends AbstractEntityVi
blockState = entity.getDisplayBlockState(); blockState = entity.getDisplayBlockState();
contents = createContentsInstance(); contents = createContentsInstance();
updatePosition(partialTick); updateInstances(partialTick);
super.init(partialTick); super.init(partialTick);
} }
private TransformedInstance createBodyInstance() { private TransformedInstance createBodyInstance() {
return instancerProvider.instancer(InstanceTypes.TRANSFORMED, BODY_MODEL.get(), RenderStage.AFTER_ENTITIES) return instancerProvider.instancer(InstanceTypes.TRANSFORMED, bodyModel.get(), RenderStage.AFTER_ENTITIES)
.createInstance(); .createInstance();
} }
@ -75,7 +89,7 @@ public class MinecartVisual<T extends AbstractMinecart> extends AbstractEntityVi
} }
@Override @Override
public void tick(VisualTickContext c) { public void tick(VisualTickContext context) {
BlockState displayBlockState = entity.getDisplayBlockState(); BlockState displayBlockState = entity.getDisplayBlockState();
if (displayBlockState != blockState) { if (displayBlockState != blockState) {
@ -98,10 +112,10 @@ public class MinecartVisual<T extends AbstractMinecart> extends AbstractEntityVi
return; return;
} }
updatePosition(context.partialTick()); updateInstances(context.partialTick());
} }
private void updatePosition(float partialTick) { private void updateInstances(float partialTick) {
stack.setIdentity(); stack.setIdentity();
double posX = Mth.lerp(partialTick, entity.xOld, entity.getX()); double posX = Mth.lerp(partialTick, entity.xOld, entity.getX());
@ -160,10 +174,9 @@ public class MinecartVisual<T extends AbstractMinecart> extends AbstractEntityVi
if (contents != null) { if (contents != null) {
stack.pushPose(); stack.pushPose();
stack.scale(0.75F, 0.75F, 0.75F); stack.scale(0.75F, 0.75F, 0.75F);
stack.translate(-0.5D, (float) (displayOffset - 8) / 16, 0.5D); stack.translate(-0.5F, (float) (displayOffset - 8) / 16, 0.5F);
stack.mulPose(Axis.YP.rotationDegrees(90)); stack.mulPose(Axis.YP.rotationDegrees(90));
contents.setTransform(stack) updateContents(contents, stack, partialTick);
.setChanged();
stack.popPose(); stack.popPose();
} }
@ -172,6 +185,11 @@ public class MinecartVisual<T extends AbstractMinecart> extends AbstractEntityVi
.setChanged(); .setChanged();
} }
protected void updateContents(TransformedInstance contents, PoseStack stack, float partialTick) {
contents.setTransform(stack)
.setChanged();
}
public void updateLight() { public void updateLight() {
relight(entity.blockPosition(), body, contents); relight(entity.blockPosition(), body, contents);
} }

View file

@ -0,0 +1,41 @@
package com.jozufozu.flywheel.vanilla;
import com.jozufozu.flywheel.api.visualization.VisualizationContext;
import com.jozufozu.flywheel.lib.instance.TransformedInstance;
import com.mojang.blaze3d.vertex.PoseStack;
import net.minecraft.client.renderer.texture.OverlayTexture;
import net.minecraft.util.Mth;
import net.minecraft.world.entity.vehicle.MinecartTNT;
public class TntMinecartVisual<T extends MinecartTNT> extends MinecartVisual<T> {
private static final int WHITE_OVERLAY = OverlayTexture.pack(OverlayTexture.u(1.0F), 10);
public TntMinecartVisual(VisualizationContext ctx, T entity) {
super(ctx, entity, TNT_BODY_MODEL);
}
@Override
protected void updateContents(TransformedInstance contents, PoseStack stack, float partialTick) {
int fuseTime = entity.getFuse();
if (fuseTime > -1 && (float) fuseTime - partialTick + 1.0F < 10.0F) {
float f = 1.0F - ((float) fuseTime - partialTick + 1.0F) / 10.0F;
f = Mth.clamp(f, 0.0F, 1.0F);
f *= f;
f *= f;
float scale = 1.0F + f * 0.3F;
stack.scale(scale, scale, scale);
}
int overlay;
if (fuseTime > -1 && fuseTime / 5 % 2 == 0) {
overlay = WHITE_OVERLAY;
} else {
overlay = OverlayTexture.NO_OVERLAY;
}
contents.setTransform(stack)
.setOverlay(overlay)
.setChanged();
}
}

View file

@ -1,7 +1,7 @@
package com.jozufozu.flywheel.vanilla; package com.jozufozu.flywheel.vanilla;
import static com.jozufozu.flywheel.lib.visual.SimpleBlockEntityVisualizer.configure; import static com.jozufozu.flywheel.lib.visual.SimpleBlockEntityVisualizer.builder;
import static com.jozufozu.flywheel.lib.visual.SimpleEntityVisualizer.configure; import static com.jozufozu.flywheel.lib.visual.SimpleEntityVisualizer.builder;
import net.minecraft.world.entity.EntityType; import net.minecraft.world.entity.EntityType;
import net.minecraft.world.level.block.entity.BlockEntityType; import net.minecraft.world.level.block.entity.BlockEntityType;
@ -28,48 +28,51 @@ import net.minecraft.world.level.block.entity.BlockEntityType;
*/ */
public class VanillaVisuals { public class VanillaVisuals {
public static void init() { public static void init() {
configure(BlockEntityType.CHEST) builder(BlockEntityType.CHEST)
.alwaysSkipRender()
.factory(ChestVisual::new) .factory(ChestVisual::new)
.apply(); .apply();
configure(BlockEntityType.ENDER_CHEST) builder(BlockEntityType.ENDER_CHEST)
.alwaysSkipRender()
.factory(ChestVisual::new) .factory(ChestVisual::new)
.apply(); .apply();
configure(BlockEntityType.TRAPPED_CHEST) builder(BlockEntityType.TRAPPED_CHEST)
.alwaysSkipRender()
.factory(ChestVisual::new) .factory(ChestVisual::new)
.apply(); .apply();
configure(BlockEntityType.BELL) builder(BlockEntityType.BELL)
.alwaysSkipRender()
.factory(BellVisual::new) .factory(BellVisual::new)
.apply(); .apply();
configure(BlockEntityType.SHULKER_BOX) builder(BlockEntityType.SHULKER_BOX)
.alwaysSkipRender()
.factory(ShulkerBoxVisual::new) .factory(ShulkerBoxVisual::new)
.apply(); .apply();
configure(EntityType.MINECART) builder(EntityType.CHEST_MINECART)
.skipRender(MinecartVisual::shouldSkipRender) .factory((ctx, entity) -> new MinecartVisual<>(ctx, entity, MinecartVisual.CHEST_BODY_MODEL))
.factory(MinecartVisual::new) .skipVanillaRender(MinecartVisual::shouldSkipRender)
.apply(); .apply();
configure(EntityType.COMMAND_BLOCK_MINECART) builder(EntityType.COMMAND_BLOCK_MINECART)
.skipRender(MinecartVisual::shouldSkipRender) .factory((ctx, entity) -> new MinecartVisual<>(ctx, entity, MinecartVisual.COMMAND_BLOCK_BODY_MODEL))
.factory(MinecartVisual::new) .skipVanillaRender(MinecartVisual::shouldSkipRender)
.apply(); .apply();
configure(EntityType.FURNACE_MINECART) builder(EntityType.FURNACE_MINECART)
.skipRender(MinecartVisual::shouldSkipRender) .factory((ctx, entity) -> new MinecartVisual<>(ctx, entity, MinecartVisual.FURNACE_BODY_MODEL))
.factory(MinecartVisual::new) .skipVanillaRender(MinecartVisual::shouldSkipRender)
.apply(); .apply();
configure(EntityType.HOPPER_MINECART) builder(EntityType.HOPPER_MINECART)
.skipRender(MinecartVisual::shouldSkipRender) .factory((ctx, entity) -> new MinecartVisual<>(ctx, entity, MinecartVisual.HOPPER_BODY_MODEL))
.factory(MinecartVisual::new) .skipVanillaRender(MinecartVisual::shouldSkipRender)
.apply(); .apply();
configure(EntityType.TNT_MINECART) builder(EntityType.MINECART)
.skipRender(MinecartVisual::shouldSkipRender) .factory((ctx, entity) -> new MinecartVisual<>(ctx, entity, MinecartVisual.STANDARD_BODY_MODEL))
.factory(MinecartVisual::new) .skipVanillaRender(MinecartVisual::shouldSkipRender)
.apply();
builder(EntityType.SPAWNER_MINECART)
.factory((ctx, entity) -> new MinecartVisual<>(ctx, entity, MinecartVisual.SPAWNER_BODY_MODEL))
.skipVanillaRender(MinecartVisual::shouldSkipRender)
.apply();
builder(EntityType.TNT_MINECART)
.factory(TntMinecartVisual::new)
.skipVanillaRender(MinecartVisual::shouldSkipRender)
.apply(); .apply();
} }
} }

View file

@ -4,5 +4,6 @@ void flw_instanceVertex(in FlwInstance i) {
flw_vertexPos = vec4(rotateByQuaternion(flw_vertexPos.xyz - i.pivot, i.rotation) + i.pivot + i.position, 1.0); flw_vertexPos = vec4(rotateByQuaternion(flw_vertexPos.xyz - i.pivot, i.rotation) + i.pivot + i.position, 1.0);
flw_vertexNormal = rotateByQuaternion(flw_vertexNormal, i.rotation); flw_vertexNormal = rotateByQuaternion(flw_vertexNormal, i.rotation);
flw_vertexColor = i.color; flw_vertexColor = i.color;
flw_vertexOverlay = i.overlay;
flw_vertexLight = i.light / 15.0; flw_vertexLight = i.light / 15.0;
} }

View file

@ -2,5 +2,6 @@ void flw_instanceVertex(in FlwInstance i) {
flw_vertexPos = i.pose * flw_vertexPos; flw_vertexPos = i.pose * flw_vertexPos;
flw_vertexNormal = i.normal * flw_vertexNormal; flw_vertexNormal = i.normal * flw_vertexNormal;
flw_vertexColor = i.color; flw_vertexColor = i.color;
flw_vertexOverlay = i.overlay;
flw_vertexLight = i.light / 15.0; flw_vertexLight = i.light / 15.0;
} }