mirror of
https://github.com/Sollace/Unicopia.git
synced 2024-11-24 22:07:59 +01:00
169 lines
5.5 KiB
Java
169 lines
5.5 KiB
Java
package com.minelittlepony.unicopia.ability;
|
|
|
|
import java.util.List;
|
|
import java.util.stream.Collectors;
|
|
|
|
import org.jetbrains.annotations.Nullable;
|
|
|
|
import com.minelittlepony.unicopia.Race;
|
|
import com.minelittlepony.unicopia.ability.data.Hit;
|
|
import com.minelittlepony.unicopia.entity.player.Pony;
|
|
import com.minelittlepony.unicopia.particle.FollowingParticleEffect;
|
|
import com.minelittlepony.unicopia.particle.ParticleUtils;
|
|
import com.minelittlepony.unicopia.particle.UParticles;
|
|
import com.minelittlepony.unicopia.util.MagicalDamageSource;
|
|
import com.minelittlepony.unicopia.util.TraceHelper;
|
|
import com.minelittlepony.unicopia.util.VecHelper;
|
|
|
|
import net.minecraft.entity.Entity;
|
|
import net.minecraft.entity.LivingEntity;
|
|
import net.minecraft.entity.damage.DamageSource;
|
|
import net.minecraft.entity.effect.StatusEffectInstance;
|
|
import net.minecraft.entity.effect.StatusEffects;
|
|
import net.minecraft.entity.mob.HostileEntity;
|
|
import net.minecraft.entity.passive.CowEntity;
|
|
import net.minecraft.entity.passive.MerchantEntity;
|
|
import net.minecraft.entity.passive.PigEntity;
|
|
import net.minecraft.entity.passive.SheepEntity;
|
|
import net.minecraft.entity.player.PlayerEntity;
|
|
import net.minecraft.particle.ParticleTypes;
|
|
|
|
/**
|
|
* Changeling ability to restore health from mobs
|
|
*/
|
|
public class ChangelingFeedAbility implements Ability<Hit> {
|
|
|
|
@Override
|
|
public int getWarmupTime(Pony player) {
|
|
return 5;
|
|
}
|
|
|
|
@Override
|
|
public int getCooldownTime(Pony player) {
|
|
return canFeed(player) ? 15 : 80;
|
|
}
|
|
|
|
@Override
|
|
public boolean canUse(Race race) {
|
|
return race == Race.CHANGELING;
|
|
}
|
|
|
|
@Nullable
|
|
@Override
|
|
public Hit tryActivate(Pony player) {
|
|
if (canFeed(player)) {
|
|
if (!getTargets(player).isEmpty()) {
|
|
return Hit.INSTANCE;
|
|
}
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
private boolean canFeed(Pony player) {
|
|
return player.asEntity().getHealth() < player.asEntity().getMaxHealth() || player.asEntity().canConsume(false);
|
|
}
|
|
|
|
private boolean canDrain(Entity e) {
|
|
return (e instanceof LivingEntity)
|
|
&& (e instanceof CowEntity
|
|
|| e instanceof MerchantEntity
|
|
|| e instanceof PlayerEntity
|
|
|| e instanceof SheepEntity
|
|
|| e instanceof PigEntity
|
|
|| e instanceof HostileEntity);
|
|
}
|
|
|
|
@Override
|
|
public Hit.Serializer<Hit> getSerializer() {
|
|
return Hit.SERIALIZER;
|
|
}
|
|
|
|
protected List<LivingEntity> getTargets(Pony player) {
|
|
List<Entity> list = VecHelper.findInRange(player.asEntity(), player.getReferenceWorld(), player.getOriginVector(), 3, this::canDrain);
|
|
|
|
TraceHelper.<LivingEntity>findEntity(player.asEntity(), 17, 1,
|
|
looked -> looked instanceof LivingEntity && !list.contains(looked) && canDrain(looked))
|
|
.ifPresent(list::add);
|
|
|
|
return list.stream().map(i -> (LivingEntity)i).collect(Collectors.toList());
|
|
}
|
|
|
|
@Override
|
|
public double getCostEstimate(Pony player) {
|
|
return 0;
|
|
}
|
|
|
|
@Override
|
|
public void apply(Pony iplayer, Hit data) {
|
|
PlayerEntity player = iplayer.asEntity();
|
|
|
|
float maximumHealthGain = player.getMaxHealth() - player.getHealth();
|
|
int maximumFoodGain = player.canConsume(false) ? (20 - player.getHungerManager().getFoodLevel()) : 0;
|
|
|
|
if (maximumHealthGain > 0 || maximumFoodGain > 0) {
|
|
|
|
float healAmount = 0;
|
|
|
|
for (LivingEntity i : getTargets(iplayer)) {
|
|
healAmount += drainFrom(player, i);
|
|
}
|
|
|
|
int foodAmount = (int)Math.floor(Math.min(healAmount / 3, maximumFoodGain));
|
|
|
|
if (foodAmount > 0) {
|
|
healAmount -= foodAmount;
|
|
player.getHungerManager().add(foodAmount, 0.125f);
|
|
}
|
|
|
|
if (healAmount > 0) {
|
|
player.heal(Math.min(healAmount, maximumHealthGain));
|
|
}
|
|
}
|
|
}
|
|
|
|
public float drainFrom(PlayerEntity changeling, LivingEntity living) {
|
|
|
|
DamageSource d = MagicalDamageSource.create("feed", changeling);
|
|
|
|
float damage = living.getHealth()/2;
|
|
|
|
if (damage > 0) {
|
|
living.damage(d, damage);
|
|
}
|
|
|
|
ParticleUtils.spawnParticles(UParticles.CHANGELING_MAGIC, living, 7);
|
|
ParticleUtils.spawnParticles(new FollowingParticleEffect(UParticles.HEALTH_DRAIN, changeling, 0.2F), living, 1);
|
|
|
|
if (changeling.hasStatusEffect(StatusEffects.NAUSEA)) {
|
|
StatusEffectInstance effect = changeling.getStatusEffect(StatusEffects.NAUSEA);
|
|
changeling.removeStatusEffect(StatusEffects.NAUSEA);
|
|
living.addStatusEffect(effect);
|
|
} else if (changeling.getEntityWorld().random.nextInt(2300) == 0) {
|
|
living.addStatusEffect(new StatusEffectInstance(StatusEffects.WITHER, 20, 1));
|
|
}
|
|
|
|
if (living instanceof PlayerEntity) {
|
|
damage ++;
|
|
damage *= 1.6F;
|
|
|
|
if (!changeling.hasStatusEffect(StatusEffects.HEALTH_BOOST)) {
|
|
changeling.addStatusEffect(new StatusEffectInstance(StatusEffects.HEALTH_BOOST, 13000, 1));
|
|
}
|
|
}
|
|
|
|
return damage;
|
|
}
|
|
|
|
@Override
|
|
public void preApply(Pony player, AbilitySlot slot) {
|
|
player.getMagicalReserves().getExertion().add(6);
|
|
}
|
|
|
|
@Override
|
|
public void postApply(Pony player, AbilitySlot slot) {
|
|
if (player.getReferenceWorld().random.nextInt(10) == 0) {
|
|
player.spawnParticles(ParticleTypes.HEART, 1);
|
|
}
|
|
}
|
|
}
|