mirror of
https://github.com/Jozufozu/Flywheel.git
synced 2025-01-06 04:16:36 +01:00
Don't tell me what to do!
- Respect minecraft's entity shadows option. - Early exit in InstanceRecycler#discardExtra when there's no extra. - Shadow radius <= 0 disables shadows. - More documentation in ShadowComponent.
This commit is contained in:
parent
a04c3f52b2
commit
0177849395
2 changed files with 56 additions and 29 deletions
|
@ -59,12 +59,15 @@ public class InstanceRecycler<I extends Instance> {
|
||||||
* Call this after your last call to {@link #get()} each frame.
|
* Call this after your last call to {@link #get()} each frame.
|
||||||
*/
|
*/
|
||||||
public void discardExtra() {
|
public void discardExtra() {
|
||||||
for (int i = count; i < instances.size(); i++) {
|
var size = instances.size();
|
||||||
instances.get(i)
|
if (count == size) {
|
||||||
.delete();
|
// No extra instances, early exit to avoid creating the sublist.
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
instances.subList(count, instances.size())
|
|
||||||
.clear();
|
var extra = instances.subList(count, size);
|
||||||
|
extra.forEach(Instance::delete);
|
||||||
|
extra.clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
public void delete() {
|
public void delete() {
|
||||||
|
|
|
@ -21,6 +21,7 @@ import com.jozufozu.flywheel.lib.instance.ShadowInstance;
|
||||||
import com.jozufozu.flywheel.lib.material.SimpleMaterial;
|
import com.jozufozu.flywheel.lib.material.SimpleMaterial;
|
||||||
import com.jozufozu.flywheel.lib.model.QuadMesh;
|
import com.jozufozu.flywheel.lib.model.QuadMesh;
|
||||||
|
|
||||||
|
import net.minecraft.client.Minecraft;
|
||||||
import net.minecraft.client.renderer.LightTexture;
|
import net.minecraft.client.renderer.LightTexture;
|
||||||
import net.minecraft.core.BlockPos;
|
import net.minecraft.core.BlockPos;
|
||||||
import net.minecraft.resources.ResourceLocation;
|
import net.minecraft.resources.ResourceLocation;
|
||||||
|
@ -33,6 +34,15 @@ import net.minecraft.world.level.chunk.ChunkAccess;
|
||||||
import net.minecraft.world.phys.AABB;
|
import net.minecraft.world.phys.AABB;
|
||||||
import net.minecraft.world.phys.shapes.VoxelShape;
|
import net.minecraft.world.phys.shapes.VoxelShape;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A component that uses instances to render an entity's shadow.
|
||||||
|
*
|
||||||
|
* <p>Use {@link #radius(float)} to set the radius of the shadow, in blocks.
|
||||||
|
* <br>
|
||||||
|
* Use {@link #strength(float)} to set the strength of the shadow.
|
||||||
|
* <br>
|
||||||
|
* The shadow will be cast on blocks at most {@code min(radius, 2 * strength)} blocks below the entity.</p>
|
||||||
|
*/
|
||||||
public class ShadowComponent {
|
public class ShadowComponent {
|
||||||
|
|
||||||
private final VisualizationContext context;
|
private final VisualizationContext context;
|
||||||
|
@ -41,9 +51,9 @@ public class ShadowComponent {
|
||||||
private final InstanceRecycler<ShadowInstance> instances = new InstanceRecycler<>(this::instance);
|
private final InstanceRecycler<ShadowInstance> instances = new InstanceRecycler<>(this::instance);
|
||||||
private final BlockPos.MutableBlockPos pos = new BlockPos.MutableBlockPos();
|
private final BlockPos.MutableBlockPos pos = new BlockPos.MutableBlockPos();
|
||||||
|
|
||||||
private float radius = 0.5F;
|
// Defaults taken from EntityRenderer.
|
||||||
|
private float radius = 0;
|
||||||
private float strength = 1.0F;
|
private float strength = 1.0F;
|
||||||
private boolean enabled = true;
|
|
||||||
|
|
||||||
public ShadowComponent(VisualizationContext context, Entity entity) {
|
public ShadowComponent(VisualizationContext context, Entity entity) {
|
||||||
this.context = context;
|
this.context = context;
|
||||||
|
@ -51,33 +61,49 @@ public class ShadowComponent {
|
||||||
this.entity = entity;
|
this.entity = entity;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set the radius of the shadow, in blocks, clamped to a maximum of 32.
|
||||||
|
*
|
||||||
|
* <p>Setting this to {@code <= 0} will disable the shadow.</p>
|
||||||
|
*
|
||||||
|
* @param radius The radius of the shadow, in blocks.
|
||||||
|
*/
|
||||||
public void radius(float radius) {
|
public void radius(float radius) {
|
||||||
this.radius = radius;
|
this.radius = Math.min(radius, 32);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void strength(float weight) {
|
/**
|
||||||
this.strength = weight;
|
* Set the strength of the shadow.
|
||||||
}
|
*
|
||||||
|
* @param strength The strength of the shadow.
|
||||||
public void enabled(boolean enabled) {
|
*/
|
||||||
this.enabled = enabled;
|
public void strength(float strength) {
|
||||||
|
this.strength = strength;
|
||||||
if (!enabled) {
|
|
||||||
instances.delete();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Update the shadow instances. You'd typically call this in your visual's
|
||||||
|
* {@link com.jozufozu.flywheel.api.visual.DynamicVisual#beginFrame(VisualFrameContext) beginFrame} method.
|
||||||
|
*
|
||||||
|
* @param context The frame context.
|
||||||
|
*/
|
||||||
public void beginFrame(VisualFrameContext context) {
|
public void beginFrame(VisualFrameContext context) {
|
||||||
if (!enabled) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
instances.resetCount();
|
instances.resetCount();
|
||||||
|
|
||||||
|
boolean shadowsEnabled = Minecraft.getInstance().options.entityShadows()
|
||||||
|
.get();
|
||||||
|
if (shadowsEnabled && radius > 0 && !entity.isInvisible()) {
|
||||||
|
setupInstances(context);
|
||||||
|
}
|
||||||
|
|
||||||
|
instances.discardExtra();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void setupInstances(VisualFrameContext context) {
|
||||||
double entityX = Mth.lerp(context.partialTick(), entity.xOld, entity.getX());
|
double entityX = Mth.lerp(context.partialTick(), entity.xOld, entity.getX());
|
||||||
double entityY = Mth.lerp(context.partialTick(), entity.yOld, entity.getY());
|
double entityY = Mth.lerp(context.partialTick(), entity.yOld, entity.getY());
|
||||||
double entityZ = Mth.lerp(context.partialTick(), entity.zOld, entity.getZ());
|
double entityZ = Mth.lerp(context.partialTick(), entity.zOld, entity.getZ());
|
||||||
float castDistance = Math.min(strength / 0.5F, radius);
|
float castDistance = Math.min(strength * 2, radius);
|
||||||
int minXPos = Mth.floor(entityX - (double) radius);
|
int minXPos = Mth.floor(entityX - (double) radius);
|
||||||
int maxXPos = Mth.floor(entityX + (double) radius);
|
int maxXPos = Mth.floor(entityX + (double) radius);
|
||||||
int minYPos = Mth.floor(entityY - (double) castDistance);
|
int minYPos = Mth.floor(entityY - (double) castDistance);
|
||||||
|
@ -92,16 +118,14 @@ public class ShadowComponent {
|
||||||
|
|
||||||
for (int y = minYPos; y <= maxYPos; ++y) {
|
for (int y = minYPos; y <= maxYPos; ++y) {
|
||||||
pos.setY(y);
|
pos.setY(y);
|
||||||
float actualWeight = strength - (float) (entityY - pos.getY()) * 0.5F;
|
float strengthGivenYFalloff = strength - (float) (entityY - pos.getY()) * 0.5F;
|
||||||
maybeSetupShadowInstance(chunkaccess, (float) entityX, (float) entityZ, actualWeight);
|
maybeSetupShadowInstance(chunkaccess, (float) entityX, (float) entityZ, strengthGivenYFalloff);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
instances.discardExtra();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private void maybeSetupShadowInstance(ChunkAccess pChunk, float entityX, float entityZ, float weight) {
|
private void maybeSetupShadowInstance(ChunkAccess pChunk, float entityX, float entityZ, float strength) {
|
||||||
// TODO: cache this?
|
// TODO: cache this?
|
||||||
var maxLocalRawBrightness = level.getMaxLocalRawBrightness(pos);
|
var maxLocalRawBrightness = level.getMaxLocalRawBrightness(pos);
|
||||||
if (maxLocalRawBrightness <= 3) {
|
if (maxLocalRawBrightness <= 3) {
|
||||||
|
@ -109,7 +133,7 @@ public class ShadowComponent {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
float blockBrightness = LightTexture.getBrightness(level.dimensionType(), maxLocalRawBrightness);
|
float blockBrightness = LightTexture.getBrightness(level.dimensionType(), maxLocalRawBrightness);
|
||||||
float alpha = weight * 0.5F * blockBrightness;
|
float alpha = strength * 0.5F * blockBrightness;
|
||||||
if (!(alpha >= 0.0F)) {
|
if (!(alpha >= 0.0F)) {
|
||||||
// Too far away/too weak to render.
|
// Too far away/too weak to render.
|
||||||
return;
|
return;
|
||||||
|
|
Loading…
Reference in a new issue