2012-06-14 09:06:06 -04:00
# pragma once
# ifndef _WIN32
# include "BlockID.h"
# else
enum ENUM_ITEM_ID ;
# endif
# define MAX_PLAYERS 65535
2012-10-13 05:53:28 -04:00
# include "Simulator/SimulatorManager.h"
2012-06-14 09:06:06 -04:00
# include "MersenneTwister.h"
2012-09-23 18:09:57 -04:00
# include "ChunkMap.h"
2012-09-23 16:22:46 -04:00
# include "WorldStorage/WorldStorage.h"
2012-09-23 16:14:04 -04:00
# include "Generating/ChunkGenerator.h"
2012-06-14 09:06:06 -04:00
# include "Vector3i.h"
# include "Vector3f.h"
# include "ChunkSender.h"
# include "Defines.h"
# include "LightingThread.h"
2012-09-23 18:09:57 -04:00
# include "Item.h"
2012-06-14 09:06:06 -04:00
class cRedstone ;
class cFireSimulator ;
2012-10-13 05:53:28 -04:00
class cFluidSimulator ;
2012-06-14 09:06:06 -04:00
class cSandSimulator ;
class cRedstoneSimulator ;
class cItem ;
class cPlayer ;
class cClientHandle ;
class cEntity ;
class cBlockEntity ;
class cWorldGenerator ; // The generator that actually generates the chunks for a single world
class cChunkGenerator ; // The thread responsible for generating chunks
2012-06-17 15:58:39 -04:00
class cChestEntity ;
2012-12-19 16:19:36 -05:00
class cDispenserEntity ;
2012-06-17 15:58:39 -04:00
class cFurnaceEntity ;
2012-06-14 09:06:06 -04:00
typedef std : : list < cPlayer * > cPlayerList ;
2012-06-17 15:58:39 -04:00
2012-12-19 16:19:36 -05:00
typedef cItemCallback < cPlayer > cPlayerListCallback ;
typedef cItemCallback < cEntity > cEntityCallback ;
typedef cItemCallback < cChestEntity > cChestCallback ;
typedef cItemCallback < cDispenserEntity > cDispenserCallback ;
typedef cItemCallback < cFurnaceEntity > cFurnaceCallback ;
2012-06-17 15:58:39 -04:00
2012-06-14 09:06:06 -04:00
2013-03-11 13:15:34 -04:00
// tolua_begin
class cWorld
{
2012-06-14 09:06:06 -04:00
public :
2013-06-15 11:29:20 -04:00
// tolua_end
/// A simple RAII locker for the chunkmap - locks the chunkmap in its constructor, unlocks it in the destructor
class cLock :
public cCSLock
{
typedef cCSLock super ;
public :
cLock ( cWorld & a_World ) ;
} ;
// tolua_begin
2013-02-02 18:55:29 -05:00
static const char * GetClassStatic ( void )
{
return " cWorld " ;
}
2012-11-01 17:38:20 -04:00
/// Return time in seconds
inline static float GetTime ( void )
2012-06-14 09:06:06 -04:00
{
2012-11-01 17:38:20 -04:00
LOGWARNING ( " cWorld:GetTime() is obsolete, use GetWorldAge() or GetTimeOfDay() for a specific world instead. " ) ;
return 0 ;
2012-06-14 09:06:06 -04:00
}
2012-11-01 17:38:20 -04:00
2013-06-09 13:38:10 -04:00
int GetTicksUntilWeatherChange ( void ) const { return m_WeatherInterval ; }
2012-11-01 17:38:20 -04:00
Int64 GetWorldAge ( void ) const { return m_WorldAge ; }
Int64 GetTimeOfDay ( void ) const { return m_TimeOfDay ; }
2013-06-09 13:38:10 -04:00
void SetTicksUntilWeatherChange ( int a_WeatherInterval )
{
m_WeatherInterval = a_WeatherInterval ;
}
2012-11-01 17:38:20 -04:00
void SetTimeOfDay ( Int64 a_TimeOfDay )
{
m_TimeOfDay = a_TimeOfDay ;
m_TimeOfDaySecs = ( double ) a_TimeOfDay / 20.0 ;
BroadcastTimeUpdate ( ) ;
}
void SetWorldTime ( Int64 a_TimeOfDay )
{
LOGWARNING ( " cWorld:SetWorldTime() is obsolete, use SetTimeOfDay() instead " ) ;
SetTimeOfDay ( a_TimeOfDay ) ;
}
eGameMode GetGameMode ( void ) const { return m_GameMode ; }
bool IsPVPEnabled ( void ) const { return m_bEnabledPVP ; }
2012-10-21 17:15:57 -04:00
bool IsDeepSnowEnabled ( void ) const { return m_IsDeepSnowEnabled ; }
2013-03-11 13:15:34 -04:00
eDimension GetDimension ( void ) const { return m_Dimension ; }
2012-06-14 09:06:06 -04:00
2013-04-13 17:02:10 -04:00
/// Returns the world height at the specified coords; waits for the chunk to get loaded / generated
2012-11-01 17:38:20 -04:00
int GetHeight ( int a_BlockX , int a_BlockZ ) ;
// tolua_end
2012-06-14 09:06:06 -04:00
2013-04-13 17:02:10 -04:00
/// Retrieves the world height at the specified coords; returns false if chunk not loaded / generated
bool TryGetHeight ( int a_BlockX , int a_BlockZ , int & a_Height ) ; // TODO: Export in ManualBindings.cpp
2013-03-03 14:05:11 -05:00
void BroadcastAttachEntity ( const cEntity & a_Entity , const cEntity * a_Vehicle ) ;
2012-10-21 03:46:28 -04:00
void BroadcastChat ( const AString & a_Message , const cClientHandle * a_Exclude = NULL ) ;
void BroadcastPlayerAnimation ( const cPlayer & a_Player , char a_Animation , const cClientHandle * a_Exclude = NULL ) ;
void BroadcastEntityEquipment ( const cEntity & a_Entity , short a_SlotNum , const cItem & a_Item , const cClientHandle * a_Exclude = NULL ) ;
2013-03-17 22:51:55 -04:00
void BroadcastEntVelocity ( const cEntity & a_Entity , const cClientHandle * a_Exclude = NULL ) ;
2012-10-21 03:46:28 -04:00
void BroadcastTeleportEntity ( const cEntity & a_Entity , const cClientHandle * a_Exclude = NULL ) ;
void BroadcastEntRelMoveLook ( const cEntity & a_Entity , char a_RelX , char a_RelY , char a_RelZ , const cClientHandle * a_Exclude = NULL ) ;
void BroadcastEntRelMove ( const cEntity & a_Entity , char a_RelX , char a_RelY , char a_RelZ , const cClientHandle * a_Exclude = NULL ) ;
void BroadcastEntLook ( const cEntity & a_Entity , const cClientHandle * a_Exclude = NULL ) ;
void BroadcastEntHeadLook ( const cEntity & a_Entity , const cClientHandle * a_Exclude = NULL ) ;
void BroadcastBlockAction ( int a_BlockX , int a_BlockY , int a_BlockZ , char a_Byte1 , char a_Byte2 , BLOCKTYPE a_BlockType , const cClientHandle * a_Exclude = NULL ) ;
void BroadcastDestroyEntity ( const cEntity & a_Entity , const cClientHandle * a_Exclude = NULL ) ;
void BroadcastEntityStatus ( const cEntity & a_Entity , char a_Status , const cClientHandle * a_Exclude = NULL ) ;
void BroadcastMetadata ( const cPawn & a_Pawn , const cClientHandle * a_Exclude = NULL ) ;
void BroadcastSpawn ( cEntity & a_Entity , const cClientHandle * a_Exclude = NULL ) ;
void BroadcastCollectPickup ( const cPickup & a_Pickup , const cPlayer & a_Player , const cClientHandle * a_Exclude = NULL ) ;
void BroadcastWeather ( eWeather a_Weather , const cClientHandle * a_Exclude = NULL ) ;
void BroadcastThunderbolt ( int a_BlockX , int a_BlockY , int a_BlockZ , const cClientHandle * a_Exclude = NULL ) ;
void BroadcastTimeUpdate ( const cClientHandle * a_Exclude = NULL ) ;
void BroadcastChunkData ( int a_ChunkX , int a_ChunkZ , cChunkDataSerializer & a_Serializer , const cClientHandle * a_Exclude = NULL ) ;
void BroadcastPlayerListItem ( const cPlayer & a_Player , bool a_IsOnline , const cClientHandle * a_Exclude = NULL ) ;
void BroadcastSoundEffect ( const AString & a_SoundName , int a_SrcX , int a_SrcY , int a_SrcZ , float a_Volume , float a_Pitch , const cClientHandle * a_Exclude = NULL ) ; // a_Src coords are Block * 8
void BroadcastSoundParticleEffect ( int a_EffectID , int a_SrcX , int a_SrcY , int a_SrcZ , int a_Data , const cClientHandle * a_Exclude = NULL ) ;
void BroadcastBlockBreakAnimation ( int a_EntityID , int a_BlockX , int a_BlockY , int a_BlockZ , char a_Stage , const cClientHandle * a_Exclude = NULL ) ;
void BroadcastUseBed ( const cEntity & a_Entity , int a_BlockX , int a_BlockY , int a_BlockZ ) ;
2012-09-25 05:54:36 -04:00
2012-08-24 03:58:26 -04:00
/// If there is a block entity at the specified coods, sends it to all clients except a_Exclude
void BroadcastBlockEntity ( int a_BlockX , int a_BlockY , int a_BlockZ , const cClientHandle * a_Exclude = NULL ) ;
/// If there is a block entity at the specified coords, sends it to the client specified
void SendBlockEntity ( int a_BlockX , int a_BlockY , int a_BlockZ , cClientHandle & a_Client ) ;
2012-08-18 06:38:15 -04:00
2013-04-13 17:02:10 -04:00
void MarkChunkDirty ( int a_ChunkX , int a_ChunkZ ) ;
void MarkChunkSaving ( int a_ChunkX , int a_ChunkZ ) ;
void MarkChunkSaved ( int a_ChunkX , int a_ChunkZ ) ;
2012-06-14 09:06:06 -04:00
/** Sets the chunk data as either loaded from the storage or generated.
a_BlockLight and a_BlockSkyLight are optional , if not present , chunk will be marked as unlighted .
a_BiomeMap is optional , if not present , biomes will be calculated by the generator
a_HeightMap is optional , if not present , will be calculated .
If a_MarkDirty is set , the chunk is set as dirty ( used after generating )
*/
void SetChunkData (
2013-04-13 17:02:10 -04:00
int a_ChunkX , int a_ChunkZ ,
2012-06-14 09:06:06 -04:00
const BLOCKTYPE * a_BlockTypes ,
const NIBBLETYPE * a_BlockMeta ,
const NIBBLETYPE * a_BlockLight ,
const NIBBLETYPE * a_BlockSkyLight ,
const cChunkDef : : HeightMap * a_HeightMap ,
const cChunkDef : : BiomeMap * a_BiomeMap ,
cEntityList & a_Entities ,
cBlockEntityList & a_BlockEntities ,
bool a_MarkDirty
) ;
void ChunkLighted (
int a_ChunkX , int a_ChunkZ ,
const cChunkDef : : BlockNibbles & a_BlockLight ,
const cChunkDef : : BlockNibbles & a_SkyLight
) ;
2013-04-13 17:02:10 -04:00
bool GetChunkData ( int a_ChunkX , int a_ChunkZ , cChunkDataCallback & a_Callback ) ;
2012-06-14 09:06:06 -04:00
/// Gets the chunk's blocks, only the block types
2013-04-13 17:02:10 -04:00
bool GetChunkBlockTypes ( int a_ChunkX , int a_ChunkZ , BLOCKTYPE * a_BlockTypes ) ;
2012-06-14 09:06:06 -04:00
2013-04-13 17:02:10 -04:00
bool IsChunkValid ( int a_ChunkX , int a_ChunkZ ) const ;
bool HasChunkAnyClients ( int a_ChunkX , int a_ChunkZ ) const ;
2012-06-14 09:06:06 -04:00
void UnloadUnusedChunks ( void ) ; // tolua_export
void CollectPickupsByPlayer ( cPlayer * a_Player ) ;
// MOTD
const AString & GetDescription ( void ) const { return m_Description ; } // FIXME: This should not be in cWorld
// Max Players
2013-01-11 23:46:01 -05:00
unsigned int GetMaxPlayers ( void ) const { return m_MaxPlayers ; } // tolua_export
void SetMaxPlayers ( int iMax ) ; // tolua_export
2012-06-14 09:06:06 -04:00
2013-06-09 13:46:40 -04:00
void AddPlayer ( cPlayer * a_Player ) ;
void RemovePlayer ( cPlayer * a_Player ) ;
2012-06-14 09:06:06 -04:00
2012-06-16 04:35:07 -04:00
/// Calls the callback for each player in the list; returns true if all players processed, false if the callback aborted by returning true
2012-06-14 09:06:06 -04:00
bool ForEachPlayer ( cPlayerListCallback & a_Callback ) ; // >> EXPORTED IN MANUALBINDINGS <<
2012-07-02 07:21:21 -04:00
/// Calls the callback for the player of the given name; returns true if the player was found and the callback called, false if player not found. Callback return ignored
bool DoWithPlayer ( const AString & a_PlayerName , cPlayerListCallback & a_Callback ) ; // >> EXPORTED IN MANUALBINDINGS <<
2012-08-22 19:05:12 -04:00
/// Finds a player from a partial or complete player name and calls the callback - case-insensitive
2013-02-01 14:59:58 -05:00
bool FindAndDoWithPlayer ( const AString & a_PlayerNameHint , cPlayerListCallback & a_Callback ) ; // >> EXPORTED IN MANUALBINDINGS <<
2012-06-14 09:06:06 -04:00
2013-01-11 23:46:01 -05:00
unsigned int GetNumPlayers ( ) ; // tolua_export
2012-06-14 09:06:06 -04:00
// TODO: This interface is dangerous - rewrite to DoWithClosestPlayer(pos, sight, action)
cPlayer * FindClosestPlayer ( const Vector3f & a_Pos , float a_SightLimit ) ;
void SendPlayerList ( cPlayer * a_DestPlayer ) ; // Sends playerlist to the player
2013-04-13 17:28:55 -04:00
/// Adds the entity into its appropriate chunk; takes ownership of the entity ptr
2013-04-13 17:02:10 -04:00
void AddEntity ( cEntity * a_Entity ) ;
2012-06-14 09:06:06 -04:00
2013-04-13 17:02:10 -04:00
bool HasEntity ( int a_UniqueID ) ;
2012-06-14 09:06:06 -04:00
2013-04-13 17:28:55 -04:00
/// Removes the entity, the entity ptr ownership is assumed taken by the caller
void RemoveEntity ( cEntity * a_Entity ) ;
2012-06-16 04:35:07 -04:00
/// Calls the callback for each entity in the entire world; returns true if all entities processed, false if the callback aborted by returning true
bool ForEachEntity ( cEntityCallback & a_Callback ) ; // Exported in ManualBindings.cpp
/// Calls the callback for each entity in the specified chunk; returns true if all entities processed, false if the callback aborted by returning true
bool ForEachEntityInChunk ( int a_ChunkX , int a_ChunkZ , cEntityCallback & a_Callback ) ; // Exported in ManualBindings.cpp
2013-04-13 17:02:10 -04:00
/// Calls the callback if the entity with the specified ID is found, with the entity object as the callback param. Returns true if entity found and callback returned false.
2012-06-16 04:35:07 -04:00
bool DoWithEntityByID ( int a_UniqueID , cEntityCallback & a_Callback ) ; // TODO: Exported in ManualBindings.cpp
2012-06-14 09:06:06 -04:00
/// Compares clients of two chunks, calls the callback accordingly
2013-04-13 17:02:10 -04:00
void CompareChunkClients ( int a_ChunkX1 , int a_ChunkZ1 , int a_ChunkX2 , int a_ChunkZ2 , cClientDiffCallback & a_Callback ) ;
2012-06-14 09:06:06 -04:00
/// Adds client to a chunk, if not already present; returns true if added, false if present
2013-04-13 17:02:10 -04:00
bool AddChunkClient ( int a_ChunkX , int a_ChunkZ , cClientHandle * a_Client ) ;
2012-06-14 09:06:06 -04:00
/// Removes client from the chunk specified
2013-04-13 17:02:10 -04:00
void RemoveChunkClient ( int a_ChunkX , int a_ChunkZ , cClientHandle * a_Client ) ;
2012-06-14 09:06:06 -04:00
/// Removes the client from all chunks it is present in
void RemoveClientFromChunks ( cClientHandle * a_Client ) ;
/// Sends the chunk to the client specified, if the chunk is valid. If not valid, the request is postponed (ChunkSender will send that chunk when it becomes valid+lighted)
2013-04-13 17:02:10 -04:00
void SendChunkTo ( int a_ChunkX , int a_ChunkZ , cClientHandle * a_Client ) ;
2012-06-14 09:06:06 -04:00
/// Removes client from ChunkSender's queue of chunks to be sent
void RemoveClientFromChunkSender ( cClientHandle * a_Client ) ;
/// Touches the chunk, causing it to be loaded or generated
void TouchChunk ( int a_ChunkX , int a_ChunkY , int a_ChunkZ ) ;
/// Loads the chunk, if not already loaded. Doesn't generate. Returns true if chunk valid (even if already loaded before)
bool LoadChunk ( int a_ChunkX , int a_ChunkY , int a_ChunkZ ) ;
/// Loads the chunks specified. Doesn't report failure, other than chunks being !IsValid()
void LoadChunks ( const cChunkCoordsList & a_Chunks ) ;
/// Marks the chunk as failed-to-load:
void ChunkLoadFailed ( int a_ChunkX , int a_ChunkY , int a_ChunkZ ) ;
2013-06-12 03:14:06 -04:00
/// Sets the sign text, asking plugins for permission first. a_Player is the player who this change belongs to, may be NULL. Returns true if sign text changed. Same as UpdateSign()
bool SetSignLines ( int a_BlockX , int a_BlockY , int a_BlockZ , const AString & a_Line1 , const AString & a_Line2 , const AString & a_Line3 , const AString & a_Line4 , cPlayer * a_Player = NULL ) ; // Exported in ManualBindings.cpp
/// Sets the sign text, asking plugins for permission first. a_Player is the player who this change belongs to, may be NULL. Returns true if sign text changed. Same as SetSignLines()
bool UpdateSign ( int a_X , int a_Y , int a_Z , const AString & a_Line1 , const AString & a_Line2 , const AString & a_Line3 , const AString & a_Line4 , cPlayer * a_Player = NULL ) ; // Exported in ManualBindings.cpp
2012-06-14 09:06:06 -04:00
/// Marks (a_Stay == true) or unmarks (a_Stay == false) chunks as non-unloadable. To be used only by cChunkStay!
void ChunksStay ( const cChunkCoordsList & a_Chunks , bool a_Stay = true ) ;
/// Regenerate the given chunk:
2013-01-11 23:46:01 -05:00
void RegenerateChunk ( int a_ChunkX , int a_ChunkZ ) ; // tolua_export
2012-06-14 09:06:06 -04:00
/// Generates the given chunk, if not already generated
2013-01-11 23:46:01 -05:00
void GenerateChunk ( int a_ChunkX , int a_ChunkZ ) ; // tolua_export
2012-06-14 09:06:06 -04:00
/// Queues a chunk for lighting; a_Callback is called after the chunk is lighted
void QueueLightChunk ( int a_ChunkX , int a_ChunkZ , cChunkCoordCallback * a_Callback = NULL ) ;
bool IsChunkLighted ( int a_ChunkX , int a_ChunkZ ) ;
2012-07-02 12:30:17 -04:00
/// Calls the callback for each chunk in the coords specified (all cords are inclusive). Returns true if all chunks have been processed successfully
bool ForEachChunkInRect ( int a_MinChunkX , int a_MaxChunkX , int a_MinChunkZ , int a_MaxChunkZ , cChunkDataCallback & a_Callback ) ;
2012-06-14 09:06:06 -04:00
2012-10-20 07:40:34 -04:00
// tolua_begin
void SetBlock ( int a_BlockX , int a_BlockY , int a_BlockZ , BLOCKTYPE a_BlockType , NIBBLETYPE a_BlockMeta ) ;
void FastSetBlock ( int a_BlockX , int a_BlockY , int a_BlockZ , BLOCKTYPE a_BlockType , NIBBLETYPE a_BlockMeta ) ;
BLOCKTYPE GetBlock ( int a_BlockX , int a_BlockY , int a_BlockZ ) ;
NIBBLETYPE GetBlockMeta ( int a_BlockX , int a_BlockY , int a_BlockZ ) ;
void SetBlockMeta ( int a_BlockX , int a_BlockY , int a_BlockZ , NIBBLETYPE a_MetaData ) ;
NIBBLETYPE GetBlockSkyLight ( int a_BlockX , int a_BlockY , int a_BlockZ ) ;
NIBBLETYPE GetBlockBlockLight ( int a_BlockX , int a_BlockY , int a_BlockZ ) ;
2012-10-26 04:47:30 -04:00
bool GetBlockTypeMeta ( int a_BlockX , int a_BlockY , int a_BlockZ , BLOCKTYPE & a_BlockType , NIBBLETYPE & a_BlockMeta ) ;
bool GetBlockInfo ( int a_BlockX , int a_BlockY , int a_BlockZ , BLOCKTYPE & a_BlockType , NIBBLETYPE & a_Meta , NIBBLETYPE & a_SkyLight , NIBBLETYPE & a_BlockLight ) ;
2012-10-20 07:40:34 -04:00
// TODO: NIBBLETYPE GetBlockActualLight(int a_BlockX, int a_BlockY, int a_BlockZ);
// Vector3i variants:
void FastSetBlock ( const Vector3i & a_Pos , BLOCKTYPE a_BlockType , NIBBLETYPE a_BlockMeta ) { FastSetBlock ( a_Pos . x , a_Pos . y , a_Pos . z , a_BlockType , a_BlockMeta ) ; }
BLOCKTYPE GetBlock ( const Vector3i & a_Pos ) { return GetBlock ( a_Pos . x , a_Pos . y , a_Pos . z ) ; }
NIBBLETYPE GetBlockMeta ( const Vector3i & a_Pos ) { return GetBlockMeta ( a_Pos . x , a_Pos . y , a_Pos . z ) ; }
void SetBlockMeta ( const Vector3i & a_Pos , NIBBLETYPE a_MetaData ) { SetBlockMeta ( a_Pos . x , a_Pos . y , a_Pos . z , a_MetaData ) ; }
// tolua_end
2012-06-14 09:06:06 -04:00
2012-10-06 12:58:31 -04:00
/** Writes the block area into the specified coords.
Returns true if all chunks have been processed .
Prefer cBlockArea : : Write ( ) instead , this is the internal implementation ; cBlockArea does error checking , too .
a_DataTypes is a bitmask of cBlockArea : : baXXX constants ORed together .
*/
bool WriteBlockArea ( cBlockArea & a_Area , int a_MinBlockX , int a_MinBlockY , int a_MinBlockZ , int a_DataTypes ) ;
2013-05-09 10:32:27 -04:00
// tolua_begin
2013-05-09 15:09:27 -04:00
/// Spawns item pickups for each item in the list. May compress pickups if too many entities:
2012-06-14 09:06:06 -04:00
void SpawnItemPickups ( const cItems & a_Pickups , double a_BlockX , double a_BlockY , double a_BlockZ , double a_FlyAwaySpeed = 1.0 ) ;
/// Spawns item pickups for each item in the list. May compress pickups if too many entities. All pickups get the speed specified:
2013-05-09 10:32:27 -04:00
void SpawnItemPickups ( const cItems & a_Pickups , double a_BlockX , double a_BlockY , double a_BlockZ , double a_SpeedX , double a_SpeedY , double a_SpeedZ ) ;
2013-06-18 15:09:51 -04:00
2013-06-21 17:19:47 -04:00
/// Spawns a new primed TNT entity at the specified block coords and specified fuse duration. Initial velocity is given based on the relative coefficient provided
void SpawnPrimedTNT ( double a_X , double a_Y , double a_Z , float a_FuseTimeInSec , double a_InitialVelocityCoeff = 1 ) ;
2013-05-09 15:09:27 -04:00
2013-05-09 10:32:27 -04:00
// tolua_end
2012-06-14 09:06:06 -04:00
/// Replaces world blocks with a_Blocks, if they are of type a_FilterBlockType
void ReplaceBlocks ( const sSetBlockVector & a_Blocks , BLOCKTYPE a_FilterBlockType ) ;
/// Retrieves block types of the specified blocks. If a chunk is not loaded, doesn't modify the block. Returns true if all blocks were read.
bool GetBlocks ( sSetBlockVector & a_Blocks , bool a_ContinueOnFailure ) ;
2012-12-21 05:59:59 -05:00
// tolua_begin
bool DigBlock ( int a_X , int a_Y , int a_Z ) ;
void SendBlockTo ( int a_X , int a_Y , int a_Z , cPlayer * a_Player ) ;
2012-06-14 09:06:06 -04:00
2012-12-21 05:59:59 -05:00
double GetSpawnX ( void ) const { return m_SpawnX ; }
double GetSpawnY ( void ) const { return m_SpawnY ; }
double GetSpawnZ ( void ) const { return m_SpawnZ ; }
2013-06-21 16:47:58 -04:00
/// Wakes up the simulators for the specified block
void WakeUpSimulators ( int a_BlockX , int a_BlockY , int a_BlockZ ) ;
/// Wakes up the simulators for the specified area of blocks
void WakeUpSimulatorsInArea ( int a_MinBlockX , int a_MaxBlockX , int a_MinBlockY , int a_MaxBlockY , int a_MinBlockZ , int a_MaxBlockZ ) ;
2012-12-21 05:59:59 -05:00
// tolua_end
2012-06-14 09:06:06 -04:00
2012-10-13 05:53:28 -04:00
inline cSimulatorManager * GetSimulatorManager ( void ) { return m_SimulatorManager ; }
inline cFluidSimulator * GetWaterSimulator ( void ) { return m_WaterSimulator ; }
inline cFluidSimulator * GetLavaSimulator ( void ) { return m_LavaSimulator ; }
2013-02-28 02:42:45 -05:00
2012-06-17 15:58:39 -04:00
/// Calls the callback for each chest in the specified chunk; returns true if all chests processed, false if the callback aborted by returning true
bool ForEachChestInChunk ( int a_ChunkX , int a_ChunkZ , cChestCallback & a_Callback ) ; // Exported in ManualBindings.cpp
2012-12-26 12:16:33 -05:00
/// Calls the callback for each dispenser in the specified chunk; returns true if all dispensers processed, false if the callback aborted by returning true
2012-12-27 10:07:12 -05:00
bool ForEachDispenserInChunk ( int a_ChunkX , int a_ChunkZ , cDispenserCallback & a_Callback ) ;
2012-12-26 12:16:33 -05:00
2013-05-26 10:39:04 -04:00
/// Calls the callback for each dropper in the specified chunk; returns true if all droppers processed, false if the callback aborted by returning true
bool ForEachDropperInChunk ( int a_ChunkX , int a_ChunkZ , cDropperCallback & a_Callback ) ;
/// Calls the callback for each dropspenser in the specified chunk; returns true if all dropspensers processed, false if the callback aborted by returning true
bool ForEachDropSpenserInChunk ( int a_ChunkX , int a_ChunkZ , cDropSpenserCallback & a_Callback ) ;
2012-06-17 15:58:39 -04:00
/// Calls the callback for each furnace in the specified chunk; returns true if all furnaces processed, false if the callback aborted by returning true
bool ForEachFurnaceInChunk ( int a_ChunkX , int a_ChunkZ , cFurnaceCallback & a_Callback ) ; // Exported in ManualBindings.cpp
2013-04-17 22:42:45 -04:00
/// Does an explosion with the specified strength at the specified coordinate
void DoExplosiontAt ( float a_ExplosionSzie , int a_BlockX , int a_BlockY , int a_BlockZ ) ; //
2012-06-17 15:58:39 -04:00
/// Calls the callback for the chest at the specified coords; returns false if there's no chest at those coords, true if found
bool DoWithChestAt ( int a_BlockX , int a_BlockY , int a_BlockZ , cChestCallback & a_Callback ) ; // Exported in ManualBindings.cpp
2013-05-25 07:57:28 -04:00
/// Calls the callback for the dispenser at the specified coords; returns false if there's no dispenser at those coords or callback returns true, returns true if found
2013-05-26 10:39:04 -04:00
bool DoWithDispenserAt ( int a_BlockX , int a_BlockY , int a_BlockZ , cDispenserCallback & a_Callback ) ; // Exported in ManualBindings.cpp
/// Calls the callback for the dropper at the specified coords; returns false if there's no dropper at those coords or callback returns true, returns true if found
bool DoWithDropperAt ( int a_BlockX , int a_BlockY , int a_BlockZ , cDropperCallback & a_Callback ) ; // Exported in ManualBindings.cpp
/// Calls the callback for the dropspenser at the specified coords; returns false if there's no dropspenser at those coords or callback returns true, returns true if found
bool DoWithDropSpenserAt ( int a_BlockX , int a_BlockY , int a_BlockZ , cDropSpenserCallback & a_Callback ) ; // Exported in ManualBindings.cpp
2012-12-26 12:16:33 -05:00
2013-05-25 07:57:28 -04:00
/// Calls the callback for the furnace at the specified coords; returns false if there's no furnace at those coords or callback returns true, returns true if found
2012-06-17 15:58:39 -04:00
bool DoWithFurnaceAt ( int a_BlockX , int a_BlockY , int a_BlockZ , cFurnaceCallback & a_Callback ) ; // Exported in ManualBindings.cpp
/// Retrieves the test on the sign at the specified coords; returns false if there's no sign at those coords, true if found
bool GetSignLines ( int a_BlockX , int a_BlockY , int a_BlockZ , AString & a_Line1 , AString & a_Line2 , AString & a_Line3 , AString & a_Line4 ) ; // tolua_export
2012-06-14 09:06:06 -04:00
/// a_Player is using block entity at [x, y, z], handle that:
2012-08-06 16:10:16 -04:00
void UseBlockEntity ( cPlayer * a_Player , int a_BlockX , int a_BlockY , int a_BlockZ ) { m_ChunkMap - > UseBlockEntity ( a_Player , a_BlockX , a_BlockY , a_BlockZ ) ; }
2012-06-14 09:06:06 -04:00
void GrowTree ( int a_BlockX , int a_BlockY , int a_BlockZ ) ; // tolua_export
void GrowTreeFromSapling ( int a_BlockX , int a_BlockY , int a_BlockZ , char a_SaplingMeta ) ; // tolua_export
void GrowTreeByBiome ( int a_BlockX , int a_BlockY , int a_BlockZ ) ; // tolua_export
void GrowTreeImage ( const sSetBlockVector & a_Blocks ) ;
/// Grows the plant at the specified block to its ripe stage (bonemeal used); returns false if the block is not growable. If a_IsBonemeal is true, block is not grown if not allowed in world.ini
2012-10-03 04:52:11 -04:00
bool GrowRipePlant ( int a_BlockX , int a_BlockY , int a_BlockZ , bool a_IsByBonemeal = false ) ; // tolua_export
2012-06-14 09:06:06 -04:00
2012-10-03 04:52:11 -04:00
/// Grows a cactus present at the block specified by the amount of blocks specified, up to the max height specified in the config
void GrowCactus ( int a_BlockX , int a_BlockY , int a_BlockZ , int a_NumBlocksToGrow ) ; // tolua_export
2012-06-14 09:06:06 -04:00
/// Grows a melon or a pumpkin next to the block specified (assumed to be the stem)
void GrowMelonPumpkin ( int a_BlockX , int a_BlockY , int a_BlockZ , char a_BlockType ) ; // tolua_export
2012-10-03 04:52:11 -04:00
/// Grows a sugarcane present at the block specified by the amount of blocks specified, up to the max height specified in the config
void GrowSugarcane ( int a_BlockX , int a_BlockY , int a_BlockZ , int a_NumBlocksToGrow ) ; // tolua_export
2012-06-14 09:06:06 -04:00
int GetBiomeAt ( int a_BlockX , int a_BlockZ ) ; // tolua_export
2013-01-11 23:46:01 -05:00
const AString & GetName ( void ) const { return m_WorldName ; } // tolua_export
2012-06-14 09:06:06 -04:00
const AString & GetIniFileName ( void ) const { return m_IniFileName ; }
inline static void AbsoluteToRelative ( int & a_X , int & a_Y , int & a_Z , int & a_ChunkX , int & a_ChunkY , int & a_ChunkZ )
{
// TODO: Use floor() instead of weird if statements
// Also fix Y
a_ChunkX = a_X / cChunkDef : : Width ;
if ( a_X < 0 & & a_X % cChunkDef : : Width ! = 0 ) a_ChunkX - - ;
a_ChunkY = 0 ;
a_ChunkZ = a_Z / cChunkDef : : Width ;
if ( a_Z < 0 & & a_Z % cChunkDef : : Width ! = 0 ) a_ChunkZ - - ;
a_X = a_X - a_ChunkX * cChunkDef : : Width ;
a_Y = a_Y - a_ChunkY * cChunkDef : : Height ;
a_Z = a_Z - a_ChunkZ * cChunkDef : : Width ;
}
inline static void BlockToChunk ( int a_X , int a_Y , int a_Z , int & a_ChunkX , int & a_ChunkY , int & a_ChunkZ )
{
// TODO: Use floor() instead of weird if statements
// Also fix Y
( void ) a_Y ; // not unused anymore
a_ChunkX = a_X / cChunkDef : : Width ;
if ( a_X < 0 & & a_X % cChunkDef : : Width ! = 0 ) a_ChunkX - - ;
a_ChunkY = 0 ;
a_ChunkZ = a_Z / cChunkDef : : Width ;
if ( a_Z < 0 & & a_Z % cChunkDef : : Width ! = 0 ) a_ChunkZ - - ;
}
2013-01-11 23:46:01 -05:00
void SaveAllChunks ( void ) ; // tolua_export
2012-06-14 09:06:06 -04:00
/// Returns the number of chunks loaded
2013-01-11 23:46:01 -05:00
int GetNumChunks ( ) const ; // tolua_export
2012-06-14 09:06:06 -04:00
/// Returns the number of chunks loaded and dirty, and in the lighting queue
void GetChunkStats ( int & a_NumValid , int & a_NumDirty , int & a_NumInLightingQueue ) ;
// Various queues length queries (cannot be const, they lock their CS):
inline int GetGeneratorQueueLength ( void ) { return m_Generator . GetQueueLength ( ) ; } // tolua_export
inline int GetLightingQueueLength ( void ) { return m_Lighting . GetQueueLength ( ) ; } // tolua_export
inline int GetStorageLoadQueueLength ( void ) { return m_Storage . GetLoadQueueLength ( ) ; } // tolua_export
inline int GetStorageSaveQueueLength ( void ) { return m_Storage . GetSaveQueueLength ( ) ; } // tolua_export
void Tick ( float a_Dt ) ;
2012-07-15 16:07:38 -04:00
void InitializeSpawn ( void ) ;
/// Stops threads that belong to this world (part of deinit)
void StopThreads ( void ) ;
2012-07-15 16:36:34 -04:00
void TickQueuedBlocks ( float a_Dt ) ;
struct BlockTickQueueItem
{
int X ;
int Y ;
int Z ;
float ToWait ;
} ;
2012-10-13 04:56:12 -04:00
void QueueBlockForTick ( int a_BlockX , int a_BlockY , int a_BlockZ , float a_TimeToWait ) ; // tolua_export
2012-06-14 09:06:06 -04:00
2013-02-13 14:22:08 -05:00
// tolua_begin
/// Casts a thunderbolt at the specified coords
void CastThunderbolt ( int a_BlockX , int a_BlockY , int a_BlockZ ) ;
/// Sets the specified weather; resets weather interval; asks and notifies plugins of the change
void SetWeather ( eWeather a_NewWeather ) ;
/// Forces a weather change in the next game tick
void ChangeWeather ( void ) ;
/// Returns the current weather
eWeather GetWeather ( void ) const { return m_Weather ; } ;
// tolua_end
2012-06-14 09:06:06 -04:00
cChunkGenerator & GetGenerator ( void ) { return m_Generator ; }
cWorldStorage & GetStorage ( void ) { return m_Storage ; }
cChunkMap * GetChunkMap ( void ) { return m_ChunkMap ; }
2012-07-15 16:36:34 -04:00
2012-06-14 09:06:06 -04:00
/// Sets the blockticking to start at the specified block. Only one blocktick per chunk may be set, second call overwrites the first call
void SetNextBlockTick ( int a_BlockX , int a_BlockY , int a_BlockZ ) ; // tolua_export
int GetMaxSugarcaneHeight ( void ) const { return m_MaxSugarcaneHeight ; } // tolua_export
int GetMaxCactusHeight ( void ) const { return m_MaxCactusHeight ; } // tolua_export
2012-07-15 16:36:34 -04:00
2012-10-13 04:56:12 -04:00
bool IsBlockDirectlyWatered ( int a_BlockX , int a_BlockY , int a_BlockZ ) ; // tolua_export
2012-06-14 09:06:06 -04:00
2012-10-28 10:57:35 -04:00
/// Spawns a mob of the specified entity type. Returns the mob's EntityID if recognized and spawned, <0 otherwise
int SpawnMob ( double a_PosX , double a_PosY , double a_PosZ , int a_EntityType ) ; // tolua_export
2013-03-01 14:35:29 -05:00
/// Returns a random number from the m_TickRand in range [0 .. a_Range]. To be used only in the tick thread!
2013-04-13 17:02:10 -04:00
int GetTickRandomNumber ( unsigned a_Range ) { return ( int ) ( m_TickRand . randInt ( a_Range ) ) ; }
2013-03-01 14:35:29 -05:00
2012-06-14 09:06:06 -04:00
private :
friend class cRoot ;
2013-03-11 13:15:34 -04:00
/// The dimension of the world, used by the client to provide correct lighting scheme
eDimension m_Dimension ;
/// This random generator is to be used only in the Tick() method, and thus only in the World-Tick-thread (MTRand is not exactly thread-safe)
2012-06-14 09:06:06 -04:00
MTRand m_TickRand ;
double m_SpawnX ;
double m_SpawnY ;
double m_SpawnZ ;
2012-11-01 17:38:20 -04:00
double m_WorldAgeSecs ; // World age, in seconds. Is only incremented, cannot be set by plugins.
double m_TimeOfDaySecs ; // Time of day in seconds. Can be adjusted. Is wrapped to zero each day.
Int64 m_WorldAge ; // World age in ticks, calculated off of m_WorldAgeSecs
Int64 m_TimeOfDay ; // Time in ticks, calculated off of m_TimeOfDaySecs
Int64 m_LastTimeUpdate ; // The tick in which the last time update has been sent.
Int64 m_LastUnload ; // The last WorldAge (in ticks) in which unloading was triggerred
Int64 m_LastSave ; // The last WorldAge (in ticks) in which save-all was triggerred
Int64 m_LastSpawnMonster ; // The last WorldAge (in ticks) in which a monster was spawned
2012-06-14 09:06:06 -04:00
eGameMode m_GameMode ;
2012-10-10 15:46:12 -04:00
bool m_bEnabledPVP ;
2012-10-21 17:15:57 -04:00
bool m_IsDeepSnowEnabled ;
2012-06-14 09:06:06 -04:00
// The cRedstone class simulates redstone and needs access to m_RSList
2013-03-02 14:57:09 -05:00
// friend class cRedstone;
2012-06-14 09:06:06 -04:00
std : : vector < int > m_RSList ;
2012-07-15 16:36:34 -04:00
std : : vector < BlockTickQueueItem * > m_BlockTickQueue ;
std : : vector < BlockTickQueueItem * > m_BlockTickQueueCopy ; //Second is for safely removing the objects from the queue
2012-06-14 09:06:06 -04:00
cSimulatorManager * m_SimulatorManager ;
cSandSimulator * m_SandSimulator ;
2012-10-13 05:53:28 -04:00
cFluidSimulator * m_WaterSimulator ;
cFluidSimulator * m_LavaSimulator ;
2012-06-14 09:06:06 -04:00
cFireSimulator * m_FireSimulator ;
cRedstoneSimulator * m_RedstoneSimulator ;
cCriticalSection m_CSClients ;
cCriticalSection m_CSPlayers ;
cWorldStorage m_Storage ;
AString m_Description ;
unsigned int m_MaxPlayers ;
cChunkMap * m_ChunkMap ;
bool m_bAnimals ;
2012-11-01 17:38:20 -04:00
Int64 m_SpawnMonsterRate ;
2012-06-14 09:06:06 -04:00
eWeather m_Weather ;
int m_WeatherInterval ;
int m_MaxCactusHeight ;
int m_MaxSugarcaneHeight ;
2013-04-05 09:45:00 -04:00
bool m_IsCactusBonemealable ;
bool m_IsCarrotsBonemealable ;
2012-06-14 09:06:06 -04:00
bool m_IsCropsBonemealable ;
bool m_IsGrassBonemealable ;
bool m_IsMelonStemBonemealable ;
bool m_IsMelonBonemealable ;
2013-04-05 09:45:00 -04:00
bool m_IsPotatoesBonemealable ;
2012-06-14 09:06:06 -04:00
bool m_IsPumpkinStemBonemealable ;
bool m_IsPumpkinBonemealable ;
2013-04-05 09:45:00 -04:00
bool m_IsSaplingBonemealable ;
2012-06-14 09:06:06 -04:00
bool m_IsSugarcaneBonemealable ;
cClientHandleList m_Clients ;
cPlayerList m_Players ;
cCriticalSection m_CSFastSetBlock ;
sSetBlockList m_FastSetBlockQueue ;
cChunkGenerator m_Generator ;
cChunkSender m_ChunkSender ;
cLightingThread m_Lighting ;
AString m_WorldName ;
AString m_IniFileName ;
cWorld ( const AString & a_WorldName ) ;
~ cWorld ( ) ;
void TickWeather ( float a_Dt ) ; // Handles weather each tick
void TickSpawnMobs ( float a_Dt ) ; // Handles mob spawning each tick
2012-10-13 05:53:28 -04:00
/// Creates a new fluid simulator, loads its settings from the inifile (a_FluidName section)
cFluidSimulator * InitializeFluidSimulator ( cIniFile & a_IniFile , const char * a_FluidName , BLOCKTYPE a_SimulateBlock , BLOCKTYPE a_StationaryBlock ) ;
2013-01-11 23:46:01 -05:00
} ; // tolua_export
2012-06-14 09:06:06 -04:00