Unicopia/src/main/java/com/minelittlepony/unicopia/ability/EarthPonyKickAbility.java

278 lines
10 KiB
Java
Raw Normal View History

package com.minelittlepony.unicopia.ability;
import java.util.List;
import java.util.Optional;
2021-08-04 15:38:03 +02:00
import org.jetbrains.annotations.Nullable;
import com.google.common.collect.Lists;
import com.minelittlepony.unicopia.Race;
import com.minelittlepony.unicopia.ability.data.Hit;
import com.minelittlepony.unicopia.ability.data.Pos;
2021-08-20 22:25:46 +02:00
import com.minelittlepony.unicopia.ability.data.tree.TreeType;
2022-09-16 17:54:46 +02:00
import com.minelittlepony.unicopia.block.data.BlockDestructionManager;
2022-09-20 23:50:15 +02:00
import com.minelittlepony.unicopia.client.minelittlepony.MineLPDelegate;
import com.minelittlepony.unicopia.client.render.PlayerPoser.Animation;
import com.minelittlepony.unicopia.entity.player.Pony;
import com.minelittlepony.unicopia.particle.ParticleUtils;
import com.minelittlepony.unicopia.particle.UParticles;
import com.minelittlepony.unicopia.util.*;
import net.minecraft.block.BeehiveBlock;
2021-02-13 12:50:24 +01:00
import net.minecraft.block.Block;
import net.minecraft.block.BlockState;
import net.minecraft.block.Blocks;
import net.minecraft.block.entity.BeehiveBlockEntity;
import net.minecraft.entity.ItemEntity;
import net.minecraft.entity.LivingEntity;
import net.minecraft.entity.passive.BeeEntity;
import net.minecraft.entity.player.PlayerEntity;
import net.minecraft.item.ItemStack;
import net.minecraft.predicate.entity.EntityPredicates;
import net.minecraft.server.world.ServerWorld;
import net.minecraft.util.math.BlockPos;
import net.minecraft.util.math.Box;
import net.minecraft.util.math.Vec3d;
import net.minecraft.world.World;
import net.minecraft.world.WorldEvents;
/**
* Earth Pony kicking ability
*/
public class EarthPonyKickAbility implements Ability<Pos> {
@Override
public int getWarmupTime(Pony player) {
return 3;
}
@Override
public int getCooldownTime(Pony player) {
return 50;
}
@Override
public boolean canUse(Race race) {
return race.canUseEarth();
}
@Override
public double getCostEstimate(Pony player) {
2022-09-20 23:50:15 +02:00
double distance = MineLPDelegate.getInstance().getPlayerPonyRace(player.getMaster()).isDefault() ? 6 : -6;
return RayTraceHelper.doTrace(player.getMaster(), distance, 1)
.getBlockPos()
2022-06-23 16:24:45 +02:00
.filter(pos -> TreeType.at(pos, player.getReferenceWorld()) != TreeType.NONE)
.isPresent() ? 3 : 1;
}
@Override
public Optional<Pos> prepareQuickAction(Pony player, ActivationType type) {
return Optional.of(getDefaultKickLocation(player));
}
@Override
public boolean onQuickAction(Pony player, ActivationType type, Optional<Pos> data) {
if (type == ActivationType.TAP) {
if (!player.isClient()) {
data.ifPresent(kickLocation -> {
Vec3d origin = player.getOriginVector();
World w = player.getReferenceWorld();
for (var e : VecHelper.findInRange(player.getEntity(), w, kickLocation.vec(), 2, EntityPredicates.EXCEPT_CREATIVE_OR_SPECTATOR)) {
if (e instanceof LivingEntity entity) {
float calculatedStrength = 0.5F * (1 + player.getLevel().getScaled(9));
entity.damage(MagicalDamageSource.KICK, player.getReferenceWorld().random.nextBetween(2, 10) + calculatedStrength);
entity.takeKnockback(calculatedStrength, origin.x - entity.getX(), origin.z - entity.getZ());
player.subtractEnergyCost(3);
player.setAnimation(Animation.KICK);
return;
}
}
BlockPos pos = kickLocation.pos();
EarthPonyStompAbility.stompBlock(w, pos, 10 * (1 + player.getLevel().getScaled(5)) * w.getBlockState(pos).calcBlockBreakingDelta(player.getMaster(), w, pos));
player.setAnimation(Animation.KICK);
});
}
return true;
}
return false;
}
@Nullable
@Override
public Pos tryActivate(Pony player) {
return RayTraceHelper.doTrace(player.getMaster(), 6 * getKickDirection(player), 1)
.getBlockPos()
2022-06-23 16:24:45 +02:00
.filter(pos -> TreeType.at(pos, player.getReferenceWorld()) != TreeType.NONE)
.map(Pos::new)
.orElseGet(() -> getDefaultKickLocation(player));
}
private int getKickDirection(Pony player) {
return MineLPDelegate.getInstance().getPlayerPonyRace(player.getMaster()).isDefault() ? 1 : -1;
}
private Pos getDefaultKickLocation(Pony player) {
Vec3d kickVector = player.getMaster().getRotationVector().multiply(1, 0, 1);
player.getMaster();
2022-09-20 23:50:15 +02:00
if (!MineLPDelegate.getInstance().getPlayerPonyRace(player.getMaster()).isDefault()) {
kickVector = kickVector.rotateY((float)Math.PI);
}
return new Pos(new BlockPos(player.getOriginVector().add(kickVector)));
2021-08-20 22:22:28 +02:00
}
2021-08-20 22:22:28 +02:00
@Override
public boolean canApply(Pony player, Pos data) {
BlockPos pos = data.pos();
2022-06-23 16:24:45 +02:00
TreeType tree = TreeType.at(pos, player.getReferenceWorld());
2022-06-23 16:24:45 +02:00
return tree == TreeType.NONE || tree.findBase(player.getReferenceWorld(), pos)
.map(base -> tree.countBlocks(player.getReferenceWorld(), pos) > 0)
.orElse(false);
}
@Override
public Hit.Serializer<Pos> getSerializer() {
return Pos.SERIALIZER;
}
@Override
public void apply(Pony iplayer, Pos data) {
BlockPos pos = data.pos();
2022-06-23 16:24:45 +02:00
TreeType tree = TreeType.at(pos, iplayer.getReferenceWorld());
iplayer.setAnimation(Animation.KICK);
iplayer.subtractEnergyCost(tree == TreeType.NONE ? 1 : 3);
if (tree == TreeType.NONE) {
return;
} else {
2022-06-23 16:24:45 +02:00
ParticleUtils.spawnParticle(iplayer.getReferenceWorld(), UParticles.GROUND_POUND, data.vec(), Vec3d.ZERO);
}
PlayerEntity player = iplayer.getMaster();
boolean harmed = player.getHealth() < player.getMaxHealth();
if (BlockDestructionManager.of(player.world).getBlockDestruction(pos) + 4 >= BlockDestructionManager.MAX_DAMAGE) {
if (!harmed || player.world.random.nextInt(30) == 0) {
tree.traverse(player.world, pos, (w, state, p, recurseLevel) -> {
2021-02-13 12:50:24 +01:00
if (recurseLevel < 5) {
w.breakBlock(p, true);
} else {
Block.dropStacks(w.getBlockState(p), w, p);
w.setBlockState(p, Blocks.AIR.getDefaultState(), 3);
}
});
}
iplayer.subtractEnergyCost(3);
} else {
2021-02-13 12:50:24 +01:00
int cost = dropApples(player, pos);
if (cost > 0) {
iplayer.subtractEnergyCost(cost / 7F);
}
}
}
@Override
public void preApply(Pony player, AbilitySlot slot) {
player.getMagicalReserves().getExertion().add(40);
}
@Override
public void postApply(Pony player, AbilitySlot slot) {
}
2021-02-13 12:50:24 +01:00
private int dropApples(PlayerEntity player, BlockPos pos) {
TreeType tree = TreeType.at(pos, player.world);
2021-02-13 12:50:24 +01:00
if (tree.countBlocks(player.world, pos) > 0) {
List<ItemEntity> capturedDrops = Lists.newArrayList();
2021-02-13 12:50:24 +01:00
tree.traverse(player.world, pos, (world, state, position, recurse) -> {
affectBlockChange(player, position);
}, (world, state, position, recurse) -> {
affectBlockChange(player, position);
BlockState below = world.getBlockState(position.down());
if (below.isAir()) {
ItemStack stack = tree.pickRandomStack(state);
if (!stack.isEmpty()) {
world.syncWorldEvent(WorldEvents.BLOCK_BROKEN, position, Block.getRawIdFromState(state));
capturedDrops.add(new ItemEntity(world,
2021-02-13 12:50:24 +01:00
position.getX() + world.random.nextFloat(),
position.getY() - 0.5,
position.getZ() + world.random.nextFloat(),
stack
2021-02-13 12:50:24 +01:00
));
}
} else if (below.getBlock() instanceof Buckable buckable) {
List<ItemStack> stacks = buckable.onBucked((ServerWorld)world, state, pos);
if (!stacks.isEmpty()) {
world.syncWorldEvent(WorldEvents.BLOCK_BROKEN, position, Block.getRawIdFromState(state));
stacks.forEach(stack -> {
capturedDrops.add(new ItemEntity(world,
position.getX() + world.random.nextFloat(),
position.getY() - 0.5,
position.getZ() + world.random.nextFloat(),
stack
));
});
}
2021-02-13 12:50:24 +01:00
}
});
capturedDrops.forEach(item -> {
item.setToDefaultPickupDelay();
2021-02-13 12:50:24 +01:00
player.world.spawnEntity(item);
});
return capturedDrops.size() / 3;
}
return 0;
}
2021-02-13 12:50:24 +01:00
private void affectBlockChange(PlayerEntity player, BlockPos position) {
BlockDestructionManager.of(player.world).damageBlock(position, 4);
PosHelper.all(position, p -> {
BlockState s = player.world.getBlockState(p);
if (s.getBlock() instanceof BeehiveBlock) {
BeehiveBlockEntity hive = (BeehiveBlockEntity)player.world.getBlockEntity(p);
if (hive != null) {
hive.angerBees(player, s, BeehiveBlockEntity.BeeState.BEE_RELEASED);
}
Box area = new Box(position).expand(8, 6, 8);
List<BeeEntity> nearbyBees = player.world.getNonSpectatingEntities(BeeEntity.class, area);
if (!nearbyBees.isEmpty()) {
List<PlayerEntity> nearbyPlayers = player.world.getNonSpectatingEntities(PlayerEntity.class, area);
int i = nearbyPlayers.size();
for (BeeEntity bee : nearbyBees) {
if (bee.getTarget() == null) {
bee.setTarget(nearbyPlayers.get(player.world.random.nextInt(i)));
}
}
}
}
}, PosHelper.HORIZONTAL);
}
public interface Buckable {
List<ItemStack> onBucked(ServerWorld world, BlockState state, BlockPos pos);
}
}