using System; using System.IO; using System.Collections.Generic; using Substrate.Core; namespace Substrate { /// /// Provides a wrapper around a physical Chunk stored in a chunk container. /// /// /// Modifying data in a ChunkRef will signal to the chunk container that the physical chunk needs to be saved. /// public class ChunkRef : IChunk { private IChunkContainer _container; private IChunk _chunk; private AlphaBlockCollection _blocks; private EntityCollection _entities; private int _cx; private int _cz; private bool _dirty; /// /// Gets the global X-coordinate of the chunk. /// public int X { get { return _container.ChunkGlobalX(_cx); } } /// /// Gets the global Z-coordinate of the chunk. /// public int Z { get { return _container.ChunkGlobalZ(_cz); } } /// /// Gets the local X-coordinate of the chunk within container. /// public int LocalX { get { return _container.ChunkLocalX(_cx); } } /// /// Gets the local Z-coordinate of the chunk within container. /// public int LocalZ { get { return _container.ChunkLocalZ(_cz); } } /// /// Gets the collection of all blocks and their data stored in the chunk. /// public AlphaBlockCollection Blocks { get { if (_blocks == null) { GetChunk(); } return _blocks; } } /// /// Gets the collection of all entities stored in the chunk. /// public EntityCollection Entities { get { if (_entities == null) { GetChunk(); } return _entities; } } /// /// Gets or sets the value indicating that the chunk has been modified, but not saved. /// public bool IsDirty { get { return _dirty || (_blocks != null && _blocks.IsDirty) || (_entities != null && _entities.IsDirty); } set { _dirty = value; if (_blocks != null) _blocks.IsDirty = false; if (_entities != null) _entities.IsDirty = false; } } /// /// Forbid direct instantiation of ChunkRef objects /// private ChunkRef () { } /// /// Create a reference to a chunk stored in a chunk container. /// /// Chunk container /// Local X-coordinate of chunk within container. /// Local Z-coordinate of chunk within container. /// ChunkRef representing a reference to a physical chunk at the specified location within the container. public static ChunkRef Create (IChunkContainer container, int cx, int cz) { if (!container.ChunkExists(cx, cz)) { return null; } ChunkRef c = new ChunkRef(); c._container = container; c._cx = cx; c._cz = cz; return c; } /// /// Gets or sets the chunk's TerrainPopulated status. /// public bool IsTerrainPopulated { get { return GetChunk().IsTerrainPopulated; } set { if (GetChunk().IsTerrainPopulated != value) { GetChunk().IsTerrainPopulated = value; _dirty = true; } } } /// /// Saves the underlying physical chunk to the specified output stream. /// /// An open output stream. /// A value indicating whether the chunk is no longer considered dirty. public bool Save (Stream outStream) { if (IsDirty) { if (GetChunk().Save(outStream)) { IsDirty = false; return true; } return false; } return true; } public void SetLocation (int x, int z) { int relX = LocalX + (x - X); int relZ = LocalZ + (z - Z); ChunkRef c = _container.SetChunk(relX, relZ, GetChunk()); _container = c._container; _cx = c._cx; _cz = c._cz; } /// /// Gets a ChunkRef to the chunk positioned immediately north (X - 1). /// /// ChunkRef to the northern neighboring chunk. public ChunkRef GetNorthNeighbor () { return _container.GetChunkRef(_cx - 1, _cz); } /// /// Gets a ChunkRef to the chunk positioned immediately south (X + 1). /// /// ChunkRef to the southern neighboring chunk. public ChunkRef GetSouthNeighbor () { return _container.GetChunkRef(_cx + 1, _cz); } /// /// Gets a ChunkRef to the chunk positioned immediatly east (Z - 1). /// /// ChunkRef to the eastern neighboring chunk. public ChunkRef GetEastNeighbor () { return _container.GetChunkRef(_cx, _cz - 1); } /// /// Gets a ChunkRef to the chunk positioned immedately west (Z + 1). /// /// ChunkRef to the western neighboring chunk. public ChunkRef GetWestNeighbor () { return _container.GetChunkRef(_cx, _cz + 1); } /// /// Returns a deep copy of the physical chunk underlying the ChunkRef. /// /// A copy of the physical Chunk object. /*public Chunk GetChunkCopy () { return GetChunk().Copy(); }*/ /// /// Returns the reference of the physical chunk underlying the ChunkRef, and releases the reference from itself. /// /// /// This function returns the reference to the chunk stored in the chunk container. Because the ChunkRef simultaneously gives up /// its "ownership" of the Chunk, the container will not consider the Chunk dirty even if it is modified. Attempting to use the ChunkRef after /// releasing its internal reference will query the container for a new reference. If the chunk is still cached, it will get the same reference /// back, otherwise it will get an independent copy. Chunks should only be taken from ChunkRefs to transfer them to another ChunkRef, or /// to modify them without intending to permanently store the changes. /// /// The physical Chunk object underlying the ChunkRef public IChunk GetChunkRef () { IChunk chunk = GetChunk(); _chunk = null; _dirty = false; return chunk; } /// /// Replaces the underlying physical chunk with a different one, updating its physical location to reflect the ChunkRef. /// /// /// Use this function to save chunks that have been created or manipulated independently of a container, or to /// move a physical chunk between locations within a container (by taking the reference from another ChunkRef). /// /// Physical Chunk to store into the location represented by this ChunkRef. public void SetChunkRef (IChunk chunk) { _chunk = chunk; _chunk.SetLocation(X, Z); _dirty = true; } /// /// Gets an internal Chunk reference from cache or queries the container for it. /// /// The ChunkRef's underlying Chunk. private IChunk GetChunk () { if (_chunk == null) { _chunk = _container.GetChunk(_cx, _cz); if (_chunk != null) { _blocks = _chunk.Blocks; _entities = _chunk.Entities; // Set callback functions in the underlying block collection _blocks.ResolveNeighbor += ResolveNeighborHandler; _blocks.TranslateCoordinates += TranslateCoordinatesHandler; } } return _chunk; } /// /// Callback function to return the block collection of a ChunkRef at a relative offset to this one. /// /// Relative offset from the X-coordinate. /// Relative offset from the Y-coordinate. /// Relative offset from the Z-coordinate. /// Another ChunkRef's underlying block collection, or null if the ChunkRef cannot be found. private AlphaBlockCollection ResolveNeighborHandler (int relx, int rely, int relz) { ChunkRef cr = _container.GetChunkRef(_cx + relx, _cz + relz); if (cr != null) { return cr.Blocks; } return null; } /// /// Translates chunk-local block coordinates to corresponding global coordinates. /// /// Chunk-local X-coordinate. /// Chunk-local Y-coordinate. /// Chunk-local Z-coordinate. /// BlockKey containing the global block coordinates. private BlockKey TranslateCoordinatesHandler (int lx, int ly, int lz) { int x = X * _blocks.XDim + lx; int z = Z * _blocks.ZDim + lz; return new BlockKey(x, ly, z); } } }