2020-04-15 14:22:03 +02:00
|
|
|
package com.minelittlepony.unicopia.ability;
|
2018-09-12 01:29:49 +02:00
|
|
|
|
2020-04-26 19:33:10 +02:00
|
|
|
import javax.annotation.Nullable;
|
|
|
|
|
2021-01-26 21:33:45 +01:00
|
|
|
import com.minelittlepony.unicopia.BlockDestructionManager;
|
2020-04-15 14:22:03 +02:00
|
|
|
import com.minelittlepony.unicopia.Race;
|
2020-04-25 15:37:17 +02:00
|
|
|
import com.minelittlepony.unicopia.ability.data.Hit;
|
2020-09-22 15:11:20 +02:00
|
|
|
import com.minelittlepony.unicopia.entity.player.Pony;
|
2020-04-15 14:22:03 +02:00
|
|
|
import com.minelittlepony.unicopia.util.MagicalDamageSource;
|
|
|
|
import com.minelittlepony.unicopia.util.WorldEvent;
|
2020-04-15 18:12:00 +02:00
|
|
|
import com.minelittlepony.unicopia.util.shape.Shape;
|
2020-04-15 14:22:03 +02:00
|
|
|
import com.minelittlepony.unicopia.util.shape.Sphere;
|
2018-09-12 01:29:49 +02:00
|
|
|
|
|
|
|
import net.minecraft.block.Block;
|
2020-01-16 12:35:46 +01:00
|
|
|
import net.minecraft.block.BlockState;
|
|
|
|
import net.minecraft.block.Blocks;
|
2020-06-26 11:44:47 +02:00
|
|
|
import net.minecraft.block.ShapeContext;
|
2020-01-16 12:35:46 +01:00
|
|
|
import net.minecraft.entity.attribute.EntityAttributes;
|
|
|
|
import net.minecraft.entity.damage.DamageSource;
|
|
|
|
import net.minecraft.entity.player.PlayerEntity;
|
|
|
|
import net.minecraft.particle.BlockStateParticleEffect;
|
|
|
|
import net.minecraft.particle.ParticleTypes;
|
2018-09-12 01:29:49 +02:00
|
|
|
import net.minecraft.util.math.BlockPos;
|
2020-01-16 12:35:46 +01:00
|
|
|
import net.minecraft.util.math.Box;
|
|
|
|
import net.minecraft.util.math.Direction;
|
2021-01-27 16:16:15 +01:00
|
|
|
import net.minecraft.util.math.MathHelper;
|
2018-09-12 01:29:49 +02:00
|
|
|
import net.minecraft.util.math.Vec3d;
|
|
|
|
import net.minecraft.world.World;
|
|
|
|
|
2020-01-27 17:37:22 +01:00
|
|
|
/**
|
|
|
|
* Earth Pony stomping ability
|
|
|
|
*/
|
2020-10-11 13:14:41 +02:00
|
|
|
public class EarthPonyStompAbility implements Ability<Hit> {
|
2018-09-12 01:29:49 +02:00
|
|
|
|
2019-02-03 10:45:45 +01:00
|
|
|
private final double rad = 4;
|
|
|
|
|
2020-01-16 12:35:46 +01:00
|
|
|
private final Box areaOfEffect = new Box(
|
2019-02-03 10:45:45 +01:00
|
|
|
-rad, -rad, -rad,
|
|
|
|
rad, rad, rad
|
|
|
|
);
|
|
|
|
|
2018-09-12 01:29:49 +02:00
|
|
|
@Override
|
2020-04-15 18:12:00 +02:00
|
|
|
public int getWarmupTime(Pony player) {
|
2019-02-07 10:46:59 +01:00
|
|
|
return 3;
|
2018-09-12 01:29:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
2020-04-15 18:12:00 +02:00
|
|
|
public int getCooldownTime(Pony player) {
|
2019-02-07 10:46:59 +01:00
|
|
|
return 50;
|
2018-09-12 01:29:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
2020-05-10 17:18:45 +02:00
|
|
|
public boolean canUse(Race race) {
|
|
|
|
return race.canUseEarth();
|
2018-09-12 01:29:49 +02:00
|
|
|
}
|
|
|
|
|
2020-10-09 19:05:12 +02:00
|
|
|
@Override
|
|
|
|
public double getCostEstimate(Pony player) {
|
|
|
|
return rad;
|
|
|
|
}
|
|
|
|
|
2020-04-26 19:33:10 +02:00
|
|
|
@Nullable
|
2018-09-12 01:29:49 +02:00
|
|
|
@Override
|
2020-10-11 13:14:41 +02:00
|
|
|
public Hit tryActivate(Pony player) {
|
2020-10-08 19:22:20 +02:00
|
|
|
if (!player.getMaster().isOnGround() && !player.getMaster().abilities.flying) {
|
|
|
|
player.getMaster().addVelocity(0, -6, 0);
|
2020-10-11 13:14:41 +02:00
|
|
|
return Hit.INSTANCE;
|
2018-09-12 01:29:49 +02:00
|
|
|
}
|
2020-04-26 19:33:10 +02:00
|
|
|
|
2018-09-12 01:29:49 +02:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
2020-10-11 13:14:41 +02:00
|
|
|
public Hit.Serializer<Hit> getSerializer() {
|
|
|
|
return Hit.SERIALIZER;
|
2018-09-12 01:29:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
2020-10-11 13:14:41 +02:00
|
|
|
public void apply(Pony iplayer, Hit data) {
|
2020-10-08 19:22:20 +02:00
|
|
|
PlayerEntity player = iplayer.getMaster();
|
2018-09-12 22:37:06 +02:00
|
|
|
|
2020-10-11 13:14:41 +02:00
|
|
|
BlockPos ppos = player.getBlockPos();
|
|
|
|
BlockPos pos = getSolidBlockBelow(ppos, player.getEntityWorld());
|
2018-09-13 14:04:24 +02:00
|
|
|
|
2020-10-11 13:14:41 +02:00
|
|
|
player.addVelocity(0, -(ppos.getSquaredDistance(pos)), 0);
|
2018-09-13 14:04:24 +02:00
|
|
|
|
2020-10-11 13:14:41 +02:00
|
|
|
iplayer.getWorld().getOtherEntities(player, areaOfEffect.offset(iplayer.getOriginVector())).forEach(i -> {
|
|
|
|
double dist = Math.sqrt(pos.getSquaredDistance(i.getBlockPos()));
|
2018-09-13 14:04:24 +02:00
|
|
|
|
2020-10-11 13:14:41 +02:00
|
|
|
if (dist <= rad + 3) {
|
|
|
|
double force = dist / 5;
|
|
|
|
i.addVelocity(
|
|
|
|
-(player.getX() - i.getX()) / force,
|
|
|
|
-(player.getY() - i.getY() - 2) / force + (dist < 1 ? dist : 0),
|
|
|
|
-(player.getZ() - i.getZ()) / force);
|
2018-09-13 14:04:24 +02:00
|
|
|
|
2020-10-11 13:14:41 +02:00
|
|
|
DamageSource damage = MagicalDamageSource.create("smash", player);
|
2018-09-13 14:04:24 +02:00
|
|
|
|
2020-10-11 13:14:41 +02:00
|
|
|
double amount = (4 * player.getAttributeInstance(EntityAttributes.GENERIC_ATTACK_DAMAGE).getValue()) / (float)dist;
|
2018-09-13 14:04:24 +02:00
|
|
|
|
2020-10-11 13:14:41 +02:00
|
|
|
if (i instanceof PlayerEntity) {
|
|
|
|
Race race = Pony.of((PlayerEntity)i).getSpecies();
|
|
|
|
if (race.canUseEarth()) {
|
|
|
|
amount /= 3;
|
2018-09-12 01:29:49 +02:00
|
|
|
}
|
2018-09-13 14:04:24 +02:00
|
|
|
|
2020-10-11 13:14:41 +02:00
|
|
|
if (race.canFly()) {
|
|
|
|
amount *= 4;
|
|
|
|
}
|
2018-09-12 01:29:49 +02:00
|
|
|
}
|
2018-09-12 22:37:06 +02:00
|
|
|
|
2020-10-11 13:14:41 +02:00
|
|
|
i.damage(damage, (float)amount);
|
2018-09-12 01:29:49 +02:00
|
|
|
}
|
2020-10-11 13:14:41 +02:00
|
|
|
});
|
2018-09-12 22:37:06 +02:00
|
|
|
|
2020-10-11 13:14:41 +02:00
|
|
|
BlockPos.iterate(pos.add(-rad, -rad, -rad), pos.add(rad, rad, rad)).forEach(i -> {
|
2021-01-26 21:33:45 +01:00
|
|
|
double dist = Math.sqrt(i.getSquaredDistance(player.getX(), player.getY(), player.getZ(), true));
|
|
|
|
|
|
|
|
if (dist <= rad) {
|
|
|
|
spawnEffect(player.world, i, dist);
|
2019-02-09 20:08:32 +01:00
|
|
|
}
|
2020-10-11 13:14:41 +02:00
|
|
|
});
|
2019-02-09 20:08:32 +01:00
|
|
|
|
2020-10-11 13:14:41 +02:00
|
|
|
for (int i = 1; i < 202; i+= 2) {
|
|
|
|
spawnParticleRing(player, i, 0);
|
2018-09-12 01:29:49 +02:00
|
|
|
}
|
2020-10-11 13:14:41 +02:00
|
|
|
|
|
|
|
iplayer.subtractEnergyCost(rad);
|
2018-09-12 01:29:49 +02:00
|
|
|
}
|
|
|
|
|
2021-01-26 21:33:45 +01:00
|
|
|
private void spawnEffect(World w, BlockPos pos, double dist) {
|
2020-01-16 12:35:46 +01:00
|
|
|
BlockState state = w.getBlockState(pos);
|
2021-01-26 21:33:45 +01:00
|
|
|
BlockDestructionManager destr = ((BlockDestructionManager.Source)w).getDestructionManager();
|
2018-09-12 01:29:49 +02:00
|
|
|
|
2020-01-16 12:35:46 +01:00
|
|
|
if (!state.isAir() && w.getBlockState(pos.up()).isAir()) {
|
2021-01-27 16:16:15 +01:00
|
|
|
|
|
|
|
double amount = (1 - dist / rad) * 9;
|
|
|
|
float hardness = state.getHardness(w, pos);
|
|
|
|
float scaledHardness = (1 - hardness / 70);
|
|
|
|
|
|
|
|
int damage = hardness < 0 ? 0 : MathHelper.clamp((int)(amount * scaledHardness), 2, 9);
|
|
|
|
|
|
|
|
if (destr.damageBlock(pos, damage) >= BlockDestructionManager.MAX_DAMAGE) {
|
2021-01-26 21:33:45 +01:00
|
|
|
w.breakBlock(pos, true);
|
|
|
|
} else {
|
|
|
|
WorldEvent.play(WorldEvent.DESTROY_BLOCK, w, pos, state);
|
|
|
|
}
|
2018-09-12 01:29:49 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
2020-05-10 20:45:07 +02:00
|
|
|
public void preApply(Pony player, AbilitySlot slot) {
|
2020-10-01 17:04:48 +02:00
|
|
|
player.getMagicalReserves().getExertion().add(40);
|
2018-09-12 01:29:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
2020-05-10 20:45:07 +02:00
|
|
|
public void postApply(Pony player, AbilitySlot slot) {
|
|
|
|
int timeDiff = getCooldownTime(player) - player.getAbilities().getStat(slot).getRemainingCooldown();
|
2018-09-12 01:29:49 +02:00
|
|
|
|
2020-10-08 19:22:20 +02:00
|
|
|
if (player.getMaster().getEntityWorld().getTime() % 1 == 0 || timeDiff == 0) {
|
|
|
|
spawnParticleRing(player.getMaster(), timeDiff, 1);
|
2018-09-12 01:29:49 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-16 12:35:46 +01:00
|
|
|
private void spawnParticleRing(PlayerEntity player, int timeDiff, double yVel) {
|
2021-01-27 16:16:15 +01:00
|
|
|
int animationTicks = timeDiff / 7;
|
2018-09-12 01:29:49 +02:00
|
|
|
if (animationTicks < 6) {
|
2020-04-15 18:12:00 +02:00
|
|
|
Shape shape = new Sphere(true, animationTicks, 1, 0, 1);
|
2018-09-12 01:29:49 +02:00
|
|
|
|
|
|
|
double y = 0.5 + (Math.sin(animationTicks) * 1.5);
|
|
|
|
|
|
|
|
yVel *= y * 5;
|
|
|
|
|
|
|
|
for (int i = 0; i < shape.getVolumeOfSpawnableSpace(); i++) {
|
2020-04-22 16:28:20 +02:00
|
|
|
Vec3d point = shape.computePoint(player.getEntityWorld().random).add(player.getPos());
|
2020-01-16 12:35:46 +01:00
|
|
|
player.world.addParticle(new BlockStateParticleEffect(ParticleTypes.BLOCK, Blocks.DIRT.getDefaultState()),
|
2020-04-22 16:28:20 +02:00
|
|
|
point.x,
|
2021-01-27 16:16:15 +01:00
|
|
|
point.y,
|
2020-04-22 16:28:20 +02:00
|
|
|
point.z,
|
2020-01-16 12:35:46 +01:00
|
|
|
0, yVel, 0
|
|
|
|
);
|
2018-09-12 01:29:49 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-26 19:33:10 +02:00
|
|
|
private static BlockPos getSolidBlockBelow(BlockPos pos, World w) {
|
2020-12-17 19:17:47 +01:00
|
|
|
while (!World.isOutOfBuildLimitVertically(pos)) {
|
2018-09-12 01:29:49 +02:00
|
|
|
pos = pos.down();
|
|
|
|
|
2020-06-26 11:44:47 +02:00
|
|
|
if (Block.isFaceFullSquare(w.getBlockState(pos).getCollisionShape(w, pos, ShapeContext.absent()), Direction.UP)) {
|
2020-04-26 19:33:10 +02:00
|
|
|
return pos;
|
|
|
|
}
|
2018-09-12 01:29:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return pos;
|
|
|
|
}
|
|
|
|
}
|