2012-02-08 07:36:54 -05:00
|
|
|
|
2012-02-13 16:47:03 -05:00
|
|
|
// cChunkMap.h
|
|
|
|
|
|
|
|
// Interfaces to the cChunkMap class representing the chunk storage for a single world
|
|
|
|
|
2011-10-03 14:41:19 -04:00
|
|
|
#pragma once
|
|
|
|
|
2012-02-13 16:47:03 -05:00
|
|
|
#include "cChunk.h"
|
|
|
|
|
2012-02-08 07:36:54 -05:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2011-10-30 20:52:20 -04:00
|
|
|
class cWorld;
|
2011-10-03 14:41:19 -04:00
|
|
|
class cEntity;
|
2012-02-08 07:36:54 -05:00
|
|
|
class MTRand;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2011-10-03 14:41:19 -04:00
|
|
|
class cChunkMap
|
|
|
|
{
|
|
|
|
public:
|
2012-02-13 16:47:03 -05:00
|
|
|
|
|
|
|
static const int LAYER_SIZE = 32;
|
|
|
|
|
2012-01-29 16:40:21 -05:00
|
|
|
cChunkMap(cWorld* a_World );
|
2011-10-03 14:41:19 -04:00
|
|
|
~cChunkMap();
|
|
|
|
|
2012-02-16 08:42:35 -05:00
|
|
|
// TODO: Get rid of these (put into Private section) in favor of the direct action methods:
|
2012-02-14 16:09:14 -05:00
|
|
|
cChunkPtr GetChunk ( int a_ChunkX, int a_ChunkY, int a_ChunkZ ); // Also queues the chunk for loading / generating if not valid
|
|
|
|
cChunkPtr GetChunkNoGen( int a_ChunkX, int a_ChunkY, int a_ChunkZ ); // Also queues the chunk for loading if not valid; doesn't generate
|
2012-02-15 09:22:44 -05:00
|
|
|
|
|
|
|
// Direct action methods:
|
2012-02-20 11:39:00 -05:00
|
|
|
/// Broadcast a_Packet to all clients in the chunk specified
|
|
|
|
void BroadcastToChunk(int a_ChunkX, int a_ChunkY, int a_ChunkZ, cPacket & a_Packet, cClientHandle * a_Exclude = NULL);
|
|
|
|
|
2012-02-15 09:22:44 -05:00
|
|
|
/// Broadcasts a_Packet to all clients in the chunk where block [x, y, z] is, except to client a_Exclude
|
|
|
|
void BroadcastToChunkOfBlock(int a_X, int a_Y, int a_Z, cPacket * a_Packet, cClientHandle * a_Exclude = NULL);
|
2012-02-20 11:39:00 -05:00
|
|
|
|
|
|
|
/// a_Player rclked block entity at the coords specified, handle it
|
|
|
|
void UseBlockEntity(cPlayer * a_Player, int a_X, int a_Y, int a_Z);
|
2011-10-03 14:41:19 -04:00
|
|
|
|
2012-02-18 12:53:22 -05:00
|
|
|
void MarkChunkDirty (int a_ChunkX, int a_ChunkY, int a_ChunkZ);
|
|
|
|
void MarkChunkSaving (int a_ChunkX, int a_ChunkY, int a_ChunkZ);
|
|
|
|
void MarkChunkSaved (int a_ChunkX, int a_ChunkY, int a_ChunkZ);
|
|
|
|
void ChunkDataLoaded (int a_ChunkX, int a_ChunkY, int a_ChunkZ, const char * a_BlockData, cEntityList & a_Entities, cBlockEntityList & a_BlockEntities);
|
|
|
|
void ChunkDataGenerated (int a_ChunkX, int a_ChunkY, int a_ChunkZ, const char * a_BlockData, cEntityList & a_Entities, cBlockEntityList & a_BlockEntities);
|
|
|
|
void GetChunkData (int a_ChunkX, int a_ChunkY, int a_ChunkZ, cChunkDataCallback * a_Callback);
|
2012-02-18 14:18:16 -05:00
|
|
|
bool GetChunkBlocks (int a_ChunkX, int a_ChunkY, int a_ChunkZ, char * a_Blocks);
|
2012-02-18 12:53:22 -05:00
|
|
|
bool IsChunkValid (int a_ChunkX, int a_ChunkY, int a_ChunkZ);
|
|
|
|
bool HasChunkAnyClients (int a_ChunkX, int a_ChunkY, int a_ChunkZ);
|
|
|
|
void SpreadChunkLighting(int a_ChunkX, int a_ChunkY, int a_ChunkZ);
|
|
|
|
int GetHeight (int a_BlockX, int a_BlockZ);
|
2012-02-18 15:10:57 -05:00
|
|
|
void FastSetBlocks (sSetBlockList & a_BlockList);
|
2012-02-20 11:39:00 -05:00
|
|
|
void CollectPickupsByPlayer(cPlayer * a_Player);
|
|
|
|
|
|
|
|
/// Compares clients of two chunks, calls the callback accordingly
|
|
|
|
void CompareChunkClients(int a_ChunkX1, int a_ChunkY1, int a_ChunkZ1, int a_ChunkX2, int a_ChunkY2, int a_ChunkZ2, cClientDiffCallback & a_Callback);
|
|
|
|
|
|
|
|
/// Moves the entity from its current chunk to the new chunk specified
|
|
|
|
void MoveEntityToChunk(cEntity * a_Entity, int a_ChunkX, int a_ChunkY, int a_ChunkZ);
|
|
|
|
|
|
|
|
/// Removes the entity from the chunk specified
|
|
|
|
void RemoveEntityFromChunk(cEntity * a_Entity, int a_ChunkX, int a_ChunkY, int a_ChunkZ);
|
2012-02-16 08:42:35 -05:00
|
|
|
|
2012-02-08 07:36:54 -05:00
|
|
|
void Tick( float a_Dt, MTRand & a_TickRand );
|
2011-10-03 14:41:19 -04:00
|
|
|
|
|
|
|
void UnloadUnusedChunks();
|
|
|
|
void SaveAllChunks();
|
2011-12-24 18:34:30 -05:00
|
|
|
|
2012-02-15 09:22:44 -05:00
|
|
|
cWorld * GetWorld() { return m_World; }
|
2012-01-01 11:20:52 -05:00
|
|
|
|
2012-02-13 16:47:03 -05:00
|
|
|
int GetNumChunks(void);
|
2012-02-08 07:36:54 -05:00
|
|
|
|
2012-02-15 09:22:44 -05:00
|
|
|
/// Converts absolute block coords into relative (chunk + block) coords:
|
|
|
|
inline static void AbsoluteToRelative(/* in-out */ int & a_X, int & a_Y, int & a_Z, /* out */ int & a_ChunkX, int & a_ChunkZ )
|
|
|
|
{
|
|
|
|
BlockToChunk(a_X, a_Y, a_Z, a_ChunkX, a_ChunkZ);
|
|
|
|
|
|
|
|
a_X = a_X - a_ChunkX * 16;
|
2012-02-16 08:42:35 -05:00
|
|
|
a_Z = a_Z - a_ChunkZ * 16;
|
2012-02-15 09:22:44 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Converts absolute block coords to chunk coords:
|
|
|
|
inline static void BlockToChunk( int a_X, int a_Y, int a_Z, int & a_ChunkX, int & a_ChunkZ )
|
|
|
|
{
|
|
|
|
(void)a_Y;
|
|
|
|
a_ChunkX = a_X / 16;
|
|
|
|
if ((a_X < 0) && (a_X % 16 != 0))
|
|
|
|
{
|
|
|
|
a_ChunkX--;
|
|
|
|
}
|
|
|
|
a_ChunkZ = a_Z / 16;
|
|
|
|
if ((a_Z < 0) && (a_Z % 16 != 0))
|
|
|
|
{
|
|
|
|
a_ChunkZ--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-02-18 12:53:22 -05:00
|
|
|
void ChunkValidated(void); // Called by chunks that have become valid
|
2012-02-15 09:22:44 -05:00
|
|
|
|
2011-10-03 14:41:19 -04:00
|
|
|
private:
|
2012-02-08 07:36:54 -05:00
|
|
|
|
2011-10-03 14:41:19 -04:00
|
|
|
class cChunkLayer
|
|
|
|
{
|
|
|
|
public:
|
2012-02-13 16:47:03 -05:00
|
|
|
cChunkLayer(int a_LayerX, int a_LayerZ, cChunkMap * a_Parent);
|
|
|
|
|
|
|
|
/// Always returns an assigned chunkptr, but the chunk needn't be valid (loaded / generated) - callers must check
|
2012-02-17 12:56:25 -05:00
|
|
|
cChunkPtr GetChunk( int a_ChunkX, int a_ChunkY, int a_ChunkZ );
|
2012-02-13 16:47:03 -05:00
|
|
|
|
|
|
|
int GetX(void) const {return m_LayerX; }
|
|
|
|
int GetZ(void) const {return m_LayerZ; }
|
2012-02-17 12:56:25 -05:00
|
|
|
|
|
|
|
int GetNumChunksLoaded(void) const ;
|
2012-01-30 08:54:39 -05:00
|
|
|
|
2012-02-13 16:47:03 -05:00
|
|
|
void Save(void);
|
|
|
|
void UnloadUnusedChunks(void);
|
|
|
|
|
|
|
|
void Tick( float a_Dt, MTRand & a_TickRand );
|
|
|
|
|
|
|
|
protected:
|
|
|
|
|
|
|
|
cChunkPtr m_Chunks[LAYER_SIZE * LAYER_SIZE];
|
|
|
|
int m_LayerX;
|
|
|
|
int m_LayerZ;
|
|
|
|
cChunkMap * m_Parent;
|
2011-10-03 14:41:19 -04:00
|
|
|
int m_NumChunksLoaded;
|
|
|
|
};
|
2012-02-13 16:47:03 -05:00
|
|
|
|
|
|
|
typedef std::list<cChunkLayer *> cChunkLayerList;
|
|
|
|
// TODO: Use smart pointers for cChunkLayerList as well, so that ticking and saving needn't lock the entire layerlist
|
|
|
|
// This however means that cChunkLayer needs to interlock its m_Chunks[]
|
2011-10-03 14:41:19 -04:00
|
|
|
|
2012-02-13 16:47:03 -05:00
|
|
|
cChunkLayer * GetLayerForChunk( int a_ChunkX, int a_ChunkZ ); // Creates the layer if it doesn't already exist
|
|
|
|
cChunkLayer * GetLayer( int a_LayerX, int a_LayerZ ); // Creates the layer if it doesn't already exist
|
|
|
|
void RemoveLayer( cChunkLayer* a_Layer );
|
2011-10-03 14:41:19 -04:00
|
|
|
|
2012-02-13 16:47:03 -05:00
|
|
|
cCriticalSection m_CSLayers;
|
|
|
|
cChunkLayerList m_Layers;
|
2012-02-18 12:53:22 -05:00
|
|
|
cEvent m_evtChunkValid; // Set whenever any chunk becomes valid, via ChunkValidated()
|
2011-10-03 14:41:19 -04:00
|
|
|
|
2012-02-13 16:47:03 -05:00
|
|
|
cWorld * m_World;
|
2011-10-29 17:19:06 -04:00
|
|
|
};
|
2012-02-08 07:36:54 -05:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|