From 6e7b387e6ad1b945152c7652c4862463c7edf3ab Mon Sep 17 00:00:00 2001 From: WayofTime Date: Thu, 29 Mar 2018 16:05:56 -0400 Subject: [PATCH 1/6] Added additional effects to the Sentient Bow when aspected to different Will types. --- changelog.txt | 1 + .../projectile/EntitySentientArrow.java | 113 +++++++-- .../bloodmagic/item/soul/ItemSentientBow.java | 228 +++++++++++------- 3 files changed, 245 insertions(+), 97 deletions(-) diff --git a/changelog.txt b/changelog.txt index e27942ad..4716494f 100644 --- a/changelog.txt +++ b/changelog.txt @@ -8,6 +8,7 @@ Version 2.2.8 - Added a new (slightly WIP?) array, the Turret Array: - Place an array on top of an inventory with arrows and then place a bow and an arrow in the array. The array will target enemies greater than 3 blocks away and less than 32, using any arrows in the inventory. - Increased the max number of items transferable by the Master Routing Node in its system to 64 per second. Will revisit this limit if I figure out a less silly upgrade system. +- Added additional effects to the Sentient Bow when aspected to different Will types. ------------------------------------------------------ Version 2.2.7 diff --git a/src/main/java/WayofTime/bloodmagic/entity/projectile/EntitySentientArrow.java b/src/main/java/WayofTime/bloodmagic/entity/projectile/EntitySentientArrow.java index b4950f64..1e91024e 100644 --- a/src/main/java/WayofTime/bloodmagic/entity/projectile/EntitySentientArrow.java +++ b/src/main/java/WayofTime/bloodmagic/entity/projectile/EntitySentientArrow.java @@ -1,41 +1,59 @@ package WayofTime.bloodmagic.entity.projectile; -import WayofTime.bloodmagic.util.Constants; -import WayofTime.bloodmagic.soul.EnumDemonWillType; -import WayofTime.bloodmagic.soul.PlayerDemonWillHandler; +import java.util.Locale; + import net.minecraft.entity.EntityLivingBase; import net.minecraft.entity.monster.IMob; import net.minecraft.entity.player.EntityPlayer; import net.minecraft.entity.projectile.EntityTippedArrow; import net.minecraft.init.Items; +import net.minecraft.init.MobEffects; import net.minecraft.item.ItemStack; import net.minecraft.nbt.NBTTagCompound; +import net.minecraft.potion.PotionEffect; import net.minecraft.world.EnumDifficulty; import net.minecraft.world.World; +import WayofTime.bloodmagic.soul.EnumDemonWillType; +import WayofTime.bloodmagic.soul.PlayerDemonWillHandler; +import WayofTime.bloodmagic.util.Constants; -import java.util.Locale; - -public class EntitySentientArrow extends EntityTippedArrow { +public class EntitySentientArrow extends EntityTippedArrow +{ public double reimbursedAmountOnHit = 0; public EnumDemonWillType type = EnumDemonWillType.DEFAULT; + public int currentLevel = 0; + public float[] destructiveExplosionRadius = { 0.5f, 1, 1.5f, 2, 2.5f, 3, 3.5f }; + public int[] poisonDuration = { 50, 100, 150, 80, 120, 160, 200 }; + public int[] poisonLevel = { 0, 0, 0, 1, 1, 1, 1 }; + public int[] levitationDuration = { 20, 40, 60, 80, 100, 120, 160 }; + public int[] levitationLevel = { 0, 0, 0, 1, 1, 1, 2 }; + public int[] slownessDuration = { 40, 60, 100, 150, 200, 250, 300 }; + public int[] slownessLevel = { 0, 0, 0, 1, 1, 1, 2 }; - public EntitySentientArrow(World worldIn) { + public EntitySentientArrow(World worldIn) + { super(worldIn); } - public EntitySentientArrow(World worldIn, double x, double y, double z) { + public EntitySentientArrow(World worldIn, double x, double y, double z) + { super(worldIn, x, y, z); } - public EntitySentientArrow(World worldIn, EntityLivingBase shooter, EnumDemonWillType type, double reinburseAmount) { + public EntitySentientArrow(World worldIn, EntityLivingBase shooter, EnumDemonWillType type, double reinburseAmount, int currentLevel) + { super(worldIn, shooter); this.reimbursedAmountOnHit = reinburseAmount; this.type = type; + this.currentLevel = currentLevel; } - public void reimbursePlayer(EntityLivingBase hitEntity, float damage) { - if (this.shootingEntity instanceof EntityPlayer) { - if (hitEntity.getEntityWorld().getDifficulty() != EnumDifficulty.PEACEFUL && !(hitEntity instanceof IMob)) { + public void reimbursePlayer(EntityLivingBase hitEntity, float damage) + { + if (this.shootingEntity instanceof EntityPlayer) + { + if (hitEntity.getEntityWorld().getDifficulty() != EnumDifficulty.PEACEFUL && !(hitEntity instanceof IMob)) + { return; } @@ -44,23 +62,88 @@ public class EntitySentientArrow extends EntityTippedArrow { } @Override - public void writeEntityToNBT(NBTTagCompound tag) { + protected void arrowHit(EntityLivingBase living) + { + super.arrowHit(living); + + switch (type) + { + case CORROSIVE: + living.addPotionEffect(new PotionEffect(MobEffects.POISON, currentLevel >= 0 ? poisonDuration[currentLevel] : 0, currentLevel >= 0 ? poisonLevel[currentLevel] : 0)); + break; + case DEFAULT: + break; + case DESTRUCTIVE: + this.world.createExplosion(this, this.posX, this.posY, this.posZ, currentLevel >= 0 ? destructiveExplosionRadius[currentLevel] : 0, false); + break; + case STEADFAST: + living.addPotionEffect(new PotionEffect(MobEffects.LEVITATION, currentLevel >= 0 ? levitationDuration[currentLevel] : 0, currentLevel >= 0 ? levitationLevel[currentLevel] : 0)); + break; + case VENGEFUL: + living.addPotionEffect(new PotionEffect(MobEffects.SLOWNESS, currentLevel >= 0 ? slownessDuration[currentLevel] : 0, currentLevel >= 0 ? slownessLevel[currentLevel] : 0)); + break; + default: + break; + } + } + + @Override + public void onUpdate() + { + super.onUpdate(); + + if (!this.world.isRemote && this.inGround && this.timeInGround > 0) + { + switch (type) + { + case DESTRUCTIVE: + this.world.createExplosion(this, this.posX, this.posY, this.posZ, currentLevel >= 0 ? destructiveExplosionRadius[currentLevel] : 0, false); + this.setDead(); + break; + case CORROSIVE: + break; + case DEFAULT: + break; + case STEADFAST: + break; + case VENGEFUL: + break; + default: + break; + } + } +// else if (this.inGround && this.timeInGround != 0 && !this.customPotionEffects.isEmpty() && this.timeInGround >= 600) +// { +// this.world.setEntityState(this, (byte)0); +// this.potion = PotionTypes.EMPTY; +// this.customPotionEffects.clear(); +// this.dataManager.set(COLOR, Integer.valueOf(-1)); +// } + } + + @Override + public void writeEntityToNBT(NBTTagCompound tag) + { super.writeEntityToNBT(tag); tag.setDouble("reimbursement", reimbursedAmountOnHit); + tag.setInteger("currentLevel", currentLevel); tag.setString(Constants.NBT.WILL_TYPE, type.toString()); } @Override - public void readEntityFromNBT(NBTTagCompound tag) { + public void readEntityFromNBT(NBTTagCompound tag) + { super.readEntityFromNBT(tag); reimbursedAmountOnHit = tag.getDouble("reimbursement"); type = EnumDemonWillType.valueOf(tag.getString(Constants.NBT.WILL_TYPE).toUpperCase(Locale.ENGLISH)); + currentLevel = tag.getInteger("currentLevel"); } @Override - protected ItemStack getArrowStack() { + protected ItemStack getArrowStack() + { return new ItemStack(Items.ARROW); } } diff --git a/src/main/java/WayofTime/bloodmagic/item/soul/ItemSentientBow.java b/src/main/java/WayofTime/bloodmagic/item/soul/ItemSentientBow.java index b4b104bc..70e48450 100644 --- a/src/main/java/WayofTime/bloodmagic/item/soul/ItemSentientBow.java +++ b/src/main/java/WayofTime/bloodmagic/item/soul/ItemSentientBow.java @@ -40,56 +40,68 @@ import java.util.Locale; public class ItemSentientBow extends ItemBow implements IMultiWillTool, ISentientTool, IVariantProvider//, IMeshProvider { - public static int[] soulBracket = new int[]{16, 60, 200, 400, 1000}; - public static double[] defaultDamageAdded = new double[]{0.25, 0.5, 0.75, 1, 1.25}; - public static float[] velocityAdded = new float[]{0.25f, 0.5f, 0.75f, 1, 1.25f}; - public static double[] soulDrainPerSwing = new double[]{0.05, 0.1, 0.2, 0.4, 0.75}; //TODO - public static double[] soulDrop = new double[]{2, 4, 7, 10, 13}; - public static double[] staticDrop = new double[]{1, 1, 2, 3, 3}; + public static int[] soulBracket = new int[] { 16, 60, 200, 400, 1000, 2000, 4000 }; + public static double[] defaultDamageAdded = new double[] { 0.25, 0.5, 0.75, 1, 1.25, 1.5, 1.75 }; + public static float[] velocityAdded = new float[] { 0.25f, 0.5f, 0.75f, 1, 1.25f, 1.5f, 1.75f }; + public static double[] soulDrainPerSwing = new double[] { 0.05, 0.1, 0.2, 0.4, 0.75, 1, 1.5 }; //TODO + public static double[] soulDrop = new double[] { 2, 4, 7, 10, 13, 16, 24 }; + public static double[] staticDrop = new double[] { 1, 1, 2, 3, 3, 3, 4 }; - public ItemSentientBow() { + public ItemSentientBow() + { super(); setUnlocalizedName(BloodMagic.MODID + ".sentientBow"); setCreativeTab(BloodMagic.TAB_BM); - this.addPropertyOverride(new ResourceLocation("pull"), new IItemPropertyGetter() { + this.addPropertyOverride(new ResourceLocation("pull"), new IItemPropertyGetter() + { @SideOnly(Side.CLIENT) - public float apply(ItemStack stack, World world, EntityLivingBase entityIn) { - if (entityIn == null) { + public float apply(ItemStack stack, World world, EntityLivingBase entityIn) + { + if (entityIn == null) + { return 0.0F; - } else { + } else + { ItemStack itemstack = entityIn.getActiveItemStack(); return !itemstack.isEmpty() && itemstack.getItem() == RegistrarBloodMagicItems.SENTIENT_BOW ? (float) (stack.getMaxItemUseDuration() - entityIn.getItemInUseCount()) / 20.0F : 0.0F; } } }); - this.addPropertyOverride(new ResourceLocation("pulling"), new IItemPropertyGetter() { + this.addPropertyOverride(new ResourceLocation("pulling"), new IItemPropertyGetter() + { @SideOnly(Side.CLIENT) - public float apply(ItemStack stack, World world, EntityLivingBase entityIn) { + public float apply(ItemStack stack, World world, EntityLivingBase entityIn) + { return entityIn != null && entityIn.isHandActive() && entityIn.getActiveItemStack() == stack ? 1.0F : 0.0F; } }); - this.addPropertyOverride(new ResourceLocation("type"), new IItemPropertyGetter() { + this.addPropertyOverride(new ResourceLocation("type"), new IItemPropertyGetter() + { @SideOnly(Side.CLIENT) - public float apply(ItemStack stack, World world, EntityLivingBase entityIn) { + public float apply(ItemStack stack, World world, EntityLivingBase entityIn) + { return ((ItemSentientBow) RegistrarBloodMagicItems.SENTIENT_BOW).getCurrentType(stack).ordinal(); } }); } @Override - public boolean getIsRepairable(ItemStack toRepair, ItemStack repair) { + public boolean getIsRepairable(ItemStack toRepair, ItemStack repair) + { return RegistrarBloodMagicItems.ITEM_DEMON_CRYSTAL == repair.getItem() || super.getIsRepairable(toRepair, repair); } - public void recalculatePowers(ItemStack stack, World world, EntityPlayer player) { + public void recalculatePowers(ItemStack stack, World world, EntityPlayer player) + { EnumDemonWillType type = PlayerDemonWillHandler.getLargestWillType(player); double soulsRemaining = PlayerDemonWillHandler.getTotalDemonWill(type, player); recalculatePowers(stack, type, soulsRemaining); } - public void recalculatePowers(ItemStack stack, EnumDemonWillType type, double will) { + public void recalculatePowers(ItemStack stack, EnumDemonWillType type, double will) + { this.setCurrentType(stack, will > 0 ? type : EnumDemonWillType.DEFAULT); - int level = getLevel(stack, will); + int level = getLevel(will); // double drain = level >= 0 ? soulDrainPerSwing[level] : 0; @@ -108,10 +120,13 @@ public class ItemSentientBow extends ItemBow implements IMultiWillTool, ISentien setDamageAdded(stack, level >= 0 ? getDamageModifier(type, level) : 0); } - private int getLevel(ItemStack stack, double soulsRemaining) { + private int getLevel(double soulsRemaining) + { int lvl = -1; - for (int i = 0; i < soulBracket.length; i++) { - if (soulsRemaining >= soulBracket[i]) { + for (int i = 0; i < soulBracket.length; i++) + { + if (soulsRemaining >= soulBracket[i]) + { lvl = i; } } @@ -120,42 +135,49 @@ public class ItemSentientBow extends ItemBow implements IMultiWillTool, ISentien } @Override - public EnumDemonWillType getCurrentType(ItemStack stack) { + public EnumDemonWillType getCurrentType(ItemStack stack) + { NBTHelper.checkNBT(stack); NBTTagCompound tag = stack.getTagCompound(); - if (!tag.hasKey(Constants.NBT.WILL_TYPE)) { + if (!tag.hasKey(Constants.NBT.WILL_TYPE)) + { return EnumDemonWillType.DEFAULT; } return EnumDemonWillType.valueOf(tag.getString(Constants.NBT.WILL_TYPE).toUpperCase(Locale.ENGLISH)); } - public double getDamageModifier(EnumDemonWillType type, int willBracket) { - switch (type) { - case VENGEFUL: - return 0; - case DEFAULT: - case CORROSIVE: - case DESTRUCTIVE: - case STEADFAST: - return defaultDamageAdded[willBracket]; + public double getDamageModifier(EnumDemonWillType type, int willBracket) + { + switch (type) + { + case VENGEFUL: + return 0; + case DEFAULT: + case CORROSIVE: + case DESTRUCTIVE: + case STEADFAST: + return defaultDamageAdded[willBracket]; } return 0; } - public float getVelocityModifier(EnumDemonWillType type, int willBracket) { - switch (type) { - case VENGEFUL: - return velocityAdded[willBracket]; - default: - return 0; + public float getVelocityModifier(EnumDemonWillType type, int willBracket) + { + switch (type) + { + case VENGEFUL: + return velocityAdded[willBracket]; + default: + return 0; } } - public void setDamageAdded(ItemStack stack, double damage) { + public void setDamageAdded(ItemStack stack, double damage) + { NBTHelper.checkNBT(stack); NBTTagCompound tag = stack.getTagCompound(); @@ -163,7 +185,8 @@ public class ItemSentientBow extends ItemBow implements IMultiWillTool, ISentien tag.setDouble("damage", damage); } - public double getDamageAdded(ItemStack stack) { + public double getDamageAdded(ItemStack stack) + { NBTHelper.checkNBT(stack); NBTTagCompound tag = stack.getTagCompound(); @@ -171,7 +194,8 @@ public class ItemSentientBow extends ItemBow implements IMultiWillTool, ISentien return tag.getDouble("damage"); } - public void setVelocityOfArrow(ItemStack stack, float velocity) { + public void setVelocityOfArrow(ItemStack stack, float velocity) + { NBTHelper.checkNBT(stack); NBTTagCompound tag = stack.getTagCompound(); @@ -179,19 +203,22 @@ public class ItemSentientBow extends ItemBow implements IMultiWillTool, ISentien tag.setFloat("velocity", velocity); } - public float getVelocityOfArrow(ItemStack stack) { + public float getVelocityOfArrow(ItemStack stack) + { NBTHelper.checkNBT(stack); NBTTagCompound tag = stack.getTagCompound(); - if (tag.hasKey("velocity")) { + if (tag.hasKey("velocity")) + { return tag.getFloat("velocity"); } return 3; } - public void setCurrentType(ItemStack stack, EnumDemonWillType type) { + public void setCurrentType(ItemStack stack, EnumDemonWillType type) + { NBTHelper.checkNBT(stack); NBTTagCompound tag = stack.getTagCompound(); @@ -199,14 +226,16 @@ public class ItemSentientBow extends ItemBow implements IMultiWillTool, ISentien tag.setString(Constants.NBT.WILL_TYPE, type.toString()); } - public double getDrainOfActivatedBow(ItemStack stack) { + public double getDrainOfActivatedBow(ItemStack stack) + { NBTHelper.checkNBT(stack); NBTTagCompound tag = stack.getTagCompound(); return tag.getDouble(Constants.NBT.SOUL_SWORD_ACTIVE_DRAIN); } - public void setDrainOfActivatedBow(ItemStack stack, double drain) { + public void setDrainOfActivatedBow(ItemStack stack, double drain) + { NBTHelper.checkNBT(stack); NBTTagCompound tag = stack.getTagCompound(); @@ -214,14 +243,16 @@ public class ItemSentientBow extends ItemBow implements IMultiWillTool, ISentien tag.setDouble(Constants.NBT.SOUL_SWORD_ACTIVE_DRAIN, drain); } - public double getStaticDropOfActivatedBow(ItemStack stack) { + public double getStaticDropOfActivatedBow(ItemStack stack) + { NBTHelper.checkNBT(stack); NBTTagCompound tag = stack.getTagCompound(); return tag.getDouble(Constants.NBT.SOUL_SWORD_STATIC_DROP); } - public void setStaticDropOfActivatedBow(ItemStack stack, double drop) { + public void setStaticDropOfActivatedBow(ItemStack stack, double drop) + { NBTHelper.checkNBT(stack); NBTTagCompound tag = stack.getTagCompound(); @@ -229,14 +260,16 @@ public class ItemSentientBow extends ItemBow implements IMultiWillTool, ISentien tag.setDouble(Constants.NBT.SOUL_SWORD_STATIC_DROP, drop); } - public double getDropOfActivatedBow(ItemStack stack) { + public double getDropOfActivatedBow(ItemStack stack) + { NBTHelper.checkNBT(stack); NBTTagCompound tag = stack.getTagCompound(); return tag.getDouble(Constants.NBT.SOUL_SWORD_DROP); } - public void setDropOfActivatedBow(ItemStack stack, double drop) { + public void setDropOfActivatedBow(ItemStack stack, double drop) + { NBTHelper.checkNBT(stack); NBTTagCompound tag = stack.getTagCompound(); @@ -245,24 +278,28 @@ public class ItemSentientBow extends ItemBow implements IMultiWillTool, ISentien } @Override - public ActionResult onItemRightClick(World world, EntityPlayer player, EnumHand hand) { + public ActionResult onItemRightClick(World world, EntityPlayer player, EnumHand hand) + { ItemStack stack = player.getHeldItem(hand); this.recalculatePowers(stack, world, player); return super.onItemRightClick(world, player, hand); } @Override - public void gatherVariants(@Nonnull Int2ObjectMap variants) { + public void gatherVariants(@Nonnull Int2ObjectMap variants) + { variants.put(0, "inventory"); } - public EntityTippedArrow getArrowEntity(World world, ItemStack stack, EntityLivingBase target, EntityLivingBase user, float velocity) { + public EntityTippedArrow getArrowEntity(World world, ItemStack stack, EntityLivingBase target, EntityLivingBase user, float velocity) + { EnumDemonWillType type = this.getCurrentType(stack); double amount = user instanceof EntityPlayer ? (this.getDropOfActivatedBow(stack) * world.rand.nextDouble() + this.getStaticDropOfActivatedBow(stack)) : 0; float newArrowVelocity = velocity * getVelocityOfArrow(stack); - EntitySentientArrow entityArrow = new EntitySentientArrow(world, user, type, amount); + double soulsRemaining = user instanceof EntityPlayer ? (PlayerDemonWillHandler.getTotalDemonWill(type, (EntityPlayer) user)) : 0; + EntitySentientArrow entityArrow = new EntitySentientArrow(world, user, type, amount, getLevel(soulsRemaining)); double d0 = target.posX - user.posX; double d1 = target.getEntityBoundingBox().minY + (double) (target.height / 3.0F) - entityArrow.posY; @@ -270,12 +307,14 @@ public class ItemSentientBow extends ItemBow implements IMultiWillTool, ISentien double d3 = (double) MathHelper.sqrt(d0 * d0 + d2 * d2); entityArrow.shoot(d0, d1 + d3 * 0.05, d2, newArrowVelocity, 0); - if (newArrowVelocity == 0) { + if (newArrowVelocity == 0) + { world.playSound(null, user.getPosition(), SoundEvents.BLOCK_FIRE_EXTINGUISH, SoundCategory.NEUTRAL, 0.4F, 1.0F); return null; } - if (velocity == 1.0F) { + if (velocity == 1.0F) + { entityArrow.setIsCritical(true); } @@ -285,11 +324,13 @@ public class ItemSentientBow extends ItemBow implements IMultiWillTool, ISentien int k = EnchantmentHelper.getEnchantmentLevel(Enchantments.PUNCH, stack); - if (k > 0) { + if (k > 0) + { entityArrow.setKnockbackStrength(k); } - if (EnchantmentHelper.getEnchantmentLevel(Enchantments.FLAME, stack) > 0) { + if (EnchantmentHelper.getEnchantmentLevel(Enchantments.FLAME, stack) > 0) + { entityArrow.setFire(100); } @@ -299,8 +340,10 @@ public class ItemSentientBow extends ItemBow implements IMultiWillTool, ISentien } @Override - public void onPlayerStoppedUsing(ItemStack stack, World world, EntityLivingBase entityLiving, int timeLeft) { - if (entityLiving instanceof EntityPlayer) { + public void onPlayerStoppedUsing(ItemStack stack, World world, EntityLivingBase entityLiving, int timeLeft) + { + if (entityLiving instanceof EntityPlayer) + { EntityPlayer player = (EntityPlayer) entityLiving; boolean flag = player.capabilities.isCreativeMode || EnchantmentHelper.getEnchantmentLevel(Enchantments.INFINITY, stack) > 0; ItemStack itemstack = this.getFiredArrow(player); @@ -310,17 +353,21 @@ public class ItemSentientBow extends ItemBow implements IMultiWillTool, ISentien if (i < 0) return; - if (itemstack != null || flag) { - if (itemstack == null) { + if (itemstack != null || flag) + { + if (itemstack == null) + { itemstack = new ItemStack(Items.ARROW); } float arrowVelocity = getArrowVelocity(i); - if ((double) arrowVelocity >= 0.1D) { + if ((double) arrowVelocity >= 0.1D) + { boolean flag1 = flag && itemstack.getItem() == Items.ARROW; //Forge: Fix consuming custom arrows. - if (!world.isRemote) { + if (!world.isRemote) + { this.recalculatePowers(stack, world, player); EnumDemonWillType type = this.getCurrentType(stack); @@ -331,15 +378,18 @@ public class ItemSentientBow extends ItemBow implements IMultiWillTool, ISentien double amount = (this.getDropOfActivatedBow(stack) * world.rand.nextDouble() + this.getStaticDropOfActivatedBow(stack)); float newArrowVelocity = arrowVelocity * getVelocityOfArrow(stack); - EntitySentientArrow entityArrow = new EntitySentientArrow(world, entityLiving, type, amount); + double soulsRemaining = PlayerDemonWillHandler.getTotalDemonWill(type, player); + EntitySentientArrow entityArrow = new EntitySentientArrow(world, entityLiving, type, amount, getLevel(soulsRemaining)); entityArrow.shoot(player, player.rotationPitch, player.rotationYaw, 0.0F, newArrowVelocity, 1.0F); - if (newArrowVelocity == 0) { + if (newArrowVelocity == 0) + { world.playSound(null, player.getPosition(), SoundEvents.BLOCK_FIRE_EXTINGUISH, SoundCategory.NEUTRAL, 0.4F, 1.0F); return; } - if (arrowVelocity == 1.0F) { + if (arrowVelocity == 1.0F) + { entityArrow.setIsCritical(true); } @@ -349,17 +399,20 @@ public class ItemSentientBow extends ItemBow implements IMultiWillTool, ISentien int k = EnchantmentHelper.getEnchantmentLevel(Enchantments.PUNCH, stack); - if (k > 0) { + if (k > 0) + { entityArrow.setKnockbackStrength(k); } - if (EnchantmentHelper.getEnchantmentLevel(Enchantments.FLAME, stack) > 0) { + if (EnchantmentHelper.getEnchantmentLevel(Enchantments.FLAME, stack) > 0) + { entityArrow.setFire(100); } stack.damageItem(1, player); - if (flag1) { + if (flag1) + { entityArrow.pickupStatus = EntityArrow.PickupStatus.CREATIVE_ONLY; } @@ -368,10 +421,12 @@ public class ItemSentientBow extends ItemBow implements IMultiWillTool, ISentien world.playSound(null, player.posX, player.posY, player.posZ, SoundEvents.ENTITY_ARROW_SHOOT, SoundCategory.NEUTRAL, 1.0F, 1.0F / (itemRand.nextFloat() * 0.4F + 1.2F) + arrowVelocity * 0.5F); - if (!flag1) { + if (!flag1) + { itemstack.shrink(1); - if (itemstack.isEmpty()) { + if (itemstack.isEmpty()) + { player.inventory.deleteStack(itemstack); } } @@ -382,16 +437,22 @@ public class ItemSentientBow extends ItemBow implements IMultiWillTool, ISentien } } - protected ItemStack getFiredArrow(EntityPlayer player) { - if (this.isArrow(player.getHeldItem(EnumHand.OFF_HAND))) { + protected ItemStack getFiredArrow(EntityPlayer player) + { + if (this.isArrow(player.getHeldItem(EnumHand.OFF_HAND))) + { return player.getHeldItem(EnumHand.OFF_HAND); - } else if (this.isArrow(player.getHeldItem(EnumHand.MAIN_HAND))) { + } else if (this.isArrow(player.getHeldItem(EnumHand.MAIN_HAND))) + { return player.getHeldItem(EnumHand.MAIN_HAND); - } else { - for (int i = 0; i < player.inventory.getSizeInventory(); ++i) { + } else + { + for (int i = 0; i < player.inventory.getSizeInventory(); ++i) + { ItemStack itemstack = player.inventory.getStackInSlot(i); - if (this.isArrow(itemstack)) { + if (this.isArrow(itemstack)) + { return itemstack; } } @@ -401,14 +462,17 @@ public class ItemSentientBow extends ItemBow implements IMultiWillTool, ISentien } @Override - public boolean spawnSentientEntityOnDrop(ItemStack droppedStack, EntityPlayer player) { + public boolean spawnSentientEntityOnDrop(ItemStack droppedStack, EntityPlayer player) + { World world = player.getEntityWorld(); - if (!world.isRemote) { + if (!world.isRemote) + { this.recalculatePowers(droppedStack, world, player); EnumDemonWillType type = this.getCurrentType(droppedStack); double soulsRemaining = PlayerDemonWillHandler.getTotalDemonWill(type, player); - if (soulsRemaining < 1024) { + if (soulsRemaining < 1024) + { return false; } From b7adad76e7e84fdc47f26414eb1c37a1ce45ff59 Mon Sep 17 00:00:00 2001 From: WayofTime Date: Fri, 30 Mar 2018 07:15:38 -0400 Subject: [PATCH 2/6] Added in book entries for the Teleport Array and the Turret Array. --- changelog.txt | 3 ++- .../AlchemyArrayEffectTeleport.java | 12 ++--------- .../compat/guideapi/book/CategoryAlchemy.java | 20 +++++++++++++++++++ .../bloodmagic/registry/ModRecipes.java | 2 +- .../assets/bloodmagicguide/lang/en_US.lang | 8 +++++++- 5 files changed, 32 insertions(+), 13 deletions(-) diff --git a/changelog.txt b/changelog.txt index 4716494f..c7161c82 100644 --- a/changelog.txt +++ b/changelog.txt @@ -5,10 +5,11 @@ Version 2.2.8 - It's a bright idea to fix this as soon as I can. - Changed the recipe of the Teleport Array: - Note from Scotty: Captain, I'll remind ya what happened last time you put an apple in her array! Use an Enderpearl and redstone dust next time! -- Added a new (slightly WIP?) array, the Turret Array: +- Added a new array, the Turret Array: - Place an array on top of an inventory with arrows and then place a bow and an arrow in the array. The array will target enemies greater than 3 blocks away and less than 32, using any arrows in the inventory. - Increased the max number of items transferable by the Master Routing Node in its system to 64 per second. Will revisit this limit if I figure out a less silly upgrade system. - Added additional effects to the Sentient Bow when aspected to different Will types. +- Added in book entries for the Teleport Array and the Turret Array. ------------------------------------------------------ Version 2.2.7 diff --git a/src/main/java/WayofTime/bloodmagic/alchemyArray/AlchemyArrayEffectTeleport.java b/src/main/java/WayofTime/bloodmagic/alchemyArray/AlchemyArrayEffectTeleport.java index 47d0e935..9437e86b 100644 --- a/src/main/java/WayofTime/bloodmagic/alchemyArray/AlchemyArrayEffectTeleport.java +++ b/src/main/java/WayofTime/bloodmagic/alchemyArray/AlchemyArrayEffectTeleport.java @@ -8,7 +8,6 @@ import net.minecraft.entity.player.EntityPlayerMP; import net.minecraft.init.SoundEvents; import net.minecraft.nbt.NBTTagCompound; import net.minecraft.network.play.server.SPacketUpdateHealth; -import net.minecraft.potion.Potion; import net.minecraft.tileentity.TileEntity; import net.minecraft.util.EnumFacing; import net.minecraft.util.SoundCategory; @@ -37,15 +36,9 @@ public class AlchemyArrayEffectTeleport extends AlchemyArrayEffect @Override public void onEntityCollidedWithBlock(IAlchemyArray array, World world, BlockPos pos, IBlockState state, Entity entity) { + EnumFacing direction = array.getRotation(); - { - double motionY = 0.5; - double speed = 3; - EnumFacing direction = array.getRotation(); - - teleportEntityInDirection(world, pos, entity, direction); - } - + teleportEntityInDirection(world, pos, entity, direction); } public void teleportEntityInDirection(World world, BlockPos currentPos, Entity entity, EnumFacing direction) @@ -66,7 +59,6 @@ public class AlchemyArrayEffectTeleport extends AlchemyArrayEffect entity.timeUntilPortal = TELEPORT_DELAY; if (!world.isRemote) { - Potion d; if (entity instanceof EntityPlayer) { EntityPlayerMP player = (EntityPlayerMP) entity; diff --git a/src/main/java/WayofTime/bloodmagic/compat/guideapi/book/CategoryAlchemy.java b/src/main/java/WayofTime/bloodmagic/compat/guideapi/book/CategoryAlchemy.java index a167ea9b..bbf67db9 100644 --- a/src/main/java/WayofTime/bloodmagic/compat/guideapi/book/CategoryAlchemy.java +++ b/src/main/java/WayofTime/bloodmagic/compat/guideapi/book/CategoryAlchemy.java @@ -93,6 +93,26 @@ public class CategoryAlchemy bouncePages.addAll(PageHelper.pagesForLongText(TextHelper.localize(keyBase + "bounce" + ".info"), 370)); entries.put(new ResourceLocation(keyBase + "bounce"), new EntryText(bouncePages, TextHelper.localize(keyBase + "bounce"), true)); + List teleportPages = new ArrayList<>(); + + PageAlchemyArray teleportRecipePage = BookUtils.getAlchemyPage("teleport"); + if (teleportRecipePage != null) + { + teleportPages.add(teleportRecipePage); + } + teleportPages.addAll(PageHelper.pagesForLongText(TextHelper.localize(keyBase + "teleport" + ".info"), 370)); + entries.put(new ResourceLocation(keyBase + "teleport"), new EntryText(teleportPages, TextHelper.localize(keyBase + "teleport"), true)); + + List standardTurretPages = new ArrayList<>(); + + PageAlchemyArray standardTurretRecipePage = BookUtils.getAlchemyPage("turret"); + if (standardTurretRecipePage != null) + { + standardTurretPages.add(standardTurretRecipePage); + } + standardTurretPages.addAll(PageHelper.pagesForLongText(TextHelper.localize(keyBase + "standardTurret" + ".info"), 370)); + entries.put(new ResourceLocation(keyBase + "standardTurret"), new EntryText(standardTurretPages, TextHelper.localize(keyBase + "standardTurret"), true)); + List buffPages = new ArrayList<>(); buffPages.addAll(PageHelper.pagesForLongText(TextHelper.localize(keyBase + "buff" + ".info"), 370)); diff --git a/src/main/java/WayofTime/bloodmagic/registry/ModRecipes.java b/src/main/java/WayofTime/bloodmagic/registry/ModRecipes.java index f8375a0e..fdebf48d 100644 --- a/src/main/java/WayofTime/bloodmagic/registry/ModRecipes.java +++ b/src/main/java/WayofTime/bloodmagic/registry/ModRecipes.java @@ -122,7 +122,7 @@ public class ModRecipes AlchemyArrayRecipeRegistry.registerRecipe(new ItemStack(Items.ARROW), new ItemStack(Items.FEATHER), new AlchemyArrayEffectSkeletonTurret("skeletonTurret"), new DualAlchemyCircleRenderer(new ResourceLocation("bloodmagic", "textures/models/AlchemyArrays/SkeletonTurret1.png"), new ResourceLocation("bloodmagic", "textures/models/AlchemyArrays/SkeletonTurret2.png"))); AlchemyArrayRecipeRegistry.registerRecipe(new ItemStack(Items.ENDER_PEARL), new ItemStack(Items.REDSTONE), new AlchemyArrayEffectTeleport("teleport"), new StaticAlchemyCircleRenderer(new ResourceLocation("bloodmagic", "textures/models/AlchemyArrays/teleportation.png"))); - AlchemyArrayRecipeRegistry.registerRecipe(new ItemStack(Items.BOW), new ItemStack(Items.ARROW), new AlchemyArrayEffectArrowTurret("turretTest"), new TurretAlchemyCircleRenderer(new ResourceLocation("bloodmagic", "textures/models/AlchemyArrays/SkeletonTurret1.png"))); + AlchemyArrayRecipeRegistry.registerRecipe(new ItemStack(Items.BOW), new ItemStack(Items.ARROW), new AlchemyArrayEffectArrowTurret("turret"), new TurretAlchemyCircleRenderer(new ResourceLocation("bloodmagic", "textures/models/AlchemyArrays/SkeletonTurret1.png"))); // AlchemyArrayRecipeRegistry.registerRecipe(new ItemStack(Items.APPLE), new ItemStack(Items.REDSTONE), new AlchemyArrayEffectLaputa("laputa"), new AttractorAlchemyCircleRenderer(new ResourceLocation("bloodmagic", "textures/models/AlchemyArrays/shardoflaputa.png"))); AlchemyArrayRecipeRegistry.registerRecipe(ComponentTypes.REAGENT_FAST_MINER.getStack(), new ItemStack(Items.IRON_PICKAXE), new AlchemyArrayEffectSigil("fastMiner", (ISigil) RegistrarBloodMagicItems.SIGIL_FAST_MINER), new SingleAlchemyCircleRenderer(new ResourceLocation("bloodmagic", "textures/models/AlchemyArrays/FastMinerSigil.png"))); diff --git a/src/main/resources/assets/bloodmagicguide/lang/en_US.lang b/src/main/resources/assets/bloodmagicguide/lang/en_US.lang index 1ba63f89..8f5d49b9 100644 --- a/src/main/resources/assets/bloodmagicguide/lang/en_US.lang +++ b/src/main/resources/assets/bloodmagicguide/lang/en_US.lang @@ -244,7 +244,8 @@ guide.bloodmagic.entry.alchemy.turret=Skeleton Turret Array guide.bloodmagic.entry.alchemy.buff=Buff Arrays guide.bloodmagic.entry.alchemy.fastMiner=Fast Miner Array guide.bloodmagic.entry.alchemy.furnace=Burning Furnace Array - +guide.bloodmagic.entry.alchemy.teleport=Teleportation Array +guide.bloodmagic.entry.alchemy.standardTurret=Turret Array @@ -258,3 +259,8 @@ guide.bloodmagic.entry.alchemy.turret.info=By utilizing the Demon Will that is s guide.bloodmagic.entry.alchemy.buff.info=Through rigorous study, you realize that alchemical arrays have a wide range of applications. So far, you have managed to create powerful items through crafting arrays, as well as a couple of arrays that provide functional benefits such as rapid movement and weak forms of teleportation. However, one of the uses that you have yet to tap into is providing powerful buffs through an active array.\n\t"Buff Arrays" are the general term for alchemy arrays that provide some buff to players within its area of effect. These areas of effect tends to have a very large radius which cannot be manipulated (unlike Rituals). However, because alchemy arrays have no concept of a Soul Network, they have to power their effects through other means: mainly through direct blood offerings. In simplified terms, this means that whenever the array will apply a buff, it will damage (take HP) from the player that it is cast on. \n\tBecause of the direct nature of these sacrifices, the HP -> buff conversion is a lot more favourable for players in the early game compared to similar buffs in the mod. For instance, if a buff from a sigil cost 100 LP for 10 seconds of activation, a more powerful buff can be applied by the array for 30 seconds for 1 HP (which is 100 LP in a T1 altar with no runes). This can be seen as a lot more efficient in the early tiers, while during the later tiers it is not as efficient of an effect. However, because of the stationary nature of the arrays, they will tend to provide a stronger effect than their sigil counter parts, so people may wish to still use it in the late-game. guide.bloodmagic.entry.alchemy.fastMiner.info=When tasked with carving out a large area of land, sometimes it is best to just do it by hand. For those occasions, this array is for you. The array applies a Haste III buff to players within a 10 block radius, costing the user 1 HP per 30 seconds. Because it is a general Haste buff, it will also increase your attack speed while in its area of effect, though you will have to be careful since other players will benefit from this too! guide.bloodmagic.entry.alchemy.furnace.info=One of the many problems that you may encounter in the beginning of your adventure is the inability to keep your furnace lit. A lit furnace can mean the difference between having a full stomach and strong weapons or starving in a cave. \n\tThe Burning Furnace array, as the name implies, will provide a much needed heat source for any nearby furnace. By placing the array directly adjacent to a vanilla furnace (it can be next to multiple), it will provide fuel to the furnace if an operation is able to be completed - nothing will happen if it is next to an empty or a full furnace, mainly for your protection. \n\tThis does not come for free, however: when a person is nearby (within a 10 block radius), it will take away half a heart of health in order to cook up to two things in the furnace. This will be helpful to either get a quick bite or to smelt a full stack of ore, but unfortunately you haven't found a way to add any safety measures to the array... +guide.bloodmagic.entry.alchemy.teleport.info=The Teleportation Array acts as a way to travel instantly from one location to another with a few specific limitations. When a player or other entity steps onto the array, the array will search up to 20 blocks away in the direction it is facing for another alchemy array (which does not need to be active). If the array manages to successfully find a destination, it will then teleport the entity to its found array instantly, even through walls. \n\tStudying this array has shown that there are further limitations added to it: because of the nature of bending the fabric of space-time, a Teleportation Array will not teleport something that has teleported within 2 seconds. This is to allow time for all components to rearrange themselves in a desirable manner. +guide.bloodmagic.entry.alchemy.standardTurret.info=The power of flinging pointy objects at far away monsters cannot be overstated. The Turret Array is able to sense a nearby hostile monster and by utilizing complex alchemical mechanisms it is able to draw back and fire an arrow in order to strike its target. \n\tThe array searches for an inventory directly beneath it. If it finds either a normal or tipped arrow it will syphon from its container and fire at a mob that is within a 32 block radius. \n\t(Due to some silly Minecraft physics that has arrows bounce off of entities that are too close to where they spawn, the turret will also only fire on a mob that is greater than 3 blocks away. Keep that in mind!) + + + From d2a84c0e7c9049870042bf3f4136b21fa0e37d22 Mon Sep 17 00:00:00 2001 From: WayofTime Date: Fri, 30 Mar 2018 11:35:40 -0400 Subject: [PATCH 3/6] Added in the Spike Array --- changelog.txt | 7 ++- .../alchemyArray/AlchemyArrayEffectSpike.java | 52 ++++++++++++++++++ .../LowStaticAlchemyCircleRenderer.java | 29 ++++++++++ .../bloodmagic/registry/ModRecipes.java | 5 +- .../models/alchemyarrays/spikearray.png | Bin 0 -> 47801 bytes 5 files changed, 90 insertions(+), 3 deletions(-) create mode 100644 src/main/java/WayofTime/bloodmagic/alchemyArray/AlchemyArrayEffectSpike.java create mode 100644 src/main/java/WayofTime/bloodmagic/client/render/alchemyArray/LowStaticAlchemyCircleRenderer.java create mode 100644 src/main/resources/assets/bloodmagic/textures/models/alchemyarrays/spikearray.png diff --git a/changelog.txt b/changelog.txt index c7161c82..f385a646 100644 --- a/changelog.txt +++ b/changelog.txt @@ -5,8 +5,11 @@ Version 2.2.8 - It's a bright idea to fix this as soon as I can. - Changed the recipe of the Teleport Array: - Note from Scotty: Captain, I'll remind ya what happened last time you put an apple in her array! Use an Enderpearl and redstone dust next time! -- Added a new array, the Turret Array: - - Place an array on top of an inventory with arrows and then place a bow and an arrow in the array. The array will target enemies greater than 3 blocks away and less than 32, using any arrows in the inventory. +- Added new arrays + - The Turret Array: + > Place an array on top of an inventory with arrows and then place a bow and an arrow in the array. The array will target enemies greater than 3 blocks away and less than 32, using any arrows in the inventory. + - Spike Array: + > Place a piece of cobblestone and iron ingot in the array. The array deals damage to any living entity that enters - Increased the max number of items transferable by the Master Routing Node in its system to 64 per second. Will revisit this limit if I figure out a less silly upgrade system. - Added additional effects to the Sentient Bow when aspected to different Will types. - Added in book entries for the Teleport Array and the Turret Array. diff --git a/src/main/java/WayofTime/bloodmagic/alchemyArray/AlchemyArrayEffectSpike.java b/src/main/java/WayofTime/bloodmagic/alchemyArray/AlchemyArrayEffectSpike.java new file mode 100644 index 00000000..a0bbf514 --- /dev/null +++ b/src/main/java/WayofTime/bloodmagic/alchemyArray/AlchemyArrayEffectSpike.java @@ -0,0 +1,52 @@ +package WayofTime.bloodmagic.alchemyArray; + +import net.minecraft.block.state.IBlockState; +import net.minecraft.entity.Entity; +import net.minecraft.entity.EntityLivingBase; +import net.minecraft.nbt.NBTTagCompound; +import net.minecraft.tileentity.TileEntity; +import net.minecraft.util.DamageSource; +import net.minecraft.util.math.BlockPos; +import net.minecraft.world.World; +import WayofTime.bloodmagic.iface.IAlchemyArray; + +public class AlchemyArrayEffectSpike extends AlchemyArrayEffect +{ + public AlchemyArrayEffectSpike(String key) + { + super(key); + } + + @Override + public boolean update(TileEntity tile, int ticksActive) + { + return false; + } + + @Override + public void onEntityCollidedWithBlock(IAlchemyArray array, World world, BlockPos pos, IBlockState state, Entity entity) + { + if (entity instanceof EntityLivingBase) + { + entity.attackEntityFrom(DamageSource.CACTUS, 2); + } + } + + @Override + public void writeToNBT(NBTTagCompound tag) + { + + } + + @Override + public void readFromNBT(NBTTagCompound tag) + { + + } + + @Override + public AlchemyArrayEffect getNewCopy() + { + return new AlchemyArrayEffectSpike(key); + } +} \ No newline at end of file diff --git a/src/main/java/WayofTime/bloodmagic/client/render/alchemyArray/LowStaticAlchemyCircleRenderer.java b/src/main/java/WayofTime/bloodmagic/client/render/alchemyArray/LowStaticAlchemyCircleRenderer.java new file mode 100644 index 00000000..0256ec93 --- /dev/null +++ b/src/main/java/WayofTime/bloodmagic/client/render/alchemyArray/LowStaticAlchemyCircleRenderer.java @@ -0,0 +1,29 @@ +package WayofTime.bloodmagic.client.render.alchemyArray; + +import net.minecraft.util.ResourceLocation; + +public class LowStaticAlchemyCircleRenderer extends LowAlchemyCircleRenderer +{ + public LowStaticAlchemyCircleRenderer() + { + this(new ResourceLocation("bloodmagic", "textures/models/AlchemyArrays/SkeletonTurret1.png")); + } + + public LowStaticAlchemyCircleRenderer(ResourceLocation arrayResource) + { + super(arrayResource); + } + + @Override + public float getRotation(float craftTime) + { + float offset = 2; + float duration = 180; + if (craftTime >= offset && craftTime < offset + duration) + { + float modifier = (craftTime - offset) * 2f; + return modifier * 1f; + } + return 0; + } +} diff --git a/src/main/java/WayofTime/bloodmagic/registry/ModRecipes.java b/src/main/java/WayofTime/bloodmagic/registry/ModRecipes.java index fdebf48d..2e2930e8 100644 --- a/src/main/java/WayofTime/bloodmagic/registry/ModRecipes.java +++ b/src/main/java/WayofTime/bloodmagic/registry/ModRecipes.java @@ -6,7 +6,6 @@ import java.util.List; import java.util.Map; import java.util.Map.Entry; -import WayofTime.bloodmagic.util.BMLog; import net.minecraft.init.Blocks; import net.minecraft.init.Items; import net.minecraft.init.MobEffects; @@ -28,12 +27,14 @@ import WayofTime.bloodmagic.alchemyArray.AlchemyArrayEffectFurnaceFuel; import WayofTime.bloodmagic.alchemyArray.AlchemyArrayEffectMovement; import WayofTime.bloodmagic.alchemyArray.AlchemyArrayEffectSigil; import WayofTime.bloodmagic.alchemyArray.AlchemyArrayEffectSkeletonTurret; +import WayofTime.bloodmagic.alchemyArray.AlchemyArrayEffectSpike; import WayofTime.bloodmagic.alchemyArray.AlchemyArrayEffectTeleport; import WayofTime.bloodmagic.alchemyArray.AlchemyArrayEffectUpdraft; import WayofTime.bloodmagic.client.render.alchemyArray.AttractorAlchemyCircleRenderer; import WayofTime.bloodmagic.client.render.alchemyArray.BindingAlchemyCircleRenderer; import WayofTime.bloodmagic.client.render.alchemyArray.DualAlchemyCircleRenderer; import WayofTime.bloodmagic.client.render.alchemyArray.LowAlchemyCircleRenderer; +import WayofTime.bloodmagic.client.render.alchemyArray.LowStaticAlchemyCircleRenderer; import WayofTime.bloodmagic.client.render.alchemyArray.SingleAlchemyCircleRenderer; import WayofTime.bloodmagic.client.render.alchemyArray.StaticAlchemyCircleRenderer; import WayofTime.bloodmagic.client.render.alchemyArray.TurretAlchemyCircleRenderer; @@ -61,6 +62,7 @@ import WayofTime.bloodmagic.livingArmour.downgrade.LivingArmourUpgradeStormTroop import WayofTime.bloodmagic.potion.BMPotionUtils; import WayofTime.bloodmagic.recipe.alchemyTable.AlchemyTableDyeableRecipe; import WayofTime.bloodmagic.recipe.alchemyTable.AlchemyTablePotionRecipe; +import WayofTime.bloodmagic.util.BMLog; import WayofTime.bloodmagic.util.Utils; import com.google.common.base.Stopwatch; @@ -124,6 +126,7 @@ public class ModRecipes AlchemyArrayRecipeRegistry.registerRecipe(new ItemStack(Items.ENDER_PEARL), new ItemStack(Items.REDSTONE), new AlchemyArrayEffectTeleport("teleport"), new StaticAlchemyCircleRenderer(new ResourceLocation("bloodmagic", "textures/models/AlchemyArrays/teleportation.png"))); AlchemyArrayRecipeRegistry.registerRecipe(new ItemStack(Items.BOW), new ItemStack(Items.ARROW), new AlchemyArrayEffectArrowTurret("turret"), new TurretAlchemyCircleRenderer(new ResourceLocation("bloodmagic", "textures/models/AlchemyArrays/SkeletonTurret1.png"))); // AlchemyArrayRecipeRegistry.registerRecipe(new ItemStack(Items.APPLE), new ItemStack(Items.REDSTONE), new AlchemyArrayEffectLaputa("laputa"), new AttractorAlchemyCircleRenderer(new ResourceLocation("bloodmagic", "textures/models/AlchemyArrays/shardoflaputa.png"))); + AlchemyArrayRecipeRegistry.registerRecipe(new ItemStack(Blocks.COBBLESTONE), new ItemStack(Items.IRON_INGOT), new AlchemyArrayEffectSpike("spike"), new LowStaticAlchemyCircleRenderer(new ResourceLocation("bloodmagic", "textures/models/AlchemyArrays/spikearray.png"))); AlchemyArrayRecipeRegistry.registerRecipe(ComponentTypes.REAGENT_FAST_MINER.getStack(), new ItemStack(Items.IRON_PICKAXE), new AlchemyArrayEffectSigil("fastMiner", (ISigil) RegistrarBloodMagicItems.SIGIL_FAST_MINER), new SingleAlchemyCircleRenderer(new ResourceLocation("bloodmagic", "textures/models/AlchemyArrays/FastMinerSigil.png"))); diff --git a/src/main/resources/assets/bloodmagic/textures/models/alchemyarrays/spikearray.png b/src/main/resources/assets/bloodmagic/textures/models/alchemyarrays/spikearray.png new file mode 100644 index 0000000000000000000000000000000000000000..7e312f51f067e9fbe9ca93b1112db801edc766ca GIT binary patch literal 47801 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 Date: Fri, 30 Mar 2018 18:01:23 -0400 Subject: [PATCH 4/6] Fixed the Haste sigil and "Quick Feet" so that they work with MC's new movement method. Also added better looking potion effects for a few things Enabled Shard of Laputa array - no book entry yet. --- .../bloodmagic/core/RegistrarBloodMagic.java | 32 ++-- .../bloodmagic/registry/ModRecipes.java | 3 +- .../handler/event/LivingArmourHandler.java | 171 ++++++++++++------ .../bloodmagic/textures/misc/potions.png | Bin 1577 -> 2109 bytes 4 files changed, 135 insertions(+), 71 deletions(-) diff --git a/src/main/java/WayofTime/bloodmagic/core/RegistrarBloodMagic.java b/src/main/java/WayofTime/bloodmagic/core/RegistrarBloodMagic.java index e9287caf..9b0c4e9a 100644 --- a/src/main/java/WayofTime/bloodmagic/core/RegistrarBloodMagic.java +++ b/src/main/java/WayofTime/bloodmagic/core/RegistrarBloodMagic.java @@ -28,7 +28,8 @@ import net.minecraftforge.registries.RegistryBuilder; @Mod.EventBusSubscriber(modid = BloodMagic.MODID) @GameRegistry.ObjectHolder(BloodMagic.MODID) -public class RegistrarBloodMagic { +public class RegistrarBloodMagic +{ private static final BloodOrb ORB_DEF = new BloodOrb("", 0, 0, 0); @GameRegistry.ObjectHolder("weak") @@ -60,7 +61,8 @@ public class RegistrarBloodMagic { public static IForgeRegistry BLOOD_ORBS = null; @SubscribeEvent - public static void registerBloodOrbs(RegistryEvent.Register event) { + public static void registerBloodOrbs(RegistryEvent.Register event) + { ResourceLocation orb = RegistrarBloodMagicItems.BLOOD_ORB.getRegistryName(); event.getRegistry().registerAll( new BloodOrb("weak", 1, 5000, 2).withModel(new ModelResourceLocation(orb, "type=weak")).setRegistryName("weak"), @@ -69,13 +71,14 @@ public class RegistrarBloodMagic { new BloodOrb("master", 4, 1000000, 25).withModel(new ModelResourceLocation(orb, "type=master")).setRegistryName("master"), new BloodOrb("archmage", 5, 10000000, 50).withModel(new ModelResourceLocation(orb, "type=archmage")).setRegistryName("archmage"), new BloodOrb("transcendent", 6, 30000000, 50).withModel(new ModelResourceLocation(orb, "type=transcendent")).setRegistryName("transcendent") - ); + ); } @SubscribeEvent - public static void registerPotions(RegistryEvent.Register event) { + public static void registerPotions(RegistryEvent.Register event) + { event.getRegistry().registerAll( - new PotionBloodMagic("Boost", false, 0xFFFFFF, 0, 1).setRegistryName("boost"), + new PotionBloodMagic("Boost", false, 0xFFFFFF, 0, 0).setRegistryName("boost"), new PotionBloodMagic("Planar Binding", false, 0, 2, 0).setRegistryName("planar_binding"), new PotionBloodMagic("Soul Snare", false, 0xFFFFFF, 3, 0).setRegistryName("soul_snare"), new PotionBloodMagic("Soul Fray", true, 0xFFFFFF, 4, 0).setRegistryName("soul_fray"), @@ -86,11 +89,12 @@ public class RegistrarBloodMagic { new PotionBloodMagic("Bounce", false, 0x000000, 1, 1).setRegistryName("bounce"), new PotionBloodMagic("Cling", false, 0x000000, 2, 1).setRegistryName("cling"), new PotionBloodMagic("S. Lamb", false, 0x000000, 3, 1).setRegistryName("sacrificial_lamb") - ); + ); } @SubscribeEvent - public static void registerEntities(RegistryEvent.Register event) { + public static void registerEntities(RegistryEvent.Register event) + { int entities = 0; event.getRegistry().registerAll( @@ -104,11 +108,12 @@ public class RegistrarBloodMagic { EntityEntryBuilder.create().id("corrupted_sheep", ++entities).entity(EntityCorruptedSheep.class).name("corrupted_sheep").tracker(16 * 4, 3, true).build(), EntityEntryBuilder.create().id("corrupted_chicken", ++entities).entity(EntityCorruptedChicken.class).name("corrupted_chicken").tracker(16 * 4, 3, true).build(), EntityEntryBuilder.create().id("corrupted_spider", ++entities).entity(EntityCorruptedSpider.class).name("corrupted_spider").tracker(16 * 4, 3, true).build() - ); + ); } @SubscribeEvent - public static void onRegistryCreation(RegistryEvent.NewRegistry event) { + public static void onRegistryCreation(RegistryEvent.NewRegistry event) + { BLOOD_ORBS = new RegistryBuilder() .setName(new ResourceLocation(BloodMagic.MODID, "blood_orb")) .setIDRange(0, Short.MAX_VALUE) @@ -119,8 +124,10 @@ public class RegistrarBloodMagic { @SideOnly(Side.CLIENT) @SubscribeEvent - public static void registerModels(ModelRegistryEvent event) { - for (BloodOrb orb : BLOOD_ORBS) { + public static void registerModels(ModelRegistryEvent event) + { + for (BloodOrb orb : BLOOD_ORBS) + { ModelResourceLocation modelLocation = orb.getModelLocation(); if (modelLocation == null) modelLocation = new ModelResourceLocation(orb.getRegistryName(), "inventory"); @@ -128,7 +135,8 @@ public class RegistrarBloodMagic { ModelLoader.registerItemVariants(RegistrarBloodMagicItems.BLOOD_ORB, modelLocation); } - ModelLoader.setCustomMeshDefinition(RegistrarBloodMagicItems.BLOOD_ORB, stack -> { + ModelLoader.setCustomMeshDefinition(RegistrarBloodMagicItems.BLOOD_ORB, stack -> + { if (!stack.hasTagCompound()) return new ModelResourceLocation(ORB_WEAK.getRegistryName(), "inventory"); diff --git a/src/main/java/WayofTime/bloodmagic/registry/ModRecipes.java b/src/main/java/WayofTime/bloodmagic/registry/ModRecipes.java index 2e2930e8..d84c7548 100644 --- a/src/main/java/WayofTime/bloodmagic/registry/ModRecipes.java +++ b/src/main/java/WayofTime/bloodmagic/registry/ModRecipes.java @@ -24,6 +24,7 @@ import WayofTime.bloodmagic.alchemyArray.AlchemyArrayEffectAttractor; import WayofTime.bloodmagic.alchemyArray.AlchemyArrayEffectBinding; import WayofTime.bloodmagic.alchemyArray.AlchemyArrayEffectBounce; import WayofTime.bloodmagic.alchemyArray.AlchemyArrayEffectFurnaceFuel; +import WayofTime.bloodmagic.alchemyArray.AlchemyArrayEffectLaputa; import WayofTime.bloodmagic.alchemyArray.AlchemyArrayEffectMovement; import WayofTime.bloodmagic.alchemyArray.AlchemyArrayEffectSigil; import WayofTime.bloodmagic.alchemyArray.AlchemyArrayEffectSkeletonTurret; @@ -125,7 +126,7 @@ public class ModRecipes AlchemyArrayRecipeRegistry.registerRecipe(new ItemStack(Items.ENDER_PEARL), new ItemStack(Items.REDSTONE), new AlchemyArrayEffectTeleport("teleport"), new StaticAlchemyCircleRenderer(new ResourceLocation("bloodmagic", "textures/models/AlchemyArrays/teleportation.png"))); AlchemyArrayRecipeRegistry.registerRecipe(new ItemStack(Items.BOW), new ItemStack(Items.ARROW), new AlchemyArrayEffectArrowTurret("turret"), new TurretAlchemyCircleRenderer(new ResourceLocation("bloodmagic", "textures/models/AlchemyArrays/SkeletonTurret1.png"))); -// AlchemyArrayRecipeRegistry.registerRecipe(new ItemStack(Items.APPLE), new ItemStack(Items.REDSTONE), new AlchemyArrayEffectLaputa("laputa"), new AttractorAlchemyCircleRenderer(new ResourceLocation("bloodmagic", "textures/models/AlchemyArrays/shardoflaputa.png"))); + AlchemyArrayRecipeRegistry.registerRecipe(new ItemStack(Items.REDSTONE), new ItemStack(Blocks.LAPIS_BLOCK), new AlchemyArrayEffectLaputa("laputa"), new AttractorAlchemyCircleRenderer(new ResourceLocation("bloodmagic", "textures/models/AlchemyArrays/shardoflaputa.png"))); AlchemyArrayRecipeRegistry.registerRecipe(new ItemStack(Blocks.COBBLESTONE), new ItemStack(Items.IRON_INGOT), new AlchemyArrayEffectSpike("spike"), new LowStaticAlchemyCircleRenderer(new ResourceLocation("bloodmagic", "textures/models/AlchemyArrays/spikearray.png"))); AlchemyArrayRecipeRegistry.registerRecipe(ComponentTypes.REAGENT_FAST_MINER.getStack(), new ItemStack(Items.IRON_PICKAXE), new AlchemyArrayEffectSigil("fastMiner", (ISigil) RegistrarBloodMagicItems.SIGIL_FAST_MINER), new SingleAlchemyCircleRenderer(new ResourceLocation("bloodmagic", "textures/models/AlchemyArrays/FastMinerSigil.png"))); diff --git a/src/main/java/WayofTime/bloodmagic/util/handler/event/LivingArmourHandler.java b/src/main/java/WayofTime/bloodmagic/util/handler/event/LivingArmourHandler.java index f78a7e59..456482c3 100644 --- a/src/main/java/WayofTime/bloodmagic/util/handler/event/LivingArmourHandler.java +++ b/src/main/java/WayofTime/bloodmagic/util/handler/event/LivingArmourHandler.java @@ -40,24 +40,31 @@ import net.minecraftforge.fml.common.eventhandler.EventPriority; import net.minecraftforge.fml.common.eventhandler.SubscribeEvent; @Mod.EventBusSubscriber(modid = BloodMagic.MODID) -public class LivingArmourHandler { +public class LivingArmourHandler +{ @SubscribeEvent - public static void onEntityHealed(LivingHealEvent event) { - if (event.getEntityLiving() instanceof EntityPlayer) { + public static void onEntityHealed(LivingHealEvent event) + { + if (event.getEntityLiving() instanceof EntityPlayer) + { EntityPlayer player = (EntityPlayer) event.getEntity(); - if (LivingArmour.hasFullSet(player)) { + if (LivingArmour.hasFullSet(player)) + { ItemStack chestStack = player.getItemStackFromSlot(EntityEquipmentSlot.CHEST); LivingArmour armour = ItemLivingArmour.getLivingArmour(chestStack); - if (armour != null) { + if (armour != null) + { double modifier = 1; LivingArmourUpgrade upgrade = ItemLivingArmour.getUpgrade(BloodMagic.MODID + ".upgrade.slowHeal", chestStack); - if (upgrade instanceof LivingArmourUpgradeSlowHeal) { + if (upgrade instanceof LivingArmourUpgradeSlowHeal) + { modifier *= ((LivingArmourUpgradeSlowHeal) upgrade).getHealingModifier(); } - if (modifier != 1) { + if (modifier != 1) + { event.setAmount((float) (event.getAmount() * modifier)); } } @@ -66,18 +73,23 @@ public class LivingArmourHandler { } @SubscribeEvent - public static void onMiningSpeedCheck(PlayerEvent.BreakSpeed event) { + public static void onMiningSpeedCheck(PlayerEvent.BreakSpeed event) + { EntityPlayer player = event.getEntityPlayer(); - if (LivingArmour.hasFullSet(player)) { + if (LivingArmour.hasFullSet(player)) + { ItemStack chestStack = player.getItemStackFromSlot(EntityEquipmentSlot.CHEST); LivingArmour armour = ItemLivingArmour.getLivingArmour(chestStack); - if (armour != null) { + if (armour != null) + { double modifier = 1; - for (LivingArmourUpgrade upgrade : armour.upgradeMap.values()) { + for (LivingArmourUpgrade upgrade : armour.upgradeMap.values()) + { modifier *= upgrade.getMiningSpeedModifier(player); } - if (modifier != 1) { + if (modifier != 1) + { event.setNewSpeed((float) (event.getOriginalSpeed() * modifier)); } } @@ -86,24 +98,31 @@ public class LivingArmourHandler { // Applies: Storm Trooper @SubscribeEvent - public static void onEntityJoinedWorld(EntityJoinWorldEvent event) { + public static void onEntityJoinedWorld(EntityJoinWorldEvent event) + { Entity owner = null; - if (event.getEntity() instanceof EntityArrow) { + if (event.getEntity() instanceof EntityArrow) + { owner = ((EntityArrow) event.getEntity()).shootingEntity; - } else if (event.getEntity() instanceof EntityThrowable) { + } else if (event.getEntity() instanceof EntityThrowable) + { owner = ((EntityThrowable) event.getEntity()).getThrower(); } - if (owner instanceof EntityPlayer) { + if (owner instanceof EntityPlayer) + { Entity projectile = event.getEntity(); EntityPlayer player = (EntityPlayer) owner; - if (LivingArmour.hasFullSet(player)) { + if (LivingArmour.hasFullSet(player)) + { ItemStack chestStack = player.getItemStackFromSlot(EntityEquipmentSlot.CHEST); LivingArmour armour = ItemLivingArmour.getLivingArmour(chestStack); - if (armour != null) { + if (armour != null) + { LivingArmourUpgrade upgrade = ItemLivingArmour.getUpgrade(BloodMagic.MODID + ".upgrade.stormTrooper", chestStack); - if (upgrade instanceof LivingArmourUpgradeStormTrooper) { + if (upgrade instanceof LivingArmourUpgradeStormTrooper) + { float velocityModifier = (float) (((LivingArmourUpgradeStormTrooper) upgrade).getArrowJiggle(player) * Math.sqrt(projectile.motionX * projectile.motionX + projectile.motionY * projectile.motionY + projectile.motionZ * projectile.motionZ)); projectile.motionX += 2 * (event.getWorld().rand.nextDouble() - 0.5) * velocityModifier; @@ -116,29 +135,37 @@ public class LivingArmourHandler { } @SubscribeEvent - public static void onPlayerClick(PlayerInteractEvent event) { - if (event.isCancelable()) { + public static void onPlayerClick(PlayerInteractEvent event) + { + if (event.isCancelable()) + { EntityPlayer player = event.getEntityPlayer(); - if (LivingArmour.hasFullSet(player)) { + if (LivingArmour.hasFullSet(player)) + { ItemStack chestStack = player.getItemStackFromSlot(EntityEquipmentSlot.CHEST); LivingArmour armour = ItemLivingArmour.getLivingArmour(chestStack); - if (armour != null) { - if (event.getHand() == EnumHand.OFF_HAND) { + if (armour != null) + { + if (event.getHand() == EnumHand.OFF_HAND) + { LivingArmourUpgrade upgrade = ItemLivingArmour.getUpgrade(BloodMagic.MODID + ".upgrade.crippledArm", chestStack); - if (upgrade instanceof LivingArmourUpgradeCrippledArm) { + if (upgrade instanceof LivingArmourUpgradeCrippledArm) + { event.setCanceled(true); } } - if (event.getItemStack().getItemUseAction() == EnumAction.DRINK) { + if (event.getItemStack().getItemUseAction() == EnumAction.DRINK) + { ItemStack drinkStack = event.getItemStack(); //TODO: See if the item is a splash potion? Those should be usable. LivingArmourUpgrade upgrade = ItemLivingArmour.getUpgrade(BloodMagic.MODID + ".upgrade.quenched", chestStack); - if (upgrade instanceof LivingArmourUpgradeQuenched) { + if (upgrade instanceof LivingArmourUpgradeQuenched) + { event.setCanceled(true); } } @@ -149,19 +176,24 @@ public class LivingArmourHandler { // Applies: Grim Reaper @SubscribeEvent(priority = EventPriority.HIGHEST) - public static void onEntityDeath(LivingDeathEvent event) { - if (event.getEntityLiving() instanceof EntityPlayer) { + public static void onEntityDeath(LivingDeathEvent event) + { + if (event.getEntityLiving() instanceof EntityPlayer) + { EntityPlayer player = (EntityPlayer) event.getEntityLiving(); - if (LivingArmour.hasFullSet(player)) { + if (LivingArmour.hasFullSet(player)) + { ItemStack chestStack = player.getItemStackFromSlot(EntityEquipmentSlot.CHEST); LivingArmour armour = ItemLivingArmour.getLivingArmour(chestStack); - if (armour != null) { + if (armour != null) + { StatTrackerGrimReaperSprint.incrementCounter(armour); LivingArmourUpgrade upgrade = ItemLivingArmour.getUpgrade(BloodMagic.MODID + ".upgrade.grimReaper", chestStack); - if (upgrade instanceof LivingArmourUpgradeGrimReaperSprint && ((LivingArmourUpgradeGrimReaperSprint) upgrade).canSavePlayer(player)) { + if (upgrade instanceof LivingArmourUpgradeGrimReaperSprint && ((LivingArmourUpgradeGrimReaperSprint) upgrade).canSavePlayer(player)) + { ((LivingArmourUpgradeGrimReaperSprint) upgrade).applyEffectOnRebirth(player); event.setCanceled(true); event.setResult(Event.Result.DENY); @@ -175,20 +207,26 @@ public class LivingArmourHandler { // Applies: Jump @SubscribeEvent - public static void onJumpEvent(LivingEvent.LivingJumpEvent event) { - if (event.getEntityLiving() instanceof EntityPlayer) { + public static void onJumpEvent(LivingEvent.LivingJumpEvent event) + { + if (event.getEntityLiving() instanceof EntityPlayer) + { EntityPlayer player = (EntityPlayer) event.getEntityLiving(); - if (LivingArmour.hasFullSet(player)) { + if (LivingArmour.hasFullSet(player)) + { ItemStack chestStack = player.getItemStackFromSlot(EntityEquipmentSlot.CHEST); LivingArmour armour = ItemLivingArmour.getLivingArmour(chestStack); - if (armour != null) { + if (armour != null) + { StatTrackerJump.incrementCounter(armour); - if (!player.isSneaking()) { + if (!player.isSneaking()) + { LivingArmourUpgrade upgrade = ItemLivingArmour.getUpgradeFromNBT(BloodMagic.MODID + ".upgrade.jump", chestStack); - if (upgrade instanceof LivingArmourUpgradeJump) { + if (upgrade instanceof LivingArmourUpgradeJump) + { player.motionY += ((LivingArmourUpgradeJump) upgrade).getJumpModifier(); } } @@ -199,21 +237,28 @@ public class LivingArmourHandler { // Applies: Step Assist, Speed Boost @SubscribeEvent(priority = EventPriority.HIGHEST) - public static void onEntityUpdate(LivingEvent.LivingUpdateEvent event) { - if (event.getEntityLiving() instanceof EntityPlayer) { + public static void onEntityUpdate(LivingEvent.LivingUpdateEvent event) + { + if (event.getEntityLiving() instanceof EntityPlayer) + { EntityPlayer player = (EntityPlayer) event.getEntityLiving(); boolean hasAssist = false; - if (event.getEntityLiving().isPotionActive(RegistrarBloodMagic.BOOST)) { + if (event.getEntityLiving().isPotionActive(RegistrarBloodMagic.BOOST)) + { hasAssist = true; player.stepHeight = Constants.Misc.ALTERED_STEP_HEIGHT; - } else { - if (LivingArmour.hasFullSet(player)) { + } else + { + if (LivingArmour.hasFullSet(player)) + { ItemStack chestStack = player.getItemStackFromSlot(EntityEquipmentSlot.CHEST); LivingArmour armour = ItemLivingArmour.getLivingArmourFromStack(chestStack); - if (armour != null) { + if (armour != null) + { LivingArmourUpgrade upgrade = ItemLivingArmour.getUpgrade(BloodMagic.MODID + ".upgrade.stepAssist", chestStack); - if (upgrade instanceof LivingArmourUpgradeStepAssist) { + if (upgrade instanceof LivingArmourUpgradeStepAssist) + { player.stepHeight = ((LivingArmourUpgradeStepAssist) upgrade).getStepAssist(); hasAssist = true; } @@ -226,27 +271,32 @@ public class LivingArmourHandler { float percentIncrease = 0; - if (LivingArmour.hasFullSet(player)) { + if (LivingArmour.hasFullSet(player)) + { ItemStack chestStack = player.getItemStackFromSlot(EntityEquipmentSlot.CHEST); LivingArmour armour = ItemLivingArmour.getLivingArmourFromStack(chestStack); - if (armour != null) { + if (armour != null) + { LivingArmourUpgrade upgrade = ItemLivingArmour.getUpgradeFromNBT(BloodMagic.MODID + ".upgrade.movement", chestStack); - if (upgrade instanceof LivingArmourUpgradeSpeed) { - percentIncrease += 0.1f * ((LivingArmourUpgradeSpeed) upgrade).getSpeedModifier(); + if (upgrade instanceof LivingArmourUpgradeSpeed) + { + percentIncrease += ((LivingArmourUpgradeSpeed) upgrade).getSpeedModifier(); } } } - if (event.getEntityLiving().isPotionActive(RegistrarBloodMagic.BOOST)) { + if (event.getEntityLiving().isPotionActive(RegistrarBloodMagic.BOOST)) + { int i = event.getEntityLiving().getActivePotionEffect(RegistrarBloodMagic.BOOST).getAmplifier(); { - percentIncrease += (i + 1) * 0.05f; + percentIncrease += (i + 1) * 0.5f; } } - if (percentIncrease > 0 && (player.onGround || player.capabilities.isFlying) && (Math.abs(player.moveForward) > 0 || Math.abs(player.moveStrafing) > 0)) { - player.moveRelative(player.moveStrafing, player.moveForward, player.capabilities.isFlying ? (percentIncrease / 2.0f) : percentIncrease, 0.02F); + if (percentIncrease > 0 && (player.onGround || player.capabilities.isFlying) && (Math.abs(player.moveForward) > 0 || Math.abs(player.moveStrafing) > 0)) + { + player.travel(player.moveStrafing * percentIncrease, 0, player.moveForward * percentIncrease); } } } @@ -254,7 +304,8 @@ public class LivingArmourHandler { // Applies: Arrow Shot // Tracks: Arrow Shot @SubscribeEvent - public static void onArrowFire(ArrowLooseEvent event) { + public static void onArrowFire(ArrowLooseEvent event) + { World world = event.getEntityPlayer().getEntityWorld(); ItemStack stack = event.getBow(); EntityPlayer player = event.getEntityPlayer(); @@ -262,14 +313,17 @@ public class LivingArmourHandler { if (world.isRemote) return; - if (LivingArmour.hasFullSet(player)) { + if (LivingArmour.hasFullSet(player)) + { ItemStack chestStack = player.getItemStackFromSlot(EntityEquipmentSlot.CHEST); LivingArmour armour = ItemLivingArmour.getLivingArmour(chestStack); - if (armour != null) { + if (armour != null) + { StatTrackerArrowShot.incrementCounter(armour); LivingArmourUpgrade upgrade = ItemLivingArmour.getUpgrade(BloodMagic.MODID + ".upgrade.arrowShot", chestStack); - if (upgrade instanceof LivingArmourUpgradeArrowShot) { + if (upgrade instanceof LivingArmourUpgradeArrowShot) + { int charge = event.getCharge(); float velocity = (float) charge / 20.0F; velocity = (velocity * velocity + velocity * 2.0F) / 3.0F; @@ -281,7 +335,8 @@ public class LivingArmourHandler { velocity = 1.0F; int extraArrows = ((LivingArmourUpgradeArrowShot) upgrade).getExtraArrows(); - for (int n = 0; n < extraArrows; n++) { + for (int n = 0; n < extraArrows; n++) + { ItemStack arrowStack = new ItemStack(Items.ARROW); ItemArrow itemarrow = (ItemArrow) ((stack.getItem() instanceof ItemArrow ? arrowStack.getItem() : Items.ARROW)); EntityArrow entityarrow = itemarrow.createArrow(world, arrowStack, player); diff --git a/src/main/resources/assets/bloodmagic/textures/misc/potions.png b/src/main/resources/assets/bloodmagic/textures/misc/potions.png index 49c30a6e3a54c8bedba5f62ceaf3a7727f82fd28..0ceb8f8ba5ae5f4d9983f4ce6bbe059eed5d4039 100644 GIT binary patch delta 1845 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 bcEbMvyHm}gW+aYk00000NkvXXu0mjf^VYIs From 58ca46b6bd7e7d61ab998ec43f210d165dc4f4eb Mon Sep 17 00:00:00 2001 From: WayofTime Date: Fri, 30 Mar 2018 18:03:31 -0400 Subject: [PATCH 5/6] Disabled the added Health in the "Quick Feet" upgrade. --- changelog.txt | 2 + .../upgrade/LivingArmourUpgradeSpeed.java | 62 ++++++++++++------- 2 files changed, 40 insertions(+), 24 deletions(-) diff --git a/changelog.txt b/changelog.txt index f385a646..beacb676 100644 --- a/changelog.txt +++ b/changelog.txt @@ -13,6 +13,8 @@ Version 2.2.8 - Increased the max number of items transferable by the Master Routing Node in its system to 64 per second. Will revisit this limit if I figure out a less silly upgrade system. - Added additional effects to the Sentient Bow when aspected to different Will types. - Added in book entries for the Teleport Array and the Turret Array. +- Fixed the Haste sigil and "Quick Feet" so that they work with MC's new movement method. +- Removed added health from "Quick Feet" - seriously, why was this a thing? ------------------------------------------------------ Version 2.2.7 diff --git a/src/main/java/WayofTime/bloodmagic/livingArmour/upgrade/LivingArmourUpgradeSpeed.java b/src/main/java/WayofTime/bloodmagic/livingArmour/upgrade/LivingArmourUpgradeSpeed.java index 446decfb..d0392eec 100644 --- a/src/main/java/WayofTime/bloodmagic/livingArmour/upgrade/LivingArmourUpgradeSpeed.java +++ b/src/main/java/WayofTime/bloodmagic/livingArmour/upgrade/LivingArmourUpgradeSpeed.java @@ -16,76 +16,90 @@ import org.apache.commons.codec.binary.StringUtils; import java.util.UUID; -public class LivingArmourUpgradeSpeed extends LivingArmourUpgrade { - public static final int[] costs = new int[]{3, 7, 13, 26, 42, 60, 90, 130, 180, 250}; - public static final double[] speedModifier = new double[]{0.1, 0.2, 0.3, 0.4, 0.5, 0.7, 0.9, 1.1, 1.3, 1.5}; - public static final int[] sprintSpeedTime = new int[]{0, 0, 0, 0, 0, 20, 60, 60, 100, 200}; - public static final int[] sprintSpeedLevel = new int[]{0, 0, 0, 0, 0, 0, 0, 1, 1, 2}; - public static final int[] healthModifier = new int[]{0, 0, 0, 0, 0, 0, 0, 4, 10, 20}; - public static final int[] sprintRegenTime = new int[]{0, 0, 0, 0, 0, 0, 0, 0, 0, 25}; +public class LivingArmourUpgradeSpeed extends LivingArmourUpgrade +{ + public static final int[] costs = new int[] { 3, 7, 13, 26, 42, 60, 90, 130, 180, 250 }; + public static final double[] speedModifier = new double[] { 0.1, 0.2, 0.3, 0.4, 0.5, 0.7, 0.9, 1.1, 1.3, 1.5 }; + public static final int[] sprintSpeedTime = new int[] { 0, 0, 0, 0, 0, 20, 60, 60, 100, 200 }; + public static final int[] sprintSpeedLevel = new int[] { 0, 0, 0, 0, 0, 0, 0, 1, 1, 2 }; + public static final int[] healthModifier = new int[] { 0, 0, 0, 0, 0, 0, 0, 4, 10, 20 }; + public static final int[] sprintRegenTime = new int[] { 0, 0, 0, 0, 0, 0, 0, 0, 0, 25 }; - public LivingArmourUpgradeSpeed(int level) { + public LivingArmourUpgradeSpeed(int level) + { super(level); } - public double getSpeedModifier() { + public double getSpeedModifier() + { return speedModifier[this.level]; } @Override - public void onTick(World world, EntityPlayer player, ILivingArmour livingArmour) { - if (player.isSprinting()) { - if (sprintSpeedTime[this.level] > 0) { + public void onTick(World world, EntityPlayer player, ILivingArmour livingArmour) + { + if (player.isSprinting()) + { + if (sprintSpeedTime[this.level] > 0) + { player.addPotionEffect(new PotionEffect(MobEffects.SPEED, sprintSpeedTime[this.level], sprintSpeedLevel[this.level], false, false)); } - if (sprintRegenTime[this.level] > 0 && !player.isPotionActive(MobEffects.REGENERATION)) { + if (sprintRegenTime[this.level] > 0 && !player.isPotionActive(MobEffects.REGENERATION)) + { player.addPotionEffect(new PotionEffect(MobEffects.REGENERATION, sprintRegenTime[this.level], 0, false, false)); } } } @Override - public Multimap getAttributeModifiers() { + public Multimap getAttributeModifiers() + { Multimap modifierMap = HashMultimap.create(); // modifierMap.put(SharedMonsterAttributes.movementSpeed.getAttributeUnlocalizedName(), new AttributeModifier(new UUID(895132, 1), "Speed modifier" + 1, speedModifier[this.level], 1)); - if (healthModifier[this.level] > 0) { - String name = getUniqueIdentifier() + "-HealthModifier1"; - modifierMap.put(SharedMonsterAttributes.MAX_HEALTH.getName(), new AttributeModifier(UUID.nameUUIDFromBytes(StringUtils.getBytesUtf8(name)), "HealthModifier1", healthModifier[this.level], 0)); - } +// if (healthModifier[this.level] > 0) { +// String name = getUniqueIdentifier() + "-HealthModifier1"; +// modifierMap.put(SharedMonsterAttributes.MAX_HEALTH.getName(), new AttributeModifier(UUID.nameUUIDFromBytes(StringUtils.getBytesUtf8(name)), "HealthModifier1", healthModifier[this.level], 0)); +// } return modifierMap; } @Override - public String getUniqueIdentifier() { + public String getUniqueIdentifier() + { return BloodMagic.MODID + ".upgrade.movement"; } @Override - public int getMaxTier() { + public int getMaxTier() + { return 10; } @Override - public int getCostOfUpgrade() { + public int getCostOfUpgrade() + { return costs[this.level]; } @Override - public void writeToNBT(NBTTagCompound tag) { + public void writeToNBT(NBTTagCompound tag) + { // EMPTY } @Override - public void readFromNBT(NBTTagCompound tag) { + public void readFromNBT(NBTTagCompound tag) + { // EMPTY } @Override - public String getUnlocalizedName() { + public String getUnlocalizedName() + { return tooltipBase + "speed"; } } From 467bcb4d52ae05b053a8bfa30114f82ff13b1043 Mon Sep 17 00:00:00 2001 From: WayofTime Date: Fri, 30 Mar 2018 19:42:11 -0400 Subject: [PATCH 6/6] Readded the ritual "Focus of the Ellipsoid": creates a hollow ellipsoid from blocks provided in the connecting chest. Also added the guide page for the Shard of Laputa. --- changelog.txt | 2 + .../WayofTime/bloodmagic/ConfigHandler.java | 82 +++++++++++-------- .../AlchemyArrayEffectLaputa.java | 2 +- .../compat/guideapi/book/CategoryAlchemy.java | 10 +++ .../bloodmagic/registry/ModRituals.java | 2 +- .../ritual/types/RitualEllipsoid.java | 41 +++++++++- .../assets/bloodmagic/lang/en_US.lang | 1 + .../assets/bloodmagicguide/lang/en_US.lang | 3 +- 8 files changed, 102 insertions(+), 41 deletions(-) diff --git a/changelog.txt b/changelog.txt index beacb676..f72e8573 100644 --- a/changelog.txt +++ b/changelog.txt @@ -15,6 +15,8 @@ Version 2.2.8 - Added in book entries for the Teleport Array and the Turret Array. - Fixed the Haste sigil and "Quick Feet" so that they work with MC's new movement method. - Removed added health from "Quick Feet" - seriously, why was this a thing? +- Readded the ritual "Focus of the Ellipsoid": creates a hollow ellipsoid from blocks provided in the connecting chest. + - Note: The dictionary definition for "Ellipsoid" is a three-dimensional figure whose plane sections are ellipses or circles. For those who weren't born in a math class, it means it is a sphere that has different radii in each direction. ------------------------------------------------------ Version 2.2.7 diff --git a/src/main/java/WayofTime/bloodmagic/ConfigHandler.java b/src/main/java/WayofTime/bloodmagic/ConfigHandler.java index 27316942..c0f2a892 100644 --- a/src/main/java/WayofTime/bloodmagic/ConfigHandler.java +++ b/src/main/java/WayofTime/bloodmagic/ConfigHandler.java @@ -9,61 +9,68 @@ import net.minecraftforge.fml.common.eventhandler.SubscribeEvent; @Config(modid = BloodMagic.MODID, name = BloodMagic.MODID + "/" + BloodMagic.MODID, category = "") @Mod.EventBusSubscriber(modid = BloodMagic.MODID) -public class ConfigHandler { +public class ConfigHandler +{ - @Config.Comment({"General settings"}) + @Config.Comment({ "General settings" }) public static ConfigGeneral general = new ConfigGeneral(); - @Config.Comment({"Blacklist options for various features"}) + @Config.Comment({ "Blacklist options for various features" }) public static ConfigBlacklist blacklist = new ConfigBlacklist(); - @Config.Comment({"Value modifiers for various features"}) + @Config.Comment({ "Value modifiers for various features" }) public static ConfigValues values = new ConfigValues(); - @Config.Comment({"Toggles for all rituals"}) + @Config.Comment({ "Toggles for all rituals" }) public static ConfigRituals rituals = new ConfigRituals(); - @Config.Comment({"Settings that only pertain to the client"}) + @Config.Comment({ "Settings that only pertain to the client" }) public static ConfigClient client = new ConfigClient(); - @Config.Comment({"Compatibility settings"}) + @Config.Comment({ "Compatibility settings" }) public static ConfigCompat compat = new ConfigCompat(); @SubscribeEvent - public static void onConfigChanged(ConfigChangedEvent.OnConfigChangedEvent event) { - if (event.getModID().equals(BloodMagic.MODID)) { + public static void onConfigChanged(ConfigChangedEvent.OnConfigChangedEvent event) + { + if (event.getModID().equals(BloodMagic.MODID)) + { ConfigManager.sync(event.getModID(), Config.Type.INSTANCE); // Sync config values MeteorConfigHandler.handleMeteors(false); // Reload meteors } } - public static class ConfigGeneral { - @Config.Comment({"Enables extra information to be printed to the log.", "Warning: May drastically increase log size."}) + public static class ConfigGeneral + { + @Config.Comment({ "Enables extra information to be printed to the log.", "Warning: May drastically increase log size." }) public boolean enableDebugLogging = false; - @Config.Comment({"Enables extra information to be printed to the log."}) + @Config.Comment({ "Enables extra information to be printed to the log." }) public boolean enableAPILogging = false; - @Config.Comment({"Enables extra information to be printed to the log.", "Warning: May drastically increase log size."}) + @Config.Comment({ "Enables extra information to be printed to the log.", "Warning: May drastically increase log size." }) public boolean enableVerboseAPILogging = false; } - public static class ConfigBlacklist { - @Config.Comment({"Stops listed blocks and entities from being teleposed.", "Use the registry name of the block or entity. Vanilla objects do not require the modid.", "If a block is specified, you can list the variants to only blacklist a given state."}) - public String[] teleposer = {"bedrock", "mob_spawner"}; - @Config.Comment({"Stops listed blocks from being transposed.", "Use the registry name of the block. Vanilla blocks do not require the modid."}) - public String[] transposer = {"bedrock", "mob_spawner"}; - @Config.Comment({"Stops the listed entities from being used in the Well of Suffering.", "Use the registry name of the entity. Vanilla entities do not require the modid."}) + public static class ConfigBlacklist + { + @Config.Comment({ "Stops listed blocks and entities from being teleposed.", "Use the registry name of the block or entity. Vanilla objects do not require the modid.", "If a block is specified, you can list the variants to only blacklist a given state." }) + public String[] teleposer = { "bedrock", "mob_spawner" }; + @Config.Comment({ "Stops listed blocks from being transposed.", "Use the registry name of the block. Vanilla blocks do not require the modid." }) + public String[] transposer = { "bedrock", "mob_spawner" }; + @Config.Comment({ "Stops the listed entities from being used in the Well of Suffering.", "Use the registry name of the entity. Vanilla entities do not require the modid." }) public String[] wellOfSuffering = {}; } - public static class ConfigValues { - @Config.Comment({"Declares the amount of LP gained per HP sacrificed for the given entity.", "Setting the value to 0 will blacklist it.", "Use the registry name of the entity followed by a ';' and then the value you want.", "Vanilla entities do not require the modid."}) - public String[] sacrificialValues = {"villager;100", "slime;15", "enderman;10", "cow;100", "chicken;100", "horse;100", "sheep;100", "wolf;100", "ocelot;100", "pig;100", "rabbit;100"}; - @Config.Comment({"Amount of LP the Coat of Arms should provide for each damage dealt."}) + public static class ConfigValues + { + @Config.Comment({ "Declares the amount of LP gained per HP sacrificed for the given entity.", "Setting the value to 0 will blacklist it.", "Use the registry name of the entity followed by a ';' and then the value you want.", "Vanilla entities do not require the modid." }) + public String[] sacrificialValues = { "villager;100", "slime;15", "enderman;10", "cow;100", "chicken;100", "horse;100", "sheep;100", "wolf;100", "ocelot;100", "pig;100", "rabbit;100" }; + @Config.Comment({ "Amount of LP the Coat of Arms should provide for each damage dealt." }) @Config.RangeInt(min = 0, max = 100) public int coatOfArmsConversion = 20; - @Config.Comment({"Amount of LP the Sacrificial Dagger should provide for each damage dealt."}) + @Config.Comment({ "Amount of LP the Sacrificial Dagger should provide for each damage dealt." }) @Config.RangeInt(min = 0, max = 10000) public int sacrificialDaggerConversion = 100; - @Config.Comment({"Will rewrite any default meteor types with new versions.", "Disable this if you want any of your changes to stay, or do not want default meteor types regenerated."}) + @Config.Comment({ "Will rewrite any default meteor types with new versions.", "Disable this if you want any of your changes to stay, or do not want default meteor types regenerated." }) public boolean shouldResyncMeteors = true; } - public static class ConfigRituals { + public static class ConfigRituals + { public boolean ritualAnimalGrowth = true; public boolean ritualContainment = true; public boolean ritualCrushing = true; @@ -93,33 +100,38 @@ public class ConfigHandler { public boolean ritualPortal = true; public boolean ritualMeteor = true; public boolean ritualDowngrade = true; + public boolean ritualEllipsoid = true; public ConfigImperfectRituals imperfect = new ConfigImperfectRituals(); } - public static class ConfigImperfectRituals { + public static class ConfigImperfectRituals + { public boolean imperfectRitualNight = true; public boolean imperfectRitualRain = true; public boolean imperfectRitualResistance = true; public boolean imperfectRitualZombie = true; } - public static class ConfigClient { - @Config.Comment({"Always render the beams between routing nodes.", "If disabled, the beams will only render while the Node Router is held."}) + public static class ConfigClient + { + @Config.Comment({ "Always render the beams between routing nodes.", "If disabled, the beams will only render while the Node Router is held." }) public boolean alwaysRenderRoutingLines = false; - @Config.Comment({"Completely hide spectral blocks from view.", "If disabled, a transparent block will be displayed."}) + @Config.Comment({ "Completely hide spectral blocks from view.", "If disabled, a transparent block will be displayed." }) public boolean invisibleSpectralBlocks = true; - @Config.Comment({"When cycling through slots, the Sigil of Holding will skip over empty slots and move to the next occupied one.", "If disabled, it will behave identically to the default hotbar."}) + @Config.Comment({ "When cycling through slots, the Sigil of Holding will skip over empty slots and move to the next occupied one.", "If disabled, it will behave identically to the default hotbar." }) public boolean sigilHoldingSkipsEmptySlots = false; } - public static class ConfigCompat { - @Config.Comment({"The display mode to use when looking at a Blood Altar.", "ALWAYS - Always display information.", "SIGIL_HELD - Only display information when a Divination or Seers sigil is held in either hand.", "SIGIL_CONTAINED - Only display information when a Divination or Seers sigil is somewhere in the inventory."}) + public static class ConfigCompat + { + @Config.Comment({ "The display mode to use when looking at a Blood Altar.", "ALWAYS - Always display information.", "SIGIL_HELD - Only display information when a Divination or Seers sigil is held in either hand.", "SIGIL_CONTAINED - Only display information when a Divination or Seers sigil is somewhere in the inventory." }) public AltarDisplayMode wailaAltarDisplayMode = AltarDisplayMode.SIGIL_HELD; - public enum AltarDisplayMode { + public enum AltarDisplayMode + { ALWAYS, SIGIL_HELD, - SIGIL_CONTAINED,; + SIGIL_CONTAINED, ; } } } diff --git a/src/main/java/WayofTime/bloodmagic/alchemyArray/AlchemyArrayEffectLaputa.java b/src/main/java/WayofTime/bloodmagic/alchemyArray/AlchemyArrayEffectLaputa.java index d3701692..0a4dcbe5 100644 --- a/src/main/java/WayofTime/bloodmagic/alchemyArray/AlchemyArrayEffectLaputa.java +++ b/src/main/java/WayofTime/bloodmagic/alchemyArray/AlchemyArrayEffectLaputa.java @@ -118,7 +118,7 @@ public class AlchemyArrayEffectLaputa extends AlchemyArrayEffect public int getRandomRadius(Random rand) { - return rand.nextInt(4) + 4; + return rand.nextInt(5) + 4; } public int getRandomHeightOffset(Random rand) diff --git a/src/main/java/WayofTime/bloodmagic/compat/guideapi/book/CategoryAlchemy.java b/src/main/java/WayofTime/bloodmagic/compat/guideapi/book/CategoryAlchemy.java index bbf67db9..307cc083 100644 --- a/src/main/java/WayofTime/bloodmagic/compat/guideapi/book/CategoryAlchemy.java +++ b/src/main/java/WayofTime/bloodmagic/compat/guideapi/book/CategoryAlchemy.java @@ -113,6 +113,16 @@ public class CategoryAlchemy standardTurretPages.addAll(PageHelper.pagesForLongText(TextHelper.localize(keyBase + "standardTurret" + ".info"), 370)); entries.put(new ResourceLocation(keyBase + "standardTurret"), new EntryText(standardTurretPages, TextHelper.localize(keyBase + "standardTurret"), true)); + List laputaPages = new ArrayList<>(); + + PageAlchemyArray laputaRecipePage = BookUtils.getAlchemyPage("laputa"); + if (laputaRecipePage != null) + { + laputaPages.add(laputaRecipePage); + } + laputaPages.addAll(PageHelper.pagesForLongText(TextHelper.localize(keyBase + "laputa" + ".info"), 370)); + entries.put(new ResourceLocation(keyBase + "laputa"), new EntryText(laputaPages, TextHelper.localize(keyBase + "laputa"), true)); + List buffPages = new ArrayList<>(); buffPages.addAll(PageHelper.pagesForLongText(TextHelper.localize(keyBase + "buff" + ".info"), 370)); diff --git a/src/main/java/WayofTime/bloodmagic/registry/ModRituals.java b/src/main/java/WayofTime/bloodmagic/registry/ModRituals.java index e1c67c19..544df8a9 100644 --- a/src/main/java/WayofTime/bloodmagic/registry/ModRituals.java +++ b/src/main/java/WayofTime/bloodmagic/registry/ModRituals.java @@ -123,7 +123,7 @@ public class ModRituals RitualRegistry.registerRitual(downgradeRitual, ConfigHandler.rituals.ritualDowngrade); ellipsoidRitual = new RitualEllipsoid(); - RitualRegistry.registerRitual(ellipsoidRitual, false); + RitualRegistry.registerRitual(ellipsoidRitual, ConfigHandler.rituals.ritualEllipsoid); RitualCrushing.registerCuttingFluid(ItemCuttingFluid.FluidType.BASIC.getStack(), 250, 0.5); RitualCrushing.registerCuttingFluid(ItemCuttingFluid.FluidType.EXPLOSIVE.getStack(), 25, 0.05); diff --git a/src/main/java/WayofTime/bloodmagic/ritual/types/RitualEllipsoid.java b/src/main/java/WayofTime/bloodmagic/ritual/types/RitualEllipsoid.java index 2b454edb..e47a8c74 100644 --- a/src/main/java/WayofTime/bloodmagic/ritual/types/RitualEllipsoid.java +++ b/src/main/java/WayofTime/bloodmagic/ritual/types/RitualEllipsoid.java @@ -198,7 +198,7 @@ public class RitualEllipsoid extends Ritual @Override public int getRefreshCost() { - return 0; + return 5; } @Override @@ -217,8 +217,43 @@ public class RitualEllipsoid extends Ritual @Override public void gatherComponents(Consumer components) { - addCornerRunes(components, 1, 0, EnumRuneType.WATER); - addCornerRunes(components, 1, 1, EnumRuneType.WATER); + addCornerRunes(components, 1, 0, EnumRuneType.DUSK); + + addRune(components, 4, 0, 0, EnumRuneType.FIRE); + addRune(components, 5, 0, 0, EnumRuneType.FIRE); + addRune(components, 5, 0, -1, EnumRuneType.FIRE); + addRune(components, 5, 0, -2, EnumRuneType.FIRE); + addRune(components, -4, 0, 0, EnumRuneType.FIRE); + addRune(components, -5, 0, 0, EnumRuneType.FIRE); + addRune(components, -5, 0, 1, EnumRuneType.FIRE); + addRune(components, -5, 0, 2, EnumRuneType.FIRE); + + addRune(components, 0, 0, 4, EnumRuneType.AIR); + addRune(components, 0, 0, 5, EnumRuneType.AIR); + addRune(components, 1, 0, 5, EnumRuneType.AIR); + addRune(components, 2, 0, 5, EnumRuneType.AIR); + addRune(components, 0, 0, -4, EnumRuneType.AIR); + addRune(components, 0, 0, -5, EnumRuneType.AIR); + addRune(components, -1, 0, -5, EnumRuneType.AIR); + addRune(components, -2, 0, -5, EnumRuneType.AIR); + + addRune(components, 3, 0, 1, EnumRuneType.EARTH); + addRune(components, 3, 0, 2, EnumRuneType.EARTH); + addRune(components, 3, 0, 3, EnumRuneType.EARTH); + addRune(components, 2, 0, 3, EnumRuneType.EARTH); + addRune(components, -3, 0, -1, EnumRuneType.EARTH); + addRune(components, -3, 0, -2, EnumRuneType.EARTH); + addRune(components, -3, 0, -3, EnumRuneType.EARTH); + addRune(components, -2, 0, -3, EnumRuneType.EARTH); + + addRune(components, 1, 0, -3, EnumRuneType.WATER); + addRune(components, 2, 0, -3, EnumRuneType.WATER); + addRune(components, 3, 0, -3, EnumRuneType.WATER); + addRune(components, 3, 0, -2, EnumRuneType.WATER); + addRune(components, -1, 0, 3, EnumRuneType.WATER); + addRune(components, -2, 0, 3, EnumRuneType.WATER); + addRune(components, -3, 0, 3, EnumRuneType.WATER); + addRune(components, -3, 0, 2, EnumRuneType.WATER); } @Override diff --git a/src/main/resources/assets/bloodmagic/lang/en_US.lang b/src/main/resources/assets/bloodmagic/lang/en_US.lang index a218e51f..befcd6ac 100644 --- a/src/main/resources/assets/bloodmagic/lang/en_US.lang +++ b/src/main/resources/assets/bloodmagic/lang/en_US.lang @@ -713,6 +713,7 @@ ritual.bloodmagic.forsakenSoulRitual.crystal.info=(Crystal) Demon Crystals in th ritual.bloodmagic.forsakenSoulRitual.damage.info=(Damage) Mobs within this range will be slowly damaged, and when killed will grow the crystals. ritual.bloodmagic.crystalHarvestRitual.crystal.info=(Crystal) All Demon Will crystal clusters have a single crystal broken off, spawning the crystal into the world. If there is only one crystal on the cluster, it will not break it. +ritual.bloodmagic.ellipseRitual.info=Creates a hollow spheroid around the ritual using the blocks in the attached chest. ritual.bloodmagic.ellipseRitual.spheroidRange.info=(Placement) The range that the ritual will place its blocks in. Spheroid is centered on the ritual - if one side is shorter than the side opposite the spheroid is truncated. ritual.bloodmagic.ellipseRitual.chest.info=(Chest) The location of the inventory that the ritual will grab blocks from to place in the world. diff --git a/src/main/resources/assets/bloodmagicguide/lang/en_US.lang b/src/main/resources/assets/bloodmagicguide/lang/en_US.lang index 8f5d49b9..aa0981c5 100644 --- a/src/main/resources/assets/bloodmagicguide/lang/en_US.lang +++ b/src/main/resources/assets/bloodmagicguide/lang/en_US.lang @@ -246,7 +246,7 @@ guide.bloodmagic.entry.alchemy.fastMiner=Fast Miner Array guide.bloodmagic.entry.alchemy.furnace=Burning Furnace Array guide.bloodmagic.entry.alchemy.teleport=Teleportation Array guide.bloodmagic.entry.alchemy.standardTurret=Turret Array - +guide.bloodmagic.entry.alchemy.laputa=Shard of Laputa # Alchemy Entry Texts @@ -261,6 +261,7 @@ guide.bloodmagic.entry.alchemy.fastMiner.info=When tasked with carving out a lar guide.bloodmagic.entry.alchemy.furnace.info=One of the many problems that you may encounter in the beginning of your adventure is the inability to keep your furnace lit. A lit furnace can mean the difference between having a full stomach and strong weapons or starving in a cave. \n\tThe Burning Furnace array, as the name implies, will provide a much needed heat source for any nearby furnace. By placing the array directly adjacent to a vanilla furnace (it can be next to multiple), it will provide fuel to the furnace if an operation is able to be completed - nothing will happen if it is next to an empty or a full furnace, mainly for your protection. \n\tThis does not come for free, however: when a person is nearby (within a 10 block radius), it will take away half a heart of health in order to cook up to two things in the furnace. This will be helpful to either get a quick bite or to smelt a full stack of ore, but unfortunately you haven't found a way to add any safety measures to the array... guide.bloodmagic.entry.alchemy.teleport.info=The Teleportation Array acts as a way to travel instantly from one location to another with a few specific limitations. When a player or other entity steps onto the array, the array will search up to 20 blocks away in the direction it is facing for another alchemy array (which does not need to be active). If the array manages to successfully find a destination, it will then teleport the entity to its found array instantly, even through walls. \n\tStudying this array has shown that there are further limitations added to it: because of the nature of bending the fabric of space-time, a Teleportation Array will not teleport something that has teleported within 2 seconds. This is to allow time for all components to rearrange themselves in a desirable manner. guide.bloodmagic.entry.alchemy.standardTurret.info=The power of flinging pointy objects at far away monsters cannot be overstated. The Turret Array is able to sense a nearby hostile monster and by utilizing complex alchemical mechanisms it is able to draw back and fire an arrow in order to strike its target. \n\tThe array searches for an inventory directly beneath it. If it finds either a normal or tipped arrow it will syphon from its container and fire at a mob that is within a 32 block radius. \n\t(Due to some silly Minecraft physics that has arrows bounce off of entities that are too close to where they spawn, the turret will also only fire on a mob that is greater than 3 blocks away. Keep that in mind!) +guide.bloodmagic.entry.alchemy.laputa.info=There exists a story of a lost kingdom that had such advanced magic that it could fly through the clouds. Although this kingdom has since crumbled to dust, a few drawings of this castle in the sky has managed to provide exquisite detail as to the mechanisms that allowed it to become the legend that it is. \n\tThe Shard of Laputa converts the life essence that is found within the earth into a more avian form, causing the surrounding area to levitate in the air. While the underlying principle is simple, the variations of the life essence in the earth cause a bit of inconsistency in how much land is moved. The array will move the earth in a spherical radius between 4 and 8 blocks up above itself a random offset between one and 5 blocks plus twice the radius of the effect. \n\tIt should be noted that as soon as the effect starts, all items used to activate the array will be lost.