package appeng.items.tools.powered; import java.util.EnumSet; import java.util.List; import net.minecraft.block.Block; import net.minecraft.block.BlockDispenser; import net.minecraft.entity.Entity; import net.minecraft.entity.EntityLivingBase; import net.minecraft.entity.item.EntityItem; import net.minecraft.entity.passive.EntitySheep; import net.minecraft.entity.player.EntityPlayer; import net.minecraft.inventory.IInventory; import net.minecraft.item.ItemStack; import net.minecraft.tileentity.TileEntity; import net.minecraft.util.AxisAlignedBB; import net.minecraft.util.DamageSource; import net.minecraft.util.MathHelper; import net.minecraft.util.MovingObjectPosition; import net.minecraft.util.MovingObjectPosition.MovingObjectType; import net.minecraft.util.Vec3; import net.minecraft.world.World; import net.minecraftforge.common.util.ForgeDirection; import appeng.api.AEApi; import appeng.api.config.Actionable; import appeng.api.config.FuzzyMode; import appeng.api.config.Upgrades; import appeng.api.implementations.items.IStorageCell; import appeng.api.networking.security.PlayerSource; import appeng.api.storage.ICellInventory; import appeng.api.storage.ICellInventoryHandler; import appeng.api.storage.IMEInventory; import appeng.api.storage.StorageChannel; import appeng.api.storage.data.IAEItemStack; import appeng.api.storage.data.IAEStack; import appeng.api.storage.data.IItemList; import appeng.api.util.AEColor; import appeng.api.util.DimensionalCoord; import appeng.core.AEConfig; import appeng.core.AELog; import appeng.core.CommonHelper; import appeng.core.features.AEFeature; import appeng.core.localization.GuiText; import appeng.core.localization.PlayerMessages; import appeng.core.sync.network.NetworkHandler; import appeng.core.sync.packets.PacketMatterCannon; import appeng.hooks.DispenserMatterCannon; import appeng.hooks.TickHandler; import appeng.hooks.TickHandler.PlayerColor; import appeng.items.contents.CellConfig; import appeng.items.contents.CellUpgrades; import appeng.items.misc.ItemPaintBall; import appeng.items.tools.powered.powersink.AEBasePoweredItem; import appeng.me.storage.CellInventoryHandler; import appeng.tile.misc.TilePaint; import appeng.util.Platform; public class ToolMassCannon extends AEBasePoweredItem implements IStorageCell { public ToolMassCannon() { super( ToolMassCannon.class, null ); setFeature( EnumSet.of( AEFeature.MatterCannon, AEFeature.PoweredTools ) ); maxStoredPower = AEConfig.instance.mattercannon_battery; } @Override public void postInit() { super.postInit(); BlockDispenser.dispenseBehaviorRegistry.putObject( this, new DispenserMatterCannon() ); } @Override public void addInformation(ItemStack is, EntityPlayer player, List lines, boolean advancedItemTooltips) { super.addInformation( is, player, lines, advancedItemTooltips ); IMEInventory cdi = AEApi.instance().registries().cell().getCellInventory( is, null, StorageChannel.ITEMS ); if ( cdi instanceof CellInventoryHandler ) { ICellInventory cd = ((ICellInventoryHandler) cdi).getCellInv(); if ( cd != null ) { lines.add( cd.getUsedBytes() + " " + GuiText.Of.getLocal() + " " + cd.getTotalBytes() + " " + GuiText.BytesUsed.getLocal() ); lines.add( cd.getStoredItemTypes() + " " + GuiText.Of.getLocal() + " " + cd.getTotalItemTypes() + " " + GuiText.Types.getLocal() ); } } } @Override public ItemStack onItemRightClick(ItemStack item, World w, EntityPlayer p) { if ( this.getAECurrentPower( item ) > 1600 ) { int shots = 1; CellUpgrades cu = (CellUpgrades) getUpgradesInventory( item ); if ( cu != null ) shots += cu.getInstalledUpgrades( Upgrades.SPEED ); IMEInventory inv = AEApi.instance().registries().cell().getCellInventory( item, null, StorageChannel.ITEMS ); if ( inv != null ) { IItemList itemList = inv.getAvailableItems( AEApi.instance().storage().createItemList() ); IAEStack aeammo = itemList.getFirstItem(); if ( aeammo instanceof IAEItemStack ) { shots = Math.min( shots, (int) aeammo.getStackSize() ); for (int sh = 0; sh < shots; sh++) { extractAEPower( item, 1600 ); if ( Platform.isClient() ) return item; aeammo.setStackSize( 1 ); ItemStack ammo = ((IAEItemStack) aeammo).getItemStack(); if ( ammo == null ) return item; ammo.stackSize = 1; aeammo = inv.extractItems( aeammo, Actionable.MODULATE, new PlayerSource( p, null ) ); if ( aeammo == null ) return item; float f = 1.0F; float f1 = p.prevRotationPitch + (p.rotationPitch - p.prevRotationPitch) * f; float f2 = p.prevRotationYaw + (p.rotationYaw - p.prevRotationYaw) * f; double d0 = p.prevPosX + (p.posX - p.prevPosX) * (double) f; double d1 = p.prevPosY + (p.posY - p.prevPosY) * (double) f + 1.62D - (double) p.yOffset; double d2 = p.prevPosZ + (p.posZ - p.prevPosZ) * (double) f; Vec3 vec3 = Vec3.createVectorHelper( d0, d1, d2 ); float f3 = MathHelper.cos( -f2 * 0.017453292F - (float) Math.PI ); float f4 = MathHelper.sin( -f2 * 0.017453292F - (float) Math.PI ); float f5 = -MathHelper.cos( -f1 * 0.017453292F ); float f6 = MathHelper.sin( -f1 * 0.017453292F ); float f7 = f4 * f5; float f8 = f3 * f5; double d3 = 32.0D; Vec3 vec31 = vec3.addVector( (double) f7 * d3, (double) f6 * d3, (double) f8 * d3 ); Vec3 direction = Vec3.createVectorHelper( (double) f7 * d3, (double) f6 * d3, (double) f8 * d3 ); direction.normalize(); float penetration = AEApi.instance().registries().matterCannon().getPenetration( ammo ); // 196.96655f; if ( penetration <= 0 ) { ItemStack type = ((IAEItemStack) aeammo).getItemStack(); if ( type.getItem() instanceof ItemPaintBall ) { shootPaintBalls( type, w, p, vec3, vec31, direction, d0, d1, d2 ); } return item; } else { standardAmmo( penetration, w, p, vec3, vec31, direction, d0, d1, d2 ); } } } else { if ( Platform.isServer() ) p.addChatMessage( PlayerMessages.AmmoDepleted.get() ); return item; } } } return item; } private void shootPaintBalls(ItemStack type, World w, EntityPlayer p, Vec3 vec3, Vec3 vec31, Vec3 direction, double d0, double d1, double d2) { AxisAlignedBB bb = AxisAlignedBB.getBoundingBox( Math.min( vec3.xCoord, vec31.xCoord ), Math.min( vec3.yCoord, vec31.yCoord ), Math.min( vec3.zCoord, vec31.zCoord ), Math.max( vec3.xCoord, vec31.xCoord ), Math.max( vec3.yCoord, vec31.yCoord ), Math.max( vec3.zCoord, vec31.zCoord ) ).expand( 16, 16, 16 ); Entity entity = null; List list = w.getEntitiesWithinAABBExcludingEntity( p, bb ); double closest = 9999999.0D; int l; for (l = 0; l < list.size(); ++l) { Entity entity1 = (Entity) list.get( l ); if ( entity1.isDead == false && entity1 != p && !(entity1 instanceof EntityItem) ) { if ( entity1.isEntityAlive() ) { // prevent killing / flying of mounts. if ( entity1.riddenByEntity == p ) continue; float f1 = 0.3F; AxisAlignedBB axisalignedbb1 = entity1.boundingBox.expand( (double) f1, (double) f1, (double) f1 ); MovingObjectPosition movingobjectposition1 = axisalignedbb1.calculateIntercept( vec3, vec31 ); if ( movingobjectposition1 != null ) { double nd = vec3.squareDistanceTo( movingobjectposition1.hitVec ); if ( nd < closest ) { entity = entity1; closest = nd; } } } } } MovingObjectPosition pos = w.rayTraceBlocks( vec3, vec31, false ); Vec3 Srec = Vec3.createVectorHelper( d0, d1, d2 ); if ( entity != null && pos != null && pos.hitVec.squareDistanceTo( Srec ) > closest ) { pos = new MovingObjectPosition( entity ); } else if ( entity != null && pos == null ) { pos = new MovingObjectPosition( entity ); } try { CommonHelper.proxy.sendToAllNearExcept( null, d0, d1, d2, 128, w, new PacketMatterCannon( d0, d1, d2, (float) direction.xCoord, (float) direction.yCoord, (float) direction.zCoord, (byte) (pos == null ? 32 : pos.hitVec.squareDistanceTo( Srec ) + 1) ) ); } catch (Exception err) { AELog.error( err ); } if ( pos != null && type != null && type.getItem() instanceof ItemPaintBall ) { ItemPaintBall ipb = (ItemPaintBall) type.getItem(); AEColor col = ipb.getColor( type ); // boolean lit = ipb.isLumen( type ); if ( pos.typeOfHit == MovingObjectType.ENTITY ) { int id = pos.entityHit.getEntityId(); PlayerColor marker = new PlayerColor( id, col, 20 * 30 ); TickHandler.instance.getPlayerColors().put( id, marker ); if ( pos.entityHit instanceof EntitySheep ) { EntitySheep sh = (EntitySheep) pos.entityHit; sh.setFleeceColor( col.ordinal() ); } pos.entityHit.attackEntityFrom( DamageSource.causePlayerDamage( p ), (float) 0 ); NetworkHandler.instance.sendToAll( marker.getPacket() ); } else if ( pos.typeOfHit == MovingObjectType.BLOCK ) { ForgeDirection side = ForgeDirection.getOrientation( pos.sideHit ); int x = pos.blockX + side.offsetX; int y = pos.blockY + side.offsetY; int z = pos.blockZ + side.offsetZ; if ( !Platform.hasPermissions( new DimensionalCoord( w, x, y, z ), p ) ) return; Block whatsThere = w.getBlock( x, y, z ); if ( whatsThere == AEApi.instance().blocks().blockPaint.block() ) { } else if ( whatsThere.isReplaceable( w, x, y, z ) && w.isAirBlock( x, y, z ) ) { w.setBlock( x, y, z, AEApi.instance().blocks().blockPaint.block(), 0, 3 ); } TileEntity te = w.getTileEntity( x, y, z ); if ( te instanceof TilePaint ) { pos.hitVec.xCoord -= x; pos.hitVec.yCoord -= y; pos.hitVec.zCoord -= z; ((TilePaint) te).addBlot( type, side.getOpposite(), pos.hitVec ); } } } } private void standardAmmo(float penetration, World w, EntityPlayer p, Vec3 vec3, Vec3 vec31, Vec3 direction, double d0, double d1, double d2) { boolean hasDestroyedSomething = true; while (penetration > 0 && hasDestroyedSomething) { hasDestroyedSomething = false; AxisAlignedBB bb = AxisAlignedBB.getBoundingBox( Math.min( vec3.xCoord, vec31.xCoord ), Math.min( vec3.yCoord, vec31.yCoord ), Math.min( vec3.zCoord, vec31.zCoord ), Math.max( vec3.xCoord, vec31.xCoord ), Math.max( vec3.yCoord, vec31.yCoord ), Math.max( vec3.zCoord, vec31.zCoord ) ).expand( 16, 16, 16 ); Entity entity = null; List list = w.getEntitiesWithinAABBExcludingEntity( p, bb ); double closest = 9999999.0D; int l; for (l = 0; l < list.size(); ++l) { Entity entity1 = (Entity) list.get( l ); if ( entity1.isDead == false && entity1 != p && !(entity1 instanceof EntityItem) ) { if ( entity1.isEntityAlive() ) { // prevent killing / flying of mounts. if ( entity1.riddenByEntity == p ) continue; float f1 = 0.3F; AxisAlignedBB axisalignedbb1 = entity1.boundingBox.expand( (double) f1, (double) f1, (double) f1 ); MovingObjectPosition movingobjectposition1 = axisalignedbb1.calculateIntercept( vec3, vec31 ); if ( movingobjectposition1 != null ) { double nd = vec3.squareDistanceTo( movingobjectposition1.hitVec ); if ( nd < closest ) { entity = entity1; closest = nd; } } } } } Vec3 Srec = Vec3.createVectorHelper( d0, d1, d2 ); MovingObjectPosition pos = w.rayTraceBlocks( vec3, vec31, true ); if ( entity != null && pos != null && pos.hitVec.squareDistanceTo( Srec ) > closest ) { pos = new MovingObjectPosition( entity ); } else if ( entity != null && pos == null ) { pos = new MovingObjectPosition( entity ); } try { CommonHelper.proxy.sendToAllNearExcept( null, d0, d1, d2, 128, w, new PacketMatterCannon( d0, d1, d2, (float) direction.xCoord, (float) direction.yCoord, (float) direction.zCoord, (byte) (pos == null ? 32 : pos.hitVec.squareDistanceTo( Srec ) + 1) ) ); } catch (Exception err) { AELog.error( err ); } if ( pos != null ) { DamageSource dmgSrc = DamageSource.causePlayerDamage( p ); dmgSrc.damageType = "masscannon"; if ( pos.typeOfHit == MovingObjectType.ENTITY ) { int dmg = (int) Math.ceil( penetration / 20.0f ); if ( pos.entityHit instanceof EntityLivingBase ) { EntityLivingBase el = (EntityLivingBase) pos.entityHit; penetration -= dmg; el.knockBack( p, 0, (double) -direction.xCoord, (double) -direction.zCoord ); // el.knockBack( p, 0, vec3.xCoord, // vec3.zCoord ); el.attackEntityFrom( dmgSrc, dmg ); if ( !el.isEntityAlive() ) hasDestroyedSomething = true; } else if ( pos.entityHit instanceof EntityItem ) { hasDestroyedSomething = true; pos.entityHit.setDead(); } else if ( pos.entityHit.attackEntityFrom( dmgSrc, dmg ) ) { hasDestroyedSomething = true; } } else if ( pos.typeOfHit == MovingObjectType.BLOCK ) { if ( !AEConfig.instance.isFeatureEnabled( AEFeature.MassCannonBlockDamage ) ) penetration = 0; else { Block b = w.getBlock( pos.blockX, pos.blockY, pos.blockZ ); // int meta = w.getBlockMetadata( // pos.blockX, pos.blockY, pos.blockZ ); float hardness = b.getBlockHardness( w, pos.blockX, pos.blockY, pos.blockZ ) * 9.0f; if ( hardness >= 0.0 ) { if ( penetration > hardness && Platform.hasPermissions( new DimensionalCoord( w, pos.blockX, pos.blockY, pos.blockZ ), p ) ) { hasDestroyedSomething = true; penetration -= hardness; penetration *= 0.60; w.func_147480_a( pos.blockX, pos.blockY, pos.blockZ, true ); // w.destroyBlock( pos.blockX, pos.blockY, pos.blockZ, true ); } } } } } } } @Override public boolean storableInStorageCell() { return true; } @Override public boolean isStorageCell(ItemStack i) { return true; } @Override public double getIdleDrain() { return 0.5; } @Override public IInventory getUpgradesInventory(ItemStack is) { return new CellUpgrades( is, 4 ); } @Override public IInventory getConfigInventory(ItemStack is) { return new CellConfig( is ); } @Override public FuzzyMode getFuzzyMode(ItemStack is) { String fz = Platform.openNbtData( is ).getString( "FuzzyMode" ); try { return FuzzyMode.valueOf( fz ); } catch (Throwable t) { return FuzzyMode.IGNORE_ALL; } } @Override public void setFuzzyMode(ItemStack is, FuzzyMode fzMode) { Platform.openNbtData( is ).setString( "FuzzyMode", fzMode.name() ); } @Override public boolean isEditable(ItemStack is) { return true; } @Override public int getBytes(ItemStack cellItem) { return 512; } @Override public int BytePerType(ItemStack iscellItem) { return 8; } @Override public int getTotalTypes(ItemStack cellItem) { return 1; } @Override public boolean isBlackListed(ItemStack cellItem, IAEItemStack requestedAddition) { float pen = AEApi.instance().registries().matterCannon().getPenetration( requestedAddition.getItemStack() ); if ( pen > 0 ) return false; if ( requestedAddition.getItem() instanceof ItemPaintBall ) return false; return true; } }