2021-08-19 19:52:35 +02:00
|
|
|
package com.minelittlepony.unicopia.entity;
|
|
|
|
|
|
|
|
import java.util.Random;
|
|
|
|
|
|
|
|
import org.jetbrains.annotations.Nullable;
|
|
|
|
|
|
|
|
import net.minecraft.block.BlockState;
|
|
|
|
import net.minecraft.entity.Entity;
|
|
|
|
import net.minecraft.entity.EntityPose;
|
|
|
|
import net.minecraft.entity.EntityType;
|
|
|
|
import net.minecraft.entity.SpawnReason;
|
|
|
|
import net.minecraft.entity.attribute.DefaultAttributeContainer;
|
|
|
|
import net.minecraft.entity.attribute.EntityAttributes;
|
|
|
|
import net.minecraft.entity.damage.DamageSource;
|
|
|
|
import net.minecraft.entity.data.DataTracker;
|
|
|
|
import net.minecraft.entity.data.TrackedData;
|
|
|
|
import net.minecraft.entity.data.TrackedDataHandlerRegistry;
|
|
|
|
import net.minecraft.entity.mob.AmbientEntity;
|
|
|
|
import net.minecraft.entity.player.PlayerEntity;
|
|
|
|
import net.minecraft.predicate.entity.EntityPredicates;
|
|
|
|
import net.minecraft.sound.SoundEvent;
|
|
|
|
import net.minecraft.sound.SoundEvents;
|
|
|
|
import net.minecraft.util.Identifier;
|
|
|
|
import net.minecraft.util.math.BlockPos;
|
|
|
|
import net.minecraft.util.math.MathHelper;
|
|
|
|
import net.minecraft.util.math.Vec3d;
|
|
|
|
import net.minecraft.world.World;
|
|
|
|
import net.minecraft.world.WorldAccess;
|
|
|
|
|
|
|
|
public class ButterflyEntity extends AmbientEntity {
|
|
|
|
private static final TrackedData<Boolean> RESTING = DataTracker.registerData(ButterflyEntity.class, TrackedDataHandlerRegistry.BOOLEAN);
|
|
|
|
private static final TrackedData<Integer> VARIANT = DataTracker.registerData(ButterflyEntity.class, TrackedDataHandlerRegistry.INTEGER);
|
|
|
|
|
2021-08-19 21:24:09 +02:00
|
|
|
@Nullable
|
2021-08-19 19:52:35 +02:00
|
|
|
private BlockPos hoveringPosition;
|
|
|
|
|
2021-08-19 21:24:09 +02:00
|
|
|
private int ticksResting;
|
|
|
|
|
2021-08-19 19:52:35 +02:00
|
|
|
public ButterflyEntity(EntityType<ButterflyEntity> type, World world) {
|
|
|
|
super(type, world);
|
|
|
|
setVariant(Variant.random(world.random));
|
|
|
|
setResting(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
public static DefaultAttributeContainer.Builder createButterflyAttributes() {
|
|
|
|
return createMobAttributes().add(EntityAttributes.GENERIC_MAX_HEALTH, 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public float getSoundPitch() {
|
|
|
|
return super.getSoundPitch() * 0.95F;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Nullable
|
|
|
|
@Override
|
|
|
|
protected SoundEvent getHurtSound(DamageSource damageSourceIn) {
|
|
|
|
return SoundEvents.ENTITY_BAT_HURT;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Nullable
|
|
|
|
@Override
|
|
|
|
protected SoundEvent getDeathSound() {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
protected void initDataTracker() {
|
|
|
|
super.initDataTracker();
|
|
|
|
getDataTracker().startTracking(VARIANT, Variant.BUTTERFLY.ordinal());
|
|
|
|
getDataTracker().startTracking(RESTING, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public boolean isPushable() {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public boolean collides() {
|
|
|
|
return true;
|
2021-08-19 21:24:09 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
protected void pushAway(Entity entity) { }
|
|
|
|
|
|
|
|
@Override
|
|
|
|
protected void tickCramming() { }
|
2021-08-19 19:52:35 +02:00
|
|
|
|
|
|
|
@Override
|
|
|
|
public void tick() {
|
|
|
|
super.tick();
|
|
|
|
|
|
|
|
Vec3d vel = getVelocity();
|
|
|
|
setVelocity(vel.x, vel.y * 0.6 + 0.02F, vel.z);
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean isResting() {
|
|
|
|
return getDataTracker().get(RESTING);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void setResting(boolean resting) {
|
|
|
|
getDataTracker().set(RESTING, resting);
|
2021-08-19 19:52:56 +02:00
|
|
|
if (!resting) {
|
|
|
|
hoveringPosition = null;
|
|
|
|
}
|
2021-08-19 19:52:35 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
public Variant getVariant() {
|
2021-08-19 21:24:09 +02:00
|
|
|
return Variant.byId(getDataTracker().get(VARIANT));
|
2021-08-19 19:52:35 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
public void setVariant(Variant variant) {
|
|
|
|
getDataTracker().set(VARIANT, variant.ordinal());
|
|
|
|
}
|
|
|
|
|
|
|
|
protected boolean isAggressor(Entity e) {
|
|
|
|
if (e instanceof ButterflyEntity) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (e instanceof PlayerEntity) {
|
|
|
|
PlayerEntity player = (PlayerEntity)e;
|
|
|
|
|
|
|
|
if (player.isCreative() || player.isSpectator()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-08-19 21:24:09 +02:00
|
|
|
if (player.isSprinting() || player.forwardSpeed > 0 || player.sidewaysSpeed > 0) {
|
2021-08-19 19:52:35 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
} else if (!EntityPredicates.EXCEPT_CREATIVE_OR_SPECTATOR.test(e)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-08-19 21:24:09 +02:00
|
|
|
return e.getVelocity().horizontalLength() > 1.4F;
|
2021-08-19 19:52:35 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void tickMovement() {
|
|
|
|
super.tickMovement();
|
|
|
|
|
2021-08-19 19:52:56 +02:00
|
|
|
BlockPos below = new BlockPos(getPos().add(0, -0.5, 0));
|
2021-08-19 19:52:35 +02:00
|
|
|
|
|
|
|
if (isResting()) {
|
2021-08-19 21:24:09 +02:00
|
|
|
if (world.getBlockState(below).isAir()
|
|
|
|
|| !world.getOtherEntities(this, getBoundingBox().expand(7), this::isAggressor).isEmpty()
|
|
|
|
|| (ticksResting++ > 40 && world.random.nextInt(500) == 0)
|
|
|
|
|| world.hasRain(below)) {
|
2021-08-19 19:52:56 +02:00
|
|
|
setResting(false);
|
2021-08-19 19:52:35 +02:00
|
|
|
}
|
|
|
|
} else {
|
2021-08-19 21:24:09 +02:00
|
|
|
ticksResting = 0;
|
2021-08-19 19:52:35 +02:00
|
|
|
|
|
|
|
// invalidate the hovering position
|
|
|
|
if (hoveringPosition != null && (!world.isAir(hoveringPosition) || hoveringPosition.getY() < 1)) {
|
|
|
|
hoveringPosition = null;
|
|
|
|
}
|
|
|
|
|
2021-08-19 21:24:09 +02:00
|
|
|
BlockPos pos = getBlockPos();
|
|
|
|
|
2021-08-19 19:52:35 +02:00
|
|
|
// select a new hovering position
|
|
|
|
if (hoveringPosition == null || random.nextInt(30) == 0 || hoveringPosition.getSquaredDistance(pos) < 4) {
|
2021-08-19 21:24:09 +02:00
|
|
|
hoveringPosition = pos.add(
|
|
|
|
random.nextInt(7) - random.nextInt(7),
|
|
|
|
random.nextInt(6) - 2,
|
|
|
|
random.nextInt(7) - random.nextInt(7)
|
2021-08-19 19:52:35 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
// hover casually towards the chosen position
|
|
|
|
Vec3d motion = Vec3d.ofCenter(hoveringPosition, 0.1).subtract(getPos());
|
|
|
|
Vec3d vel = getVelocity();
|
|
|
|
|
|
|
|
addVelocity(
|
|
|
|
(Math.signum(motion.getX()) * 0.5 - vel.x) * 0.1,
|
|
|
|
(Math.signum(motion.getY()) * 0.7 - vel.y) * 0.1,
|
|
|
|
(Math.signum(motion.getZ()) * 0.5 - vel.z) * 0.1
|
|
|
|
);
|
|
|
|
|
|
|
|
float direction = (float)(MathHelper.atan2(vel.z, vel.x) * (180 / Math.PI)) - 90;
|
|
|
|
|
|
|
|
forwardSpeed = 0.5F;
|
|
|
|
headYaw += MathHelper.wrapDegrees(direction - headYaw);
|
|
|
|
|
|
|
|
if (random.nextInt(100) == 0 && world.getBlockState(below).isOpaque()) {
|
|
|
|
setResting(true);
|
|
|
|
}
|
2021-08-19 21:24:09 +02:00
|
|
|
|
|
|
|
if (!world.isClient && age % 20 == 0 && world.random.nextInt(200) == 0) {
|
|
|
|
for (Entity i : world.getOtherEntities(this, getBoundingBox().expand(20))) {
|
|
|
|
if (i.getType() == getType()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ButterflyEntity copy = (ButterflyEntity)getType().create(world);
|
|
|
|
copy.copyPositionAndRotation(this);
|
|
|
|
world.spawnEntity(copy);
|
|
|
|
}
|
2021-08-19 19:52:35 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-08-19 21:24:09 +02:00
|
|
|
@Override
|
|
|
|
public boolean shouldRender(double distance) {
|
|
|
|
double d = 64 * getRenderDistanceMultiplier();
|
|
|
|
return distance < d * d;
|
|
|
|
}
|
|
|
|
|
2021-08-19 19:52:35 +02:00
|
|
|
@Override
|
|
|
|
public boolean handleFallDamage(float distance, float damageMultiplier, DamageSource cause) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
protected void fall(double y, boolean onGroundIn, BlockState state, BlockPos pos) {
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public boolean canSpawn(WorldAccess world, SpawnReason reason) {
|
2021-08-19 21:24:09 +02:00
|
|
|
return reason != SpawnReason.NATURAL || (getY() >= world.getSeaLevel() && world.getLightLevel(getBlockPos()) > 3);
|
2021-08-19 19:52:35 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public float getEyeHeight(EntityPose pose) {
|
|
|
|
return getHeight() / 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
public enum Variant {
|
|
|
|
BUTTERFLY,
|
|
|
|
YELLOW,
|
|
|
|
LIME,
|
|
|
|
RED,
|
|
|
|
GREEN,
|
|
|
|
BLUE,
|
|
|
|
PURPLE,
|
|
|
|
MAGENTA,
|
|
|
|
PINK,
|
|
|
|
HEDYLIDAE,
|
|
|
|
LYCAENIDAE,
|
|
|
|
NYMPHALIDAE,
|
|
|
|
MONARCH,
|
|
|
|
WHITE_MONARCH,
|
|
|
|
BRIMSTONE;
|
|
|
|
|
2021-08-19 21:24:09 +02:00
|
|
|
private static final Variant[] VALUES = Variant.values();
|
|
|
|
|
2021-08-19 19:52:35 +02:00
|
|
|
private final Identifier skin = new Identifier("unicopia", "textures/entity/butterfly/" + name().toLowerCase() + ".png");
|
|
|
|
|
|
|
|
public Identifier getSkin() {
|
|
|
|
return skin;
|
|
|
|
}
|
|
|
|
|
2021-08-19 21:24:09 +02:00
|
|
|
static Variant byId(int index) {
|
|
|
|
return VALUES[Math.max(0, index) % VALUES.length];
|
|
|
|
}
|
|
|
|
|
2021-08-19 19:52:35 +02:00
|
|
|
static Variant random(Random rand) {
|
2021-08-20 22:22:12 +02:00
|
|
|
return VALUES[rand.nextInt(VALUES.length)];
|
2021-08-19 19:52:35 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|