2011-04-06 04:43:54 +00:00
|
|
|
|
using System;
|
|
|
|
|
using System.IO;
|
2011-05-13 03:09:57 +00:00
|
|
|
|
using System.Collections.Generic;
|
2011-04-06 04:43:54 +00:00
|
|
|
|
|
2011-04-06 21:20:35 +00:00
|
|
|
|
namespace Substrate
|
2011-04-06 04:43:54 +00:00
|
|
|
|
{
|
|
|
|
|
using NBT;
|
|
|
|
|
using Utility;
|
|
|
|
|
|
2011-06-18 03:01:39 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// A Minecraft Alpha-compatible chunk data structure.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <remarks>
|
|
|
|
|
/// A Chunk internally wraps an NBT_Tree of raw chunk data. Modifying the chunk will update the tree, and vice-versa.
|
|
|
|
|
/// </remarks>
|
2011-04-06 04:43:54 +00:00
|
|
|
|
public class Chunk : IChunk, INBTObject<Chunk>, ICopyable<Chunk>
|
|
|
|
|
{
|
2011-05-13 03:09:57 +00:00
|
|
|
|
private const int XDIM = 16;
|
|
|
|
|
private const int YDIM = 128;
|
|
|
|
|
private const int ZDIM = 16;
|
|
|
|
|
|
2011-06-18 03:01:39 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// An NBT Schema definition for valid chunk data.
|
|
|
|
|
/// </summary>
|
2011-06-20 03:51:40 +00:00
|
|
|
|
public static SchemaNodeCompound LevelSchema = new SchemaNodeCompound()
|
2011-04-06 04:43:54 +00:00
|
|
|
|
{
|
2011-06-20 03:51:40 +00:00
|
|
|
|
new SchemaNodeCompound("Level")
|
2011-04-06 04:43:54 +00:00
|
|
|
|
{
|
2011-06-20 03:51:40 +00:00
|
|
|
|
new SchemaNodeArray("Blocks", 32768),
|
|
|
|
|
new SchemaNodeArray("Data", 16384),
|
|
|
|
|
new SchemaNodeArray("SkyLight", 16384),
|
|
|
|
|
new SchemaNodeArray("BlockLight", 16384),
|
|
|
|
|
new SchemaNodeArray("HeightMap", 256),
|
|
|
|
|
new SchemaNodeList("Entities", TagType.TAG_COMPOUND, 0, SchemaOptions.CREATE_ON_MISSING),
|
|
|
|
|
new SchemaNodeList("TileEntities", TagType.TAG_COMPOUND, TileEntity.BaseSchema, SchemaOptions.CREATE_ON_MISSING),
|
|
|
|
|
new SchemaNodeScaler("LastUpdate", TagType.TAG_LONG, SchemaOptions.CREATE_ON_MISSING),
|
|
|
|
|
new SchemaNodeScaler("xPos", TagType.TAG_INT),
|
|
|
|
|
new SchemaNodeScaler("zPos", TagType.TAG_INT),
|
|
|
|
|
new SchemaNodeScaler("TerrainPopulated", TagType.TAG_BYTE, SchemaOptions.CREATE_ON_MISSING),
|
2011-04-06 04:43:54 +00:00
|
|
|
|
},
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
private NBT_Tree _tree;
|
|
|
|
|
|
|
|
|
|
private int _cx;
|
|
|
|
|
private int _cz;
|
|
|
|
|
|
2011-05-13 03:09:57 +00:00
|
|
|
|
private XZYByteArray _blocks;
|
|
|
|
|
private XZYNibbleArray _data;
|
|
|
|
|
private XZYNibbleArray _blockLight;
|
|
|
|
|
private XZYNibbleArray _skyLight;
|
|
|
|
|
private ZXByteArray _heightMap;
|
2011-04-06 04:43:54 +00:00
|
|
|
|
|
2011-06-20 03:51:40 +00:00
|
|
|
|
private TagNodeList _entities;
|
|
|
|
|
private TagNodeList _tileEntities;
|
2011-04-06 04:43:54 +00:00
|
|
|
|
|
2011-05-13 03:09:57 +00:00
|
|
|
|
private AlphaBlockCollection _blockManager;
|
|
|
|
|
private EntityCollection _entityManager;
|
2011-04-06 04:43:54 +00:00
|
|
|
|
|
2011-06-18 03:01:39 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Gets the global X-coordinate of the chunk.
|
|
|
|
|
/// </summary>
|
2011-04-06 04:43:54 +00:00
|
|
|
|
public int X
|
|
|
|
|
{
|
|
|
|
|
get { return _cx; }
|
|
|
|
|
}
|
|
|
|
|
|
2011-06-18 03:01:39 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Gets the global Z-coordinate of the chunk.
|
|
|
|
|
/// </summary>
|
2011-04-06 04:43:54 +00:00
|
|
|
|
public int Z
|
|
|
|
|
{
|
|
|
|
|
get { return _cz; }
|
|
|
|
|
}
|
|
|
|
|
|
2011-06-18 03:01:39 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Gets the collection of all blocks and their data stored in the chunk.
|
|
|
|
|
/// </summary>
|
2011-05-13 03:09:57 +00:00
|
|
|
|
public AlphaBlockCollection Blocks
|
|
|
|
|
{
|
|
|
|
|
get { return _blockManager; }
|
|
|
|
|
}
|
|
|
|
|
|
2011-06-18 03:01:39 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Gets the collection of all entities stored in the chunk.
|
|
|
|
|
/// </summary>
|
2011-05-13 03:09:57 +00:00
|
|
|
|
public EntityCollection Entities
|
|
|
|
|
{
|
|
|
|
|
get { return _entityManager; }
|
|
|
|
|
}
|
|
|
|
|
|
2011-06-18 03:01:39 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Provides raw access to the underlying NBT_Tree.
|
|
|
|
|
/// </summary>
|
2011-04-06 04:43:54 +00:00
|
|
|
|
public NBT_Tree Tree
|
|
|
|
|
{
|
|
|
|
|
get { return _tree; }
|
|
|
|
|
}
|
|
|
|
|
|
2011-06-18 03:01:39 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Gets or sets the chunk's TerrainPopulated status.
|
|
|
|
|
/// </summary>
|
2011-04-06 04:43:54 +00:00
|
|
|
|
public bool IsTerrainPopulated
|
|
|
|
|
{
|
2011-04-09 02:50:42 +00:00
|
|
|
|
get { return _tree.Root["Level"].ToTagCompound()["TerrainPopulated"].ToTagByte() == 1; }
|
|
|
|
|
set { _tree.Root["Level"].ToTagCompound()["TerrainPopulated"].ToTagByte().Data = (byte)(value ? 1 : 0); }
|
2011-04-06 04:43:54 +00:00
|
|
|
|
}
|
|
|
|
|
|
2011-06-18 03:01:39 +00:00
|
|
|
|
|
2011-04-19 07:50:17 +00:00
|
|
|
|
private Chunk ()
|
2011-04-06 04:43:54 +00:00
|
|
|
|
{
|
2011-04-19 07:50:17 +00:00
|
|
|
|
}
|
2011-04-06 04:43:54 +00:00
|
|
|
|
|
2011-06-18 03:01:39 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Creates a default (empty) chunk.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="x">Global X-coordinate of the chunk.</param>
|
|
|
|
|
/// <param name="z">Global Z-coordinate of the chunk.</param>
|
|
|
|
|
/// <returns>A new Chunk object.</returns>
|
2011-04-19 07:50:17 +00:00
|
|
|
|
public static Chunk Create (int x, int z)
|
|
|
|
|
{
|
|
|
|
|
Chunk c = new Chunk();
|
2011-04-06 04:43:54 +00:00
|
|
|
|
|
2011-04-19 07:50:17 +00:00
|
|
|
|
c._cx = x;
|
|
|
|
|
c._cz = z;
|
|
|
|
|
|
|
|
|
|
c.BuildNBTTree();
|
|
|
|
|
|
|
|
|
|
return c;
|
2011-04-06 04:43:54 +00:00
|
|
|
|
}
|
|
|
|
|
|
2011-06-18 03:01:39 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Creates a chunk object from an existing NBT_Tree.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="tree">An NBT_Tree conforming to the chunk schema definition.</param>
|
|
|
|
|
/// <returns>A new Chunk object wrapping an existing NBT_Tree.</returns>
|
2011-04-19 07:50:17 +00:00
|
|
|
|
public static Chunk Create (NBT_Tree tree)
|
2011-04-06 04:43:54 +00:00
|
|
|
|
{
|
2011-04-19 07:50:17 +00:00
|
|
|
|
Chunk c = new Chunk();
|
|
|
|
|
|
|
|
|
|
return c.LoadTree(tree.Root);
|
|
|
|
|
}
|
|
|
|
|
|
2011-06-18 03:01:39 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Creates a chunk object from a verified NBT_Tree.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="tree">An NBT_Tree conforming to the chunk schema definition.</param>
|
|
|
|
|
/// <returns>A new Chunk object wrapping an existing NBT_Tree, or null on verification failure.</returns>
|
2011-04-19 07:50:17 +00:00
|
|
|
|
public static Chunk CreateVerified (NBT_Tree tree)
|
|
|
|
|
{
|
|
|
|
|
Chunk c = new Chunk();
|
|
|
|
|
|
|
|
|
|
return c.LoadTreeSafe(tree.Root);
|
2011-04-06 04:43:54 +00:00
|
|
|
|
}
|
|
|
|
|
|
2011-06-18 03:01:39 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Updates the chunk's global world coordinates.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="x">Global X-coordinate.</param>
|
|
|
|
|
/// <param name="z">Global Z-coordinate.</param>
|
2011-05-13 03:09:57 +00:00
|
|
|
|
public virtual void SetLocation (int x, int z)
|
2011-04-06 04:43:54 +00:00
|
|
|
|
{
|
2011-05-13 03:09:57 +00:00
|
|
|
|
_cx = x;
|
|
|
|
|
_cz = z;
|
2011-04-06 04:43:54 +00:00
|
|
|
|
}
|
|
|
|
|
|
2011-06-18 03:01:39 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Saves a Chunk's underlying NBT_Tree to an output stream.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="outStream">An open, writable output stream.</param>
|
|
|
|
|
/// <returns>True if the data is written out to the stream.</returns>
|
2011-04-06 04:43:54 +00:00
|
|
|
|
public bool Save (Stream outStream)
|
|
|
|
|
{
|
|
|
|
|
if (outStream == null || !outStream.CanWrite) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_tree.WriteTo(outStream);
|
|
|
|
|
outStream.Close();
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2011-04-06 23:54:41 +00:00
|
|
|
|
|
2011-04-06 04:43:54 +00:00
|
|
|
|
#region INBTObject<Chunk> Members
|
|
|
|
|
|
2011-06-18 03:01:39 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Loads the Chunk from an NBT tree rooted at the given TagValue node.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="tree">Root node of an NBT tree.</param>
|
|
|
|
|
/// <returns>A reference to the current Chunk, or null if the tree is unparsable.</returns>
|
2011-06-20 03:51:40 +00:00
|
|
|
|
public Chunk LoadTree (TagNode tree)
|
2011-04-06 04:43:54 +00:00
|
|
|
|
{
|
2011-06-20 03:51:40 +00:00
|
|
|
|
TagNodeCompound ctree = tree as TagNodeCompound;
|
2011-04-06 04:43:54 +00:00
|
|
|
|
if (ctree == null) {
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_tree = new NBT_Tree(ctree);
|
2011-05-13 03:09:57 +00:00
|
|
|
|
|
2011-06-20 03:51:40 +00:00
|
|
|
|
TagNodeCompound level = _tree.Root["Level"] as TagNodeCompound;
|
2011-04-06 04:43:54 +00:00
|
|
|
|
|
2011-06-20 03:51:40 +00:00
|
|
|
|
_blocks = new XZYByteArray(XDIM, YDIM, ZDIM, level["Blocks"] as TagNodeByteArray);
|
|
|
|
|
_data = new XZYNibbleArray(XDIM, YDIM, ZDIM, level["Data"] as TagNodeByteArray);
|
|
|
|
|
_blockLight = new XZYNibbleArray(XDIM, YDIM, ZDIM, level["BlockLight"] as TagNodeByteArray);
|
|
|
|
|
_skyLight = new XZYNibbleArray(XDIM, YDIM, ZDIM, level["SkyLight"] as TagNodeByteArray);
|
|
|
|
|
_heightMap = new ZXByteArray(XDIM, ZDIM, level["HeightMap"] as TagNodeByteArray);
|
2011-04-06 04:43:54 +00:00
|
|
|
|
|
2011-06-20 03:51:40 +00:00
|
|
|
|
_entities = level["Entities"] as TagNodeList;
|
|
|
|
|
_tileEntities = level["TileEntities"] as TagNodeList;
|
2011-04-06 04:43:54 +00:00
|
|
|
|
|
|
|
|
|
// List-type patch up
|
|
|
|
|
if (_entities.Count == 0) {
|
2011-06-20 03:51:40 +00:00
|
|
|
|
level["Entities"] = new TagNodeList(TagType.TAG_COMPOUND);
|
|
|
|
|
_entities = level["Entities"] as TagNodeList;
|
2011-04-06 04:43:54 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (_tileEntities.Count == 0) {
|
2011-06-20 03:51:40 +00:00
|
|
|
|
level["TileEntities"] = new TagNodeList(TagType.TAG_COMPOUND);
|
|
|
|
|
_tileEntities = level["TileEntities"] as TagNodeList;
|
2011-04-06 04:43:54 +00:00
|
|
|
|
}
|
|
|
|
|
|
2011-04-09 02:50:42 +00:00
|
|
|
|
_cx = level["xPos"].ToTagInt();
|
|
|
|
|
_cz = level["zPos"].ToTagInt();
|
2011-04-06 04:43:54 +00:00
|
|
|
|
|
2011-06-27 04:49:29 +00:00
|
|
|
|
_blockManager = new AlphaBlockCollection(_blocks, _data, _blockLight, _skyLight, _heightMap, _tileEntities);
|
2011-05-13 03:09:57 +00:00
|
|
|
|
_entityManager = new EntityCollection(_entities);
|
2011-04-06 04:43:54 +00:00
|
|
|
|
|
|
|
|
|
return this;
|
|
|
|
|
}
|
|
|
|
|
|
2011-06-18 03:01:39 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Loads the Chunk from a validated NBT tree rooted at the given TagValue node.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="tree">Root node of an NBT tree.</param>
|
|
|
|
|
/// <returns>A reference to the current Chunk, or null if the tree does not conform to the chunk's NBT Schema definition.</returns>
|
2011-06-20 03:51:40 +00:00
|
|
|
|
public Chunk LoadTreeSafe (TagNode tree)
|
2011-04-06 04:43:54 +00:00
|
|
|
|
{
|
|
|
|
|
if (!ValidateTree(tree)) {
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return LoadTree(tree);
|
|
|
|
|
}
|
|
|
|
|
|
2011-06-18 03:01:39 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Gets a valid NBT tree representing the Chunk.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <returns>The root node of the Chunk's NBT tree.</returns>
|
2011-06-20 03:51:40 +00:00
|
|
|
|
public TagNode BuildTree ()
|
2011-04-06 04:43:54 +00:00
|
|
|
|
{
|
|
|
|
|
return _tree.Root;
|
|
|
|
|
}
|
|
|
|
|
|
2011-06-18 03:01:39 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Validates an NBT tree against the chunk's NBT schema definition.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="tree">The root node of the NBT tree to verify.</param>
|
|
|
|
|
/// <returns>Status indicating if the tree represents a valid chunk.</returns>
|
2011-06-20 03:51:40 +00:00
|
|
|
|
public bool ValidateTree (TagNode tree)
|
2011-04-06 04:43:54 +00:00
|
|
|
|
{
|
2011-04-16 01:46:46 +00:00
|
|
|
|
NBTVerifier v = new NBTVerifier(tree, LevelSchema);
|
|
|
|
|
return v.Verify();
|
2011-04-06 04:43:54 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#endregion
|
2011-04-06 23:54:41 +00:00
|
|
|
|
|
|
|
|
|
|
2011-05-13 03:09:57 +00:00
|
|
|
|
#region ICopyable<Chunk> Members
|
2011-04-06 23:54:41 +00:00
|
|
|
|
|
2011-06-18 03:01:39 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Creates a deep copy of the Chunk and its underlying NBT tree.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <returns>A new Chunk with copied data.</returns>
|
2011-05-13 03:09:57 +00:00
|
|
|
|
public Chunk Copy ()
|
2011-04-06 23:54:41 +00:00
|
|
|
|
{
|
2011-05-13 03:09:57 +00:00
|
|
|
|
return Chunk.Create(_tree.Copy());
|
2011-04-06 23:54:41 +00:00
|
|
|
|
}
|
|
|
|
|
|
2011-05-13 03:09:57 +00:00
|
|
|
|
#endregion
|
2011-04-06 23:54:41 +00:00
|
|
|
|
|
|
|
|
|
|
2011-05-13 03:09:57 +00:00
|
|
|
|
private void BuildNBTTree ()
|
2011-04-06 23:54:41 +00:00
|
|
|
|
{
|
2011-05-13 03:09:57 +00:00
|
|
|
|
int elements2 = XDIM * ZDIM;
|
|
|
|
|
int elements3 = elements2 * YDIM;
|
2011-04-06 23:54:41 +00:00
|
|
|
|
|
2011-06-20 03:51:40 +00:00
|
|
|
|
TagNodeByteArray blocks = new TagNodeByteArray(new byte[elements3]);
|
|
|
|
|
TagNodeByteArray data = new TagNodeByteArray(new byte[elements3 >> 1]);
|
|
|
|
|
TagNodeByteArray blocklight = new TagNodeByteArray(new byte[elements3 >> 1]);
|
|
|
|
|
TagNodeByteArray skylight = new TagNodeByteArray(new byte[elements3 >> 1]);
|
|
|
|
|
TagNodeByteArray heightMap = new TagNodeByteArray(new byte[elements2]);
|
2011-04-06 23:54:41 +00:00
|
|
|
|
|
2011-05-13 03:09:57 +00:00
|
|
|
|
_blocks = new XZYByteArray(XDIM, YDIM, ZDIM, blocks);
|
|
|
|
|
_data = new XZYNibbleArray(XDIM, YDIM, ZDIM, data);
|
|
|
|
|
_blockLight = new XZYNibbleArray(XDIM, YDIM, ZDIM, blocklight);
|
|
|
|
|
_skyLight = new XZYNibbleArray(XDIM, YDIM, ZDIM, skylight);
|
|
|
|
|
_heightMap = new ZXByteArray(XDIM, ZDIM, heightMap);
|
2011-04-06 23:54:41 +00:00
|
|
|
|
|
2011-06-20 03:51:40 +00:00
|
|
|
|
_entities = new TagNodeList(TagType.TAG_COMPOUND);
|
|
|
|
|
_tileEntities = new TagNodeList(TagType.TAG_COMPOUND);
|
2011-04-06 23:54:41 +00:00
|
|
|
|
|
2011-06-20 03:51:40 +00:00
|
|
|
|
TagNodeCompound level = new TagNodeCompound();
|
2011-05-13 03:09:57 +00:00
|
|
|
|
level.Add("Blocks", blocks);
|
|
|
|
|
level.Add("Data", data);
|
|
|
|
|
level.Add("SkyLight", blocklight);
|
|
|
|
|
level.Add("BlockLight", skylight);
|
|
|
|
|
level.Add("HeightMap", heightMap);
|
|
|
|
|
level.Add("Entities", _entities);
|
|
|
|
|
level.Add("TileEntities", _tileEntities);
|
2011-06-20 03:51:40 +00:00
|
|
|
|
level.Add("LastUpdate", new TagNodeLong(Timestamp()));
|
|
|
|
|
level.Add("xPos", new TagNodeInt(_cx));
|
|
|
|
|
level.Add("zPos", new TagNodeInt(_cz));
|
|
|
|
|
level.Add("TerrainPopulated", new TagNodeByte());
|
2011-04-11 06:13:33 +00:00
|
|
|
|
|
2011-05-13 03:09:57 +00:00
|
|
|
|
_tree = new NBT_Tree();
|
|
|
|
|
_tree.Root.Add("Level", level);
|
2011-04-21 00:30:13 +00:00
|
|
|
|
|
2011-06-27 04:49:29 +00:00
|
|
|
|
_blockManager = new AlphaBlockCollection(_blocks, _data, _blockLight, _skyLight, _heightMap, _tileEntities);
|
2011-05-13 03:09:57 +00:00
|
|
|
|
_entityManager = new EntityCollection(_entities);
|
2011-04-14 02:30:40 +00:00
|
|
|
|
}
|
|
|
|
|
|
2011-04-13 08:46:06 +00:00
|
|
|
|
private int Timestamp ()
|
|
|
|
|
{
|
|
|
|
|
DateTime epoch = new DateTime(1970, 1, 1, 0, 0, 0, 0);
|
|
|
|
|
return (int)((DateTime.UtcNow - epoch).Ticks / (10000L * 1000L));
|
|
|
|
|
}
|
2011-04-06 04:43:54 +00:00
|
|
|
|
}
|
|
|
|
|
}
|