package StevenDimDoors.mod_pocketDim.helpers; /** * This class regulates all the operations involving the storage and manipulation of dimensions. It handles saving dim data, teleporting the player, and * creating/registering new dimensions as well as loading old dimensions on startup * @Return */ import java.io.File; import java.io.FileInputStream; import java.io.FileOutputStream; import java.io.ObjectOutputStream; import java.util.ArrayList; import java.util.HashMap; import java.util.Iterator; import java.util.Random; import java.util.Set; import net.minecraft.block.Block; import net.minecraft.entity.Entity; import net.minecraft.entity.EntityList; import net.minecraft.entity.player.EntityPlayer; import net.minecraft.entity.player.EntityPlayerMP; import net.minecraft.item.ItemStack; import net.minecraft.nbt.NBTTagCompound; import net.minecraft.network.packet.Packet41EntityEffect; import net.minecraft.network.packet.Packet43Experience; import net.minecraft.network.packet.Packet9Respawn; import net.minecraft.potion.PotionEffect; import net.minecraft.tileentity.TileEntity; import net.minecraft.util.MathHelper; import net.minecraft.world.World; import net.minecraft.world.WorldServer; import net.minecraft.world.chunk.Chunk; import net.minecraft.world.chunk.storage.ExtendedBlockStorage; import net.minecraftforge.common.DimensionManager; import StevenDimDoors.mod_pocketDim.DDProperties; import StevenDimDoors.mod_pocketDim.DimData; import StevenDimDoors.mod_pocketDim.LinkData; import StevenDimDoors.mod_pocketDim.ObjectSaveInputStream; import StevenDimDoors.mod_pocketDim.PacketHandler; import StevenDimDoors.mod_pocketDim.Point3D; import StevenDimDoors.mod_pocketDim.TileEntityRift; import StevenDimDoors.mod_pocketDim.mod_pocketDim; import StevenDimDoors.mod_pocketDim.world.LimboProvider; import StevenDimDoors.mod_pocketDim.world.pocketProvider; import cpw.mods.fml.common.FMLCommonHandler; import cpw.mods.fml.common.registry.GameRegistry; import cpw.mods.fml.relauncher.Side; public class dimHelper extends DimensionManager { /** * hashMap for the private pocket dimensions */ public static HashMap<String, DimData> privatePockets = new HashMap<String, DimData>(); /** * HashMap containing all the dims registered with DimDoors, sorted by dim ID. loaded on startup * @Return */ public static HashMap<Integer, DimData> dimList=new HashMap<Integer, DimData>(); public static boolean isSaving=false; /** * ArrayList containing any blocks in limbo that have been placed by the player. Cycled through in the common tick manager * @Return */ public static ArrayList<Point3D> blocksToDecay = new ArrayList<Point3D>(); /** * instance of the dimHelper * @Return */ public static dimHelper instance = new dimHelper(); /** * HashMap for temporary storage of Link Signature damage hash values. See itemLinkSignature for more details * @Return */ public HashMap<Integer, LinkData> interDimLinkList= new HashMap<Integer,LinkData>(); /** * ArrayList containing all link data not sorted for easy random access, used for random doors and for recreating rifts if they have a block placed over them. * See the common tick manager and the Chaos door for details on usage * @Return */ //public ArrayList<LinkData> linksForRendering =new ArrayList<LinkData>(); Random rand= new Random(); //Stupid function I use because I don't understand bitwise operations yet. Used in door orientation //TODO get rid of this public int flipDoorMetadata(int data) { if(data==0) { return 2; } if(data==1) { return 3; } if(data==2) { return 0; } if(data==3) { return 1; } if(data==4) { return 6; } if(data==5) { return 7; } if(data==6) { return 4; } if(data==7) { return 5; } else return -10; } public int getDimDepth(int DimID) { if (dimList.containsKey(DimID)) { return dimList.get(DimID).depth; } else return 1; } private Entity teleportEntity(World oldWorld, Entity entity, LinkData link) //this beautiful teleport method is based off of xCompWiz's teleport function. { Entity cart=entity.ridingEntity; if(entity.riddenByEntity!=null) { return this.teleportEntity(oldWorld,entity.riddenByEntity, link); } if (entity.ridingEntity != null) { entity.mountEntity(null); cart = teleportEntity(oldWorld, cart, link); } WorldServer newWorld; if(this.getWorld(link.destDimID)==null) { this.initDimension(link.destDimID); } boolean difDest = link.destDimID != link.locDimID; if(difDest) { newWorld = this.getWorld(link.destDimID); } else { newWorld=(WorldServer)oldWorld; } mod_pocketDim.teleporter.placeInPortal(entity, newWorld, link); if ((entity instanceof EntityPlayerMP)) { EntityPlayerMP player = (EntityPlayerMP)entity; //player.closeScreen(); if (difDest) { player.dimension = link.destDimID; player.playerNetServerHandler.sendPacketToPlayer(new Packet9Respawn(player.dimension, (byte)player.worldObj.difficultySetting, newWorld.getWorldInfo().getTerrainType(), newWorld.getHeight(), player.theItemInWorldManager.getGameType())); WorldServer.class.cast(oldWorld).removeEntity(player); player.isDead=false; oldWorld.playerEntities.remove(player); WorldServer.class.cast(oldWorld).getPlayerManager().removePlayer(player); newWorld.getPlayerManager().addPlayer(player); player.theItemInWorldManager.setWorld((WorldServer)newWorld); player.mcServer.getConfigurationManager().updateTimeAndWeatherForPlayer(player, (WorldServer)newWorld); player.mcServer.getConfigurationManager().syncPlayerInventory(player); for(Object potionEffect : player.getActivePotionEffects()) { PotionEffect effect = (PotionEffect)potionEffect; player.playerNetServerHandler.sendPacketToPlayer(new Packet41EntityEffect(player.entityId, effect)); } player.playerNetServerHandler.sendPacketToPlayer(new Packet43Experience(player.experience, player.experienceTotal, player.experienceLevel)); } } if(difDest) { int entX = entity.chunkCoordX; int entZ = entity.chunkCoordZ; if ((entity.addedToChunk) && (oldWorld.getChunkProvider().chunkExists(entX, entZ))) { oldWorld.getChunkFromChunkCoords(entX, entZ).removeEntity(entity); oldWorld.getChunkFromChunkCoords(entX, entZ).isModified = true; } oldWorld.releaseEntitySkin(entity); // entity.isDead = false; if (!(entity instanceof EntityPlayer)) { NBTTagCompound entityNBT = new NBTTagCompound(); entity.isDead = false; entity.addEntityID(entityNBT); entity.isDead = true; entity = EntityList.createEntityFromNBT(entityNBT, newWorld); if (entity == null) { } } newWorld.spawnEntityInWorld(entity); entity.setWorld(newWorld); } entity.worldObj.updateEntityWithOptionalForce(entity, false); if ((entity != null) && (cart != null)) { if ((entity instanceof EntityPlayerMP)) { EntityPlayerMP playerMP = (EntityPlayerMP)entity; entity.worldObj.updateEntityWithOptionalForce(entity, true); } entity.mountEntity(cart); } if(entity instanceof EntityPlayerMP) { WorldServer.class.cast(newWorld).getChunkProvider().loadChunk(MathHelper.floor_double(entity.posX) >> 4, MathHelper.floor_double(entity.posZ) >> 4); } mod_pocketDim.teleporter.placeInPortal(entity, newWorld, link); return entity; } /** * Primary function used to teleport the player using doors. Performs numerous null checks, and also generates the destination door/pocket if it has not done so already. * Also ensures correct orientation relative to the door using the pocketTeleporter. * @param world- world the player is currently in * @param linkData- the link the player is using to teleport, sends the player to its dest information. * @param player- the instance of the player to be teleported * @param orientation- the orientation of the door used to teleport, determines player orientation and door placement on arrival * @Return */ public void teleportToPocket(World world,LinkData linkData, Entity entity) { DDProperties properties = DDProperties.instance(); if (world.isRemote) { return; } if (linkData != null) { int destinationID=linkData.destDimID; int x=linkData.destXCoord; int y=linkData.destYCoord; int z=linkData.destZCoord; int depth= this.getDimDepth(world.provider.dimensionId); if(this.dimList.containsKey(destinationID) && this.dimList.containsKey(world.provider.dimensionId)) { this.generatePocket(linkData); if(mod_pocketDim.teleTimer==0||entity instanceof EntityPlayer) { mod_pocketDim.teleTimer=2+rand.nextInt(2); } else { return; } if(!world.isRemote) { entity = this.teleportEntity(world, entity, linkData); } if(entity instanceof EntityPlayerMP) { if(world.provider.dimensionId!=linkData.destDimID) { GameRegistry.onPlayerChangedDimension((EntityPlayer)entity); } } entity.worldObj.playSoundEffect(entity.posX, entity.posY, entity.posZ, "mob.endermen.portal", 1.0F, 1.0F); int playerXCoord=MathHelper.floor_double(entity.posX); int playerYCoord=MathHelper.floor_double(entity.posY); int playerZCoord=MathHelper.floor_double(entity.posZ); if(!entity.worldObj.isBlockOpaqueCube(playerXCoord, playerYCoord-1,playerZCoord )&&this.dimList.get(linkData.locDimID).isDimRandomRift&&!linkData.hasGennedDoor) { for(int count=0;count<20;count++) { if(entity.worldObj.isBlockOpaqueCube(playerXCoord, playerYCoord-1-count,playerZCoord)) { break; } if(count==19) { entity.worldObj.setBlock(playerXCoord, playerYCoord-1, playerZCoord, properties.FabricBlockID); } } } if(entity.worldObj.getBlockId(playerXCoord, playerYCoord-1,playerZCoord )==Block.lavaStill.blockID) { entity.worldObj.setBlock(playerXCoord, playerYCoord-1, playerZCoord, properties.FabricBlockID); } this.generateDoor(world,linkData); if(Block.blocksList.length>=entity.worldObj.getBlockId(playerXCoord,playerYCoord+1,playerZCoord)&&!entity.worldObj.isAirBlock(playerXCoord,playerYCoord+1,playerZCoord)) { if(Block.blocksList[entity.worldObj.getBlockId(playerXCoord,playerYCoord+1,playerZCoord)].isOpaqueCube()&&!mod_pocketDim.blocksImmuneToRift.contains(entity.worldObj.getBlockId(playerXCoord,playerYCoord+1,playerZCoord))) { entity.worldObj.setBlock(playerXCoord,playerYCoord+1,playerZCoord,0); } } if(Block.blocksList.length>=entity.worldObj.getBlockId(playerXCoord,playerYCoord,playerZCoord)&&!entity.worldObj.isAirBlock(playerXCoord,playerYCoord,playerZCoord)) { if(Block.blocksList[entity.worldObj.getBlockId(playerXCoord,playerYCoord,playerZCoord)].isOpaqueCube()&&!mod_pocketDim.blocksImmuneToRift.contains(entity.worldObj.getBlockId(playerXCoord,playerYCoord,playerZCoord))) { entity.worldObj.setBlock(playerXCoord,playerYCoord,playerZCoord,0); } } } } else if(!this.dimList.containsKey(world.provider.dimensionId)) { if(!(world.provider instanceof pocketProvider ||world.provider instanceof LimboProvider)) { DimData data = new DimData(world.provider.dimensionId, false, 0, 0, world.getSpawnPoint().posX, world.getSpawnPoint().posY, world.getSpawnPoint().posZ); } } return; } /** * Creates a link at the location, pointing to the destination. Does NOT create a pair, so must be called twice. * @param locationDimID * @param destinationDimID * @param locationXCoord * @param locationYCoord * @param locationZCoord * @param destinationXCoord * @param destinationYCoord * @param destinationZCoord * @return */ public LinkData createLink( int locationDimID, int destinationDimID, int locationXCoord, int locationYCoord, int locationZCoord, int destinationXCoord, int destinationYCoord, int destinationZCoord) { if(this.getLinkDataFromCoords(locationXCoord, locationYCoord, locationZCoord, locationDimID)!=null) { return this.createLink(locationDimID, destinationDimID, locationXCoord, locationYCoord, locationZCoord, destinationXCoord, destinationYCoord, destinationZCoord, this.getLinkDataFromCoords(locationXCoord, locationYCoord, locationZCoord, locationDimID).linkOrientation); } else { return this.createLink(locationDimID, destinationDimID, locationXCoord, locationYCoord, locationZCoord, destinationXCoord, destinationYCoord, destinationZCoord, -10); } } /** * Creates a link at the location, pointing to the destination. Does NOT create a pair, so must be called twice. * @param locationDimID * @param destinationDimID * @param locationXCoord * @param locationYCoord * @param locationZCoord * @param destinationXCoord * @param destinationYCoord * @param destinationZCoord * @param linkOrientation * @return */ public LinkData createLink( int locationDimID, int destinationDimID, int locationXCoord, int locationYCoord, int locationZCoord, int destinationXCoord, int destinationYCoord, int destinationZCoord,int linkOrientation) { LinkData linkData =new LinkData( locationDimID, destinationDimID, locationXCoord, locationYCoord, locationZCoord, destinationXCoord, destinationYCoord ,destinationZCoord,false,linkOrientation); return this.createLink(linkData); } public LinkData createLink(LinkData link) { DDProperties properties = DDProperties.instance(); if(!this.dimList.containsKey(link.locDimID)) { DimData locationDimData= new DimData(link.locDimID, false, 0, link.locDimID,link.locXCoord,link.locYCoord,link.locZCoord); this.dimList.put(link.locDimID, locationDimData); link.isLocPocket=false; } if(!dimList.containsKey(link.destDimID)) { dimHelper.dimList.put(link.destDimID, new DimData(link.destDimID, false, 0, link.locDimID,link.locXCoord,link.locYCoord,link.locZCoord)); } DimData locationDimData= this.dimList.get(link.locDimID); link.isLocPocket=locationDimData.isPocket; locationDimData.addLinkToDim(link); //this.linksForRendering.add(link); if(dimHelper.getWorld(link.locDimID)!=null) { //World world =dimHelper.getWorld(link.locDimID); int blocktoReplace = dimHelper.getWorld(link.locDimID).getBlockId(link.locXCoord, link.locYCoord, link.locZCoord); if(!mod_pocketDim.blocksImmuneToRift.contains(blocktoReplace)) { dimHelper.getWorld(link.locDimID).setBlock(link.locXCoord, link.locYCoord, link.locZCoord, properties.RiftBlockID); } } //Notifies other players that a link has been created. if(FMLCommonHandler.instance().getEffectiveSide()==Side.SERVER) { // System.out.println("Sending link creation packet with orientation "+link.linkOrientation); PacketHandler.onLinkCreatedPacket(link); } return link; } public int getDestOrientation(LinkData link) { if(link !=null) { LinkData destLink = this.getLinkDataFromCoords(link.destXCoord, link.destYCoord, link.destZCoord, link.destDimID); if(destLink!=null) { return destLink.linkOrientation; } else { //System.out.println("Cant find destination link"); return 0; } } else { // System.out.println("sending link is null"); return 0; } } public void removeLink(LinkData link) { this.removeLink(link.locDimID, link.locXCoord, link.locYCoord, link.locZCoord); } /** * properly deletes a link at the given coordinates. used by the rift remover. Also notifies clients of change. * @param locationDimID * @param locationXCoord * @param locationYCoord * @param locationZCoord */ public void removeLink( int locationDimID, int locationXCoord, int locationYCoord, int locationZCoord) { if(!this.dimList.containsKey(locationDimID)) { DimData locationDimData= new DimData(locationDimID, false, 0, locationDimID,locationXCoord,locationYCoord,locationZCoord); this.dimList.put(locationDimID, locationDimData); } LinkData link = this.getLinkDataFromCoords(locationXCoord, locationYCoord, locationZCoord, locationDimID); // this.linksForRendering.remove(link); this.dimList.get(locationDimID).removeLinkAtCoords(link); //updates clients that a rift has been removed if(FMLCommonHandler.instance().getEffectiveSide()==Side.SERVER) { PacketHandler.onLinkRemovedPacket(link); this.save(); } } public LinkData findNearestRift(World world, int x, int y, int z, int range) { return this.dimList.get(world).findNearestRift(world, range, x, y, z); } /** * generates a door based on what door was used to teleport. Only funtions once per linking. * @param world- door * @param incLink */ public void generateDoor(World world, LinkData incLink) { int locX = incLink.locXCoord; int locY = incLink.locYCoord; int locZ = incLink.locZCoord; int destX = incLink.destXCoord; int destY = incLink.destYCoord; int destZ = incLink.destZCoord; DDProperties properties = DDProperties.instance(); if(!incLink.hasGennedDoor) { int destinationID = incLink.destDimID; DimData data = this.dimList.get(destinationID); int id =world.getBlockId(locX, locY, locZ); if(id==properties.WarpDoorID||id==properties.DimensionalDoorID||id==properties.TransientDoorID) { int doorTypeToPlace=id; // World destWorld = this.getWorld(destinationID); if(this.getWorld(destinationID)==null) { this.initDimension(destinationID); } int locOrientation = incLink.linkOrientation; LinkData destLink = this.getLinkDataFromCoords(destX, destY, destZ, destinationID); int destOrientation=0 ; if(destLink!=null) { destOrientation = destLink.linkOrientation; destLink.hasGennedDoor=true; } int blockToReplace= this.getWorld(destinationID).getBlockId(destX, destY, destZ); if(blockToReplace!=properties.DimensionalDoorID&&blockToReplace!=properties.WarpDoorID&&blockToReplace != properties.TransientDoorID) { this.getWorld(destinationID).setBlock(destX, destY-1, destZ, doorTypeToPlace,destOrientation,2); this.getWorld(destinationID).setBlock(destX, destY, destZ, doorTypeToPlace,8,2); // System.out.println("Genned door"); } incLink.hasGennedDoor=true; } } } /** * Generates the black pocket out of fabric of reality blocks. Placement of the pocket is based off of the orignial doors orientation. Kind of a clunky method, * but is nessesarry to maintain a one to one relationship with the overworld. Is called every teleport, but checks if the dim has been filled first and is a pocket . * Also responsible for generation the random dungeons. * @param world- id of the world TO BE FILLED * @param x * @param y * @param z * @param orientation * @return */ public void generatePocket(LinkData incomingLink) { DDProperties properties = DDProperties.instance(); try { if(this.getWorld(incomingLink.destDimID)==null) { this.initDimension(incomingLink.destDimID); } if(this.getWorld(incomingLink.destDimID).provider==null) { this.initDimension(incomingLink.destDimID); } } catch(Exception E) { E.printStackTrace(); return; } // World world = this.getWorld(incomingLink.destDimID); DimData data = this.dimList.get(incomingLink.destDimID); if(!data.hasBeenFilled&&data.isPocket&&!data.isDimRandomRift) { data.hasBeenFilled=true; //System.out.println("genning pocket"); int x = incomingLink.destXCoord; int y = incomingLink.destYCoord; int z = incomingLink.destZCoord; int orientation= (incomingLink.linkOrientation); int depth= this.getDimDepth(incomingLink.locDimID); //x=x*depth; //y=y*depth; //z=z*depth; y=y+13; if(orientation==0) { x=x+15; this.getWorld(incomingLink.destDimID).provider.setSpawnPoint(x-1, y, z); } else if(orientation==1) { z=z+15; this.getWorld(incomingLink.destDimID).provider.setSpawnPoint(x, y, z-1); } else if(orientation==2) { x=x-15; this.getWorld(incomingLink.destDimID).provider.setSpawnPoint(x+1, y, z); } else if(orientation==3) { z=z-15; this.getWorld(incomingLink.destDimID).provider.setSpawnPoint(x, y, z+1); } int searchRadius=19; if(!this.getWorld(incomingLink.destDimID).isRemote) { int xCount=-searchRadius; int yCount=-searchRadius; int zCount=-searchRadius; while (xCount<=searchRadius) { while(yCount<=searchRadius) { while(zCount<=searchRadius) { if((Math.abs(xCount)>=15||Math.abs(yCount)>=15||Math.abs(zCount)>=15)&&this.getWorld(incomingLink.destDimID).isAirBlock( x+xCount, y+yCount, z+zCount)&&((yCount+y)>0)) { if(Math.abs(xCount)>=19||Math.abs(yCount)>=19||Math.abs(zCount)>=19) { this.setBlockDirectly(this.getWorld(incomingLink.destDimID), x+xCount, y+yCount, z+zCount,properties.PermaFabricBlockID,0); } else { this.setBlockDirectly(this.getWorld(incomingLink.destDimID), x+xCount, y+yCount, z+zCount,properties.FabricBlockID,0); if(properties.TNFREAKINGT_Enabled) { if((Math.abs(xCount)>=16||Math.abs(yCount)>=16||Math.abs(zCount)>=16) && rand.nextInt(properties.NonTntWeight + 1) == 0) { this.getWorld(incomingLink.destDimID).setBlock( x+xCount, y+yCount, z+zCount,Block.tnt.blockID); } } } } zCount++; } zCount=-searchRadius; yCount++; } yCount=-searchRadius; xCount++; } } } else if(!data.hasBeenFilled&&data.isPocket&&data.isDimRandomRift) { //System.out.println("genning dungeon pocket"); mod_pocketDim.loader.generateDungeonPocket(incomingLink); data.hasBeenFilled=true; } } /** * simple method called on startup to register all dims saved in the dim list. Only tries to register pocket dims, though. Also calls load() * @return */ public void initPockets() { DDProperties properties = DDProperties.instance(); mod_pocketDim.hasInitDims=true; this.load(); if(!this.dimList.isEmpty()) { Set allDimIds=dimList.keySet(); Iterator itr =allDimIds.iterator(); while(itr.hasNext()) { DimData dimData = (DimData) dimList.get(itr.next()); if(dimData.isPocket) { try { this.getNextFreeDimId(); registerDimension(dimData.dimID,properties.PocketProviderID); } catch (Exception e) { if(dimData.isPocket) { System.out.println("Warning- could not register dim "+dimData.depth+" . Probably caused by a version update/save data corruption/other mods. "); } else { e.printStackTrace(); } } } } } } public boolean resetPocket(DimData dimData) { //TODO: Should we add a check to see if the dimension is currently loaded? How could we check that? ~SenseiKiwi if (!dimData.isPocket || getWorld(dimData.dimID) != null) { return false; } File save = new File(getCurrentSaveRootDirectory() + "/DimensionalDoors/pocketDimID" + dimData.dimID); DeleteFolder.deleteFolder(save); dimData.hasBeenFilled = false; dimData.hasDoor = false; for(LinkData link : dimData.getLinksInDim()) { link.hasGennedDoor = false; LinkData linkOut = this.getLinkDataFromCoords(link.destXCoord, link.destYCoord, link.destZCoord, link.destDimID); if (linkOut != null) { linkOut.hasGennedDoor = false; } } return true; } public boolean pruneDimension(DimData dimData, boolean deleteFolder) { //TODO: Should we add a check to see if the dimension is currently loaded? How could we check that? ~SenseiKiwi //TODO: All the logic for checking that this is an isolated pocket should be moved in here. if (!dimData.isPocket || getWorld(dimData.dimID) != null) { return false; } dimList.remove(dimData.dimID); if (deleteFolder) { File save = new File(getCurrentSaveRootDirectory() + "/DimensionalDoors/pocketDimID" + dimData.dimID); DeleteFolder.deleteFolder(save); } return true; } /** * method called when the client disconnects/server stops to unregister dims. * @Return */ public void unregsisterDims() { if(!this.dimList.isEmpty()) { Set allDimIds=dimList.keySet(); Iterator itr =allDimIds.iterator(); while(itr.hasNext()) { DimData dimData = (DimData) dimList.get(itr.next()); if(dimData.isPocket) { try { this.unregisterDimension(dimData.dimID); } catch(Exception e) { System.out.println("Dim-"+String.valueOf(dimData.dimID)+"is already unregistered, ok? Enough with it already."); } // initDimension(dimData.dimID); } } } } /** * Used to associate a damage value on a Rift Signature with a link pair. See LinkSignature for details. * @return */ public int createUniqueInterDimLinkKey() { int linkKey; Random rand= new Random(); do { linkKey=rand.nextInt(30000); } while(this.interDimLinkList.containsKey(linkKey)); return linkKey; } /** * Method used to create and register a new pocket dimension. Called on door placement and rift generation. It does NOT actually generate the structure of the dim, just * registers it with the dimension manager and adds the necessary links and dim info to the dimlist/linklists. * Also registers existing dims with the dimList, so link data can be stored for them. * * Handles the randomization associated with depth as well, going far enough causes the next dims exit link to be randomized. * * @param world- World currently occupied, the parent of the pocket dim to be created. * @param x * @param y * @param z * @param isGoingDown * @param isRandomRift * @param orientation- determines the orientation of the entrance link to this dim. Should be the metaData of the door occupying the rift. -1 if no door. * @return */ public LinkData createPocket(LinkData link , boolean isGoingDown, boolean isRandomRift) { DDProperties properties = DDProperties.instance(); if(dimHelper.getWorld(0)==null) { return link; } if (dimHelper.getWorld(link.locDimID) == null) { dimHelper.initDimension(link.locDimID); } int dimensionID; int depth = this.getDimDepth(link.locDimID); // World world = this.getWorld(link.locDimID); dimensionID = getNextFreeDimId(); registerDimension(dimensionID, properties.PocketProviderID); DimData locationDimData; DimData destDimData; if(this.dimList.containsKey(link.locDimID)&&!this.getWorld(link.locDimID).isRemote) //checks to see if dim is already registered. If not, it creates a DimData entry for it later { //randomizes exit if deep enough locationDimData= dimList.get(this.getWorld(link.locDimID).provider.dimensionId); if(depth>5) { if(depth>=12) { depth=11; } if(rand.nextInt(13-depth)==0) { LinkData link1=getRandomLinkData(false); if(link1!=null) { // locationDimData.exitDimLink=new LinkData(link1.locDimID, link1.locDimID, link1.locXCoord, link1.locYCoord, link1.locZCoord, link1.locXCoord, link1.locYCoord, link1.locZCoord, false); } } } if(locationDimData.isPocket) //determines the qualites of the pocket dim being created, based on parent dim. { if(isGoingDown) { destDimData= new DimData(dimensionID, true, locationDimData.depth+1, locationDimData.exitDimLink); } else { destDimData= new DimData(dimensionID, true, locationDimData.depth-1, locationDimData.exitDimLink); } } else { destDimData= new DimData(dimensionID, true, 1, link.locDimID,link.locXCoord,link.locYCoord,link.locZCoord); } } else { locationDimData= new DimData(link.locDimID, false, 0, link.locDimID,link.locXCoord,link.locYCoord,link.locZCoord); destDimData= new DimData(dimensionID, true, 1, link.locDimID,link.locXCoord,link.locYCoord,link.locZCoord); } destDimData.isDimRandomRift=isRandomRift; this.dimList.put(this.getWorld(link.locDimID).provider.dimensionId, locationDimData); this.dimList.put(dimensionID, destDimData); if(FMLCommonHandler.instance().getEffectiveSide()==Side.SERVER)//sends packet to clients notifying them that a new dim has been created. { PacketHandler.onDimCreatedPacket(destDimData); } link = this.createLink(this.getWorld(link.locDimID).provider.dimensionId,dimensionID,link.locXCoord,link.locYCoord,link.locZCoord, link.destXCoord,link.destYCoord,link.destZCoord,link.linkOrientation); //creates and registers the two rifts that link the parent and pocket dim. this.createLink(dimensionID,this.getWorld(link.locDimID).provider.dimensionId, link.destXCoord,link.destYCoord,link.destZCoord, link.locXCoord,link.locYCoord,link.locZCoord, this.flipDoorMetadata(link.linkOrientation)); if (isRandomRift) { DungeonHelper.instance().generateDungeonLink(link); } return link; } /** * function that saves all dim data in a hashMap. Calling too often can cause Concurrent modification exceptions, so be careful. * @return */ //TODO change from saving serialized objects to just saving data for compatabilies sake. //TODO If saving is multithreaded as the concurrent modification exception implies, you should be synchronizing access. ~SenseiKiwi public void save() { if(this.isSaving) return; World world = DimensionManager.getWorld(0); if(world==null || world.isRemote) return; if(this.getCurrentSaveRootDirectory()!=null) { //System.out.println("saving"); this.isSaving=true; HashMap comboSave=new HashMap(); comboSave.put("dimList", this.dimList); comboSave.put("interDimLinkList", this.interDimLinkList); comboSave.put("blocksToDecay", this.blocksToDecay); FileOutputStream saveFile = null; try { //World world=FMLCommonHandler.instance().getMinecraftServerInstance().worldServers[0]; String saveFileName=this.getCurrentSaveRootDirectory()+"/DimensionalDoorsDataTEMP"; saveFile = new FileOutputStream(saveFileName); ObjectOutputStream save = new ObjectOutputStream(saveFile); save.writeObject(comboSave); save.close(); saveFile.close(); if(new File(this.getCurrentSaveRootDirectory()+"/DimensionalDoorsDataOLD").exists()) { new File(this.getCurrentSaveRootDirectory()+"/DimensionalDoorsDataOLD").delete(); } new File(this.getCurrentSaveRootDirectory()+"/DimensionalDoorsData").renameTo(new File(this.getCurrentSaveRootDirectory()+"/DimensionalDoorsDataOLD")); new File(saveFileName).renameTo( new File(this.getCurrentSaveRootDirectory()+"/DimensionalDoorsData")); } catch(Exception e) { e.printStackTrace(); System.out.println("Could not save data-- SEVERE"); } this.isSaving=false; } } /** * loads the dim data from the saved hashMap. Also handles compatabilty with old saves, see OldSaveHandler * @return */ //TODO change to loading vars instead of objects public void load() { boolean firstRun=false; System.out.println("Loading DimDoors data"); FileInputStream saveFile = null; if(!DimensionManager.getWorld(0).isRemote&&this.getCurrentSaveRootDirectory()!=null) { try { World world=FMLCommonHandler.instance().getMinecraftServerInstance().worldServers[0]; File dataStore =new File( this.getCurrentSaveRootDirectory()+"/DimensionalDoorsData"); if(!dataStore.exists()) { if(!new File( this.getCurrentSaveRootDirectory()+"/DimensionalDoorsDataOLD").exists()) { firstRun=true; } } saveFile = new FileInputStream(dataStore); ObjectSaveInputStream save = new ObjectSaveInputStream(saveFile); HashMap comboSave =((HashMap)save.readObject()); try { this.interDimLinkList=(HashMap) comboSave.get("interDimLinkList"); } catch(Exception e) { System.out.println("Could not load Link Signature list. Link Sig items will loose restored locations."); } try { this.dimList=(HashMap) comboSave.get("dimList"); } catch(Exception e) { System.out.println("Could not load pocket dim list. Saves probably lost, but repairable. Move the files from indivual pocket dim files to active ones. See MC thread for details."); } try { this.blocksToDecay=(ArrayList) comboSave.get("blocksToDecay"); } catch(Exception e) { System.out.println("Could not load list of blocks to decay in Limbo. Probably because you updated versions, in which case this is normal. "); } save.close(); saveFile.close(); } catch(Exception e4) { try { if(!firstRun) { System.out.println("Save data damaged, trying backup..."); } World world=FMLCommonHandler.instance().getMinecraftServerInstance().worldServers[0]; File dataStore =new File( world.getSaveHandler().getMapFileFromName("idcounts").getParentFile().getParent()+"/DimensionalDoorsDataOLD"); saveFile = new FileInputStream(dataStore); ObjectSaveInputStream save = new ObjectSaveInputStream(saveFile); HashMap comboSave =((HashMap)save.readObject()); try { this.interDimLinkList=(HashMap) comboSave.get("interDimLinkList"); } catch(Exception e) { System.out.println("Could not load Link Signature list. Link Sig items will loose restored locations."); } try { this.dimList=(HashMap) comboSave.get("dimList"); } catch(Exception e) { System.out.println("Could not load pocket dim list. Saves probably lost, but repairable. Move the files from indivual pocket dim files to active ones. See MC thread for details."); } try { this.blocksToDecay=(ArrayList) comboSave.get("blocksToDecay"); } catch(Exception e) { System.out.println("Could not load list of blocks to decay in Limbo. Probably because you updated versions, in which case this is normal. "); } save.close(); saveFile.close(); } catch(Exception e2) { if(!firstRun) { e2.printStackTrace(); System.out.println("Could not read data-- SEVERE"); } } } } } public LinkData getRandomLinkData(boolean allowInPocket) { boolean foundRandomDest=false; int i=0; int size = dimHelper.dimList.size(); while (!foundRandomDest&&size>0&&i<100) { i++; DimData dimData; ArrayList linksInDim=new ArrayList(); for(size--;size>0;) { dimData = dimHelper.dimList.get(dimList.keySet().toArray()[rand.nextInt(dimList.keySet().size())]); if(dimData==null) { break; } linksInDim = dimData.getLinksInDim(); if(!linksInDim.isEmpty()) { break; } } if(linksInDim.isEmpty()) { break; } LinkData link1 = (LinkData) linksInDim.get(rand.nextInt(linksInDim.size())); if(link1!=null) { if(!link1.isLocPocket||allowInPocket) { foundRandomDest=true; return link1; } } } return null; } /** * gets a link based on coords and a world object * @param x * @param y * @param z * @param par1World * @return */ public LinkData getLinkDataFromCoords(int x, int y, int z, World par1World) { return this.getLinkDataFromCoords(x, y, z, par1World.provider.dimensionId); } /** * gets a link based on coords and a world ID * @param x * @param y * @param z * @param worldID * @return */ public LinkData getLinkDataFromCoords(int x, int y, int z, int worldID) { if(this.dimList.containsKey(worldID)) { DimData dimData=this.dimList.get(worldID); return dimData.findLinkAtCoords(x, y, z); } return null; } /** * function called by rift tile entities and the rift remover to find and spread between rifts. Does not actually de-register the rift data, see deleteRift for that. * @param world * @param x * @param y * @param z * @param range * @param player * @param item * @return */ public static boolean removeRift(World world, int x, int y, int z, int range, EntityPlayer player, ItemStack item) { DDProperties properties = DDProperties.instance(); LinkData nearest=null; float distance=range+1; int i=-range; int j=-range; int k=-range; while (i<range) { while (j<range) { while (k<range) { if(world.getBlockId(x+i, y+j, z+k)==properties.RiftBlockID&&MathHelper.abs(i)+MathHelper.abs(j)+MathHelper.abs(k)<distance) { if(MathHelper.abs(i)+MathHelper.abs(j)+MathHelper.abs(k)!=0||range==1) { nearest=dimHelper.instance.getLinkDataFromCoords(x+i, y+j, z+k,world.provider.dimensionId); distance=MathHelper.abs(i)+MathHelper.abs(j)+MathHelper.abs(k); } } k++; } k=-range; j++; } j=-range; i++; } if(nearest!=null) { if(world.getBlockTileEntity(nearest.locXCoord,nearest.locYCoord, nearest.locZCoord)!=null) { TileEntity tile = world.getBlockTileEntity(nearest.locXCoord,nearest.locYCoord, nearest.locZCoord); TileEntityRift tilerift = (TileEntityRift) tile; tilerift.shouldClose=true; //System.out.println("erasing rift"); item.damageItem(1, player); return true; } } return false; } public static void setBlockDirectly(World world, int x, int y, int z,int id, int metadata) { int cX=x >>4; int cZ=z >>4; int cY=y >>4; Chunk chunk; int chunkX=(x % 16)< 0 ? ((x) % 16)+16 : ((x) % 16); int chunkY=y; int chunkZ=((z) % 16)< 0 ? ((z) % 16)+16 : ((z) % 16); // this.chunk=new EmptyChunk(world,cX, cZ); try { chunk=world.getChunkFromChunkCoords(cX, cZ); if (chunk.getBlockStorageArray()[cY] == null) { chunk.getBlockStorageArray()[cY] = new ExtendedBlockStorage(cY << 4, !world.provider.hasNoSky); } chunk.getBlockStorageArray()[cY].setExtBlockID(chunkX, (y) & 15, chunkZ, id); chunk.getBlockStorageArray()[cY].setExtBlockMetadata(chunkX, (y) & 15, chunkZ, metadata); } catch(Exception e) { e.printStackTrace(); } } public void addDimData(DimData dimData) { this.dimList.put(dimData.dimID, dimData); } public void createDimData(World world) { this.dimList.put(world.provider.dimensionId, new DimData(world.provider.dimensionId, false, 0,0,world.provider.getSpawnPoint().posX,world.provider.getSpawnPoint().posY,world.provider.getSpawnPoint().posZ)); } }