2014-08-19 16:14:37 -04:00
2012-06-14 09:06:06 -04:00
# include "Globals.h" // NOTE: MSVC stupidness requires this to be the same across all modules
# ifndef _WIN32
# include <cstdlib>
# endif
2012-09-23 18:09:57 -04:00
# include "Chunk.h"
# include "World.h"
# include "ClientHandle.h"
# include "Server.h"
2013-11-26 12:14:46 -05:00
# include "zlib/zlib.h"
2012-06-14 09:06:06 -04:00
# include "Defines.h"
2014-07-30 16:19:51 -04:00
# include "BlockEntities/BeaconEntity.h"
2015-09-24 04:48:33 -04:00
# include "BlockEntities/BrewingstandEntity.h"
2013-05-28 15:12:47 -04:00
# include "BlockEntities/ChestEntity.h"
2017-05-22 16:27:55 -04:00
# include "BlockEntities/CommandBlockEntity.h"
2013-05-28 15:12:47 -04:00
# include "BlockEntities/DispenserEntity.h"
# include "BlockEntities/DropperEntity.h"
2014-09-19 17:00:54 -04:00
# include "BlockEntities/FlowerPotEntity.h"
2013-05-28 15:12:47 -04:00
# include "BlockEntities/FurnaceEntity.h"
2013-06-13 03:36:43 -04:00
# include "BlockEntities/HopperEntity.h"
2013-05-28 15:12:47 -04:00
# include "BlockEntities/JukeboxEntity.h"
2014-09-19 17:00:54 -04:00
# include "BlockEntities/MobHeadEntity.h"
# include "BlockEntities/MobSpawnerEntity.h"
2013-05-28 15:12:47 -04:00
# include "BlockEntities/NoteEntity.h"
# include "BlockEntities/SignEntity.h"
2013-08-19 05:39:13 -04:00
# include "Entities/Pickup.h"
2012-09-23 18:09:57 -04:00
# include "Item.h"
2014-11-18 06:07:08 -05:00
# include "Noise/Noise.h"
2012-09-23 18:09:57 -04:00
# include "Root.h"
2013-08-19 05:39:13 -04:00
# include "Entities/Player.h"
2012-07-09 03:00:28 -04:00
# include "BlockArea.h"
2013-12-08 06:17:54 -05:00
# include "Bindings/PluginManager.h"
2012-09-29 09:59:32 -04:00
# include "Blocks/BlockHandler.h"
2013-03-02 10:44:31 -05:00
# include "Simulator/FluidSimulator.h"
2013-09-07 16:19:56 -04:00
# include "MobCensus.h"
2013-09-07 18:11:38 -04:00
# include "MobSpawner.h"
2014-02-02 09:49:37 -05:00
# include "BlockInServerPluginInterface.h"
2014-07-24 12:32:05 -04:00
# include "SetChunkData.h"
2014-09-03 11:00:26 -04:00
# include "BoundingBox.h"
2014-09-26 13:13:19 -04:00
# include "Blocks/ChunkInterface.h"
2012-06-14 09:06:06 -04:00
2013-11-27 03:17:25 -05:00
# include "json/json.h"
2012-06-14 09:06:06 -04:00
2012-07-02 15:22:05 -04:00
2014-07-17 16:15:34 -04:00
////////////////////////////////////////////////////////////////////////////////
2012-06-14 09:06:06 -04:00
// sSetBlock:
2014-12-24 01:20:17 -05:00
sSetBlock : : sSetBlock ( int a_BlockX , int a_BlockY , int a_BlockZ , BLOCKTYPE a_BlockType , NIBBLETYPE a_BlockMeta ) :
m_RelX ( a_BlockX ) ,
m_RelY ( a_BlockY ) ,
m_RelZ ( a_BlockZ ) ,
m_BlockType ( a_BlockType ) ,
m_BlockMeta ( a_BlockMeta )
2012-06-14 09:06:06 -04:00
{
2014-12-24 01:20:17 -05:00
cChunkDef : : AbsoluteToRelative ( m_RelX , m_RelY , m_RelZ , m_ChunkX , m_ChunkZ ) ;
2012-06-14 09:06:06 -04:00
}
2014-07-17 16:15:34 -04:00
////////////////////////////////////////////////////////////////////////////////
2012-06-14 09:06:06 -04:00
// cChunk:
2012-12-14 17:38:30 -05:00
cChunk : : cChunk (
2014-08-28 05:36:35 -04:00
int a_ChunkX , int a_ChunkZ ,
2012-12-14 17:38:30 -05:00
cChunkMap * a_ChunkMap , cWorld * a_World ,
2014-06-16 10:12:50 -04:00
cChunk * a_NeighborXM , cChunk * a_NeighborXP , cChunk * a_NeighborZM , cChunk * a_NeighborZP ,
cAllocationPool < cChunkData : : sChunkSection > & a_Pool
2013-12-20 13:10:07 -05:00
) :
2014-09-05 16:16:48 -04:00
m_Presence ( cpInvalid ) ,
2014-12-21 16:19:53 -05:00
m_ShouldGenerateIfLoadFailed ( false ) ,
2013-12-20 13:10:07 -05:00
m_IsLightValid ( false ) ,
m_IsDirty ( false ) ,
m_IsSaving ( false ) ,
m_HasLoadFailed ( false ) ,
m_StayCount ( 0 ) ,
m_PosX ( a_ChunkX ) ,
m_PosZ ( a_ChunkZ ) ,
m_World ( a_World ) ,
m_ChunkMap ( a_ChunkMap ) ,
2014-06-16 10:12:50 -04:00
m_ChunkData ( a_Pool ) ,
2013-12-20 13:10:07 -05:00
m_BlockTickX ( 0 ) ,
m_BlockTickY ( 0 ) ,
m_BlockTickZ ( 0 ) ,
m_NeighborXM ( a_NeighborXM ) ,
m_NeighborXP ( a_NeighborXP ) ,
m_NeighborZM ( a_NeighborZM ) ,
m_NeighborZP ( a_NeighborZP ) ,
m_WaterSimulatorData ( a_World - > GetWaterSimulator ( ) - > CreateChunkData ( ) ) ,
2014-07-10 12:18:32 -04:00
m_LavaSimulatorData ( a_World - > GetLavaSimulator ( ) - > CreateChunkData ( ) ) ,
2014-09-28 12:57:01 -04:00
m_RedstoneSimulatorData ( a_World - > GetRedstoneSimulator ( ) - > CreateChunkData ( ) ) ,
2014-07-10 12:18:32 -04:00
m_AlwaysTicked ( 0 )
2012-06-14 09:06:06 -04:00
{
2014-10-20 16:55:07 -04:00
if ( a_NeighborXM ! = nullptr )
2012-12-14 17:38:30 -05:00
{
a_NeighborXM - > m_NeighborXP = this ;
}
2014-10-20 16:55:07 -04:00
if ( a_NeighborXP ! = nullptr )
2012-12-14 17:38:30 -05:00
{
a_NeighborXP - > m_NeighborXM = this ;
}
2014-10-20 16:55:07 -04:00
if ( a_NeighborZM ! = nullptr )
2012-12-14 17:38:30 -05:00
{
a_NeighborZM - > m_NeighborZP = this ;
}
2014-10-20 16:55:07 -04:00
if ( a_NeighborZP ! = nullptr )
2012-12-14 17:38:30 -05:00
{
a_NeighborZP - > m_NeighborZM = this ;
}
2012-06-14 09:06:06 -04:00
}
cChunk : : ~ cChunk ( )
{
2014-10-15 13:01:55 -04:00
cPluginManager : : Get ( ) - > CallHookChunkUnloaded ( * m_World , m_PosX , m_PosZ ) ;
2016-01-17 09:09:25 -05:00
2014-07-21 09:19:48 -04:00
// LOGINFO("### delete cChunk() (%i, %i) from %p, thread 0x%x ###", m_PosX, m_PosZ, this, GetCurrentThreadId());
2016-01-17 09:09:25 -05:00
2017-05-22 16:27:55 -04:00
for ( auto & KeyPair : m_BlockEntities )
2012-06-14 09:06:06 -04:00
{
2017-05-22 16:27:55 -04:00
delete KeyPair . second ;
2012-06-14 09:06:06 -04:00
}
m_BlockEntities . clear ( ) ;
// Remove and destroy all entities that are not players:
cEntityList Entities ;
2013-04-13 17:02:10 -04:00
std : : swap ( Entities , m_Entities ) ; // Need another list because cEntity destructors check if they've been removed from chunk
2017-05-22 16:27:55 -04:00
for ( auto Entity : Entities )
2012-06-14 09:06:06 -04:00
{
2017-05-22 16:27:55 -04:00
if ( ! Entity - > IsPlayer ( ) )
2012-06-14 09:06:06 -04:00
{
2016-02-07 12:07:14 -05:00
// Scheduling a normal destruction is neither possible (Since this chunk will be gone till the schedule occurs) nor necessary.
2017-05-22 16:27:55 -04:00
Entity - > DestroyNoScheduling ( false ) ; // No point in broadcasting in an unloading chunk. Chunks unload when no one is nearby.
delete Entity ;
2012-06-14 09:06:06 -04:00
}
}
2016-01-17 09:09:25 -05:00
2014-10-20 16:55:07 -04:00
if ( m_NeighborXM ! = nullptr )
2012-12-14 17:38:30 -05:00
{
2014-10-20 16:55:07 -04:00
m_NeighborXM - > m_NeighborXP = nullptr ;
2012-12-14 17:38:30 -05:00
}
2014-10-20 16:55:07 -04:00
if ( m_NeighborXP ! = nullptr )
2012-12-14 17:38:30 -05:00
{
2014-10-20 16:55:07 -04:00
m_NeighborXP - > m_NeighborXM = nullptr ;
2012-12-14 17:38:30 -05:00
}
2014-10-20 16:55:07 -04:00
if ( m_NeighborZM ! = nullptr )
2012-12-14 17:38:30 -05:00
{
2014-10-20 16:55:07 -04:00
m_NeighborZM - > m_NeighborZP = nullptr ;
2012-12-14 17:38:30 -05:00
}
2014-10-20 16:55:07 -04:00
if ( m_NeighborZP ! = nullptr )
2012-12-14 17:38:30 -05:00
{
2014-10-20 16:55:07 -04:00
m_NeighborZP - > m_NeighborZM = nullptr ;
2012-12-14 17:38:30 -05:00
}
2013-03-02 10:44:31 -05:00
delete m_WaterSimulatorData ;
2014-10-20 16:55:07 -04:00
m_WaterSimulatorData = nullptr ;
2013-03-02 10:44:31 -05:00
delete m_LavaSimulatorData ;
2014-10-20 16:55:07 -04:00
m_LavaSimulatorData = nullptr ;
2014-09-26 12:00:34 -04:00
delete m_RedstoneSimulatorData ;
2014-10-20 16:55:07 -04:00
m_RedstoneSimulatorData = nullptr ;
2012-06-14 09:06:06 -04:00
}
2014-09-05 16:16:48 -04:00
void cChunk : : SetPresence ( cChunk : : ePresence a_Presence )
2012-06-14 09:06:06 -04:00
{
2014-09-05 16:16:48 -04:00
m_Presence = a_Presence ;
if ( a_Presence = = cpPresent )
{
m_World - > GetChunkMap ( ) - > ChunkValidated ( ) ;
}
}
void cChunk : : SetShouldGenerateIfLoadFailed ( bool a_ShouldGenerateIfLoadFailed )
{
m_ShouldGenerateIfLoadFailed = a_ShouldGenerateIfLoadFailed ;
2012-06-14 09:06:06 -04:00
}
void cChunk : : MarkRegenerating ( void )
{
2014-09-05 16:16:48 -04:00
// Set as queued again:
SetPresence ( cpQueued ) ;
2012-06-14 09:06:06 -04:00
// Tell all clients attached to this chunk that they want this chunk:
2015-09-16 12:04:05 -04:00
for ( auto ClientHandle : m_LoadedByClient )
2012-06-14 09:06:06 -04:00
{
2015-09-16 12:04:05 -04:00
ClientHandle - > AddWantedChunk ( m_PosX , m_PosZ ) ;
2012-06-14 09:06:06 -04:00
} // for itr - m_LoadedByClient[]
}
bool cChunk : : CanUnload ( void )
{
2014-09-05 16:16:48 -04:00
return
m_LoadedByClient . empty ( ) & & // The chunk is not used by any client
! m_IsDirty & & // The chunk has been saved properly or hasn't been touched since the load / gen
( m_StayCount = = 0 ) & & // The chunk is not in a ChunkStay
( m_Presence ! = cpQueued ) ; // The chunk is not queued for loading / generating (otherwise multi-load / multi-gen could occur)
2012-06-14 09:06:06 -04:00
}
2016-09-03 11:38:29 -04:00
bool cChunk : : CanUnloadAfterSaving ( void )
{
return
m_LoadedByClient . empty ( ) & & // The chunk is not used by any client
m_IsDirty & & // The chunk is dirty
( m_StayCount = = 0 ) & & // The chunk is not in a ChunkStay
( m_Presence ! = cpQueued ) ; // The chunk is not queued for loading / generating (otherwise multi-load / multi-gen could occur)
}
2012-06-14 09:06:06 -04:00
void cChunk : : MarkSaving ( void )
{
m_IsSaving = true ;
}
void cChunk : : MarkSaved ( void )
{
if ( ! m_IsSaving )
{
return ;
}
m_IsDirty = false ;
}
void cChunk : : MarkLoaded ( void )
{
m_IsDirty = false ;
2014-09-05 16:16:48 -04:00
SetPresence ( cpPresent ) ;
2012-06-14 09:06:06 -04:00
}
void cChunk : : MarkLoadFailed ( void )
{
2014-09-05 16:16:48 -04:00
ASSERT ( m_Presence = = cpQueued ) ;
// If the chunk is marked as needed, generate it:
if ( m_ShouldGenerateIfLoadFailed )
2012-06-14 09:06:06 -04:00
{
2014-09-05 16:16:48 -04:00
m_World - > GetGenerator ( ) . QueueGenerateChunk ( m_PosX , m_PosZ , false ) ;
}
else
{
m_Presence = cpInvalid ;
2012-06-14 09:06:06 -04:00
}
}
void cChunk : : GetAllData ( cChunkDataCallback & a_Callback )
{
2014-09-05 16:16:48 -04:00
ASSERT ( m_Presence = = cpPresent ) ;
2014-05-21 14:58:48 -04:00
a_Callback . HeightMap ( & m_HeightMap ) ;
a_Callback . BiomeData ( & m_BiomeMap ) ;
2014-03-23 10:34:19 -04:00
2014-05-21 14:58:48 -04:00
a_Callback . LightIsValid ( m_IsLightValid ) ;
2014-04-04 18:16:52 -04:00
2014-05-21 14:58:48 -04:00
a_Callback . ChunkData ( m_ChunkData ) ;
2016-01-17 09:09:25 -05:00
2017-05-22 16:27:55 -04:00
for ( auto Entity : m_Entities )
2012-06-14 09:06:06 -04:00
{
2017-05-22 16:27:55 -04:00
a_Callback . Entity ( Entity ) ;
2012-06-14 09:06:06 -04:00
}
2016-01-17 09:09:25 -05:00
2017-05-22 16:27:55 -04:00
for ( auto & KeyPair : m_BlockEntities )
2012-06-14 09:06:06 -04:00
{
2017-05-22 16:27:55 -04:00
a_Callback . BlockEntity ( KeyPair . second ) ;
2012-06-14 09:06:06 -04:00
}
}
2014-07-24 12:32:05 -04:00
void cChunk : : SetAllData ( cSetChunkData & a_SetChunkData )
2012-06-14 09:06:06 -04:00
{
2014-07-24 12:32:05 -04:00
ASSERT ( a_SetChunkData . IsHeightMapValid ( ) ) ;
ASSERT ( a_SetChunkData . AreBiomesValid ( ) ) ;
2016-01-17 09:09:25 -05:00
2014-07-24 12:32:05 -04:00
memcpy ( m_BiomeMap , a_SetChunkData . GetBiomes ( ) , sizeof ( m_BiomeMap ) ) ;
memcpy ( m_HeightMap , a_SetChunkData . GetHeightMap ( ) , sizeof ( m_HeightMap ) ) ;
2014-04-02 17:53:03 -04:00
2014-07-24 12:32:05 -04:00
m_ChunkData . SetBlockTypes ( a_SetChunkData . GetBlockTypes ( ) ) ;
m_ChunkData . SetMetas ( a_SetChunkData . GetBlockMetas ( ) ) ;
if ( a_SetChunkData . IsLightValid ( ) )
2012-06-14 09:06:06 -04:00
{
2014-07-24 12:32:05 -04:00
m_ChunkData . SetBlockLight ( a_SetChunkData . GetBlockLight ( ) ) ;
m_ChunkData . SetSkyLight ( a_SetChunkData . GetSkyLight ( ) ) ;
m_IsLightValid = true ;
2012-06-14 09:06:06 -04:00
}
2014-07-24 12:32:05 -04:00
else
2014-04-06 18:30:21 -04:00
{
2014-07-24 12:32:05 -04:00
m_IsLightValid = false ;
2014-04-07 15:57:14 -04:00
}
2014-04-06 18:30:21 -04:00
2012-06-14 09:06:06 -04:00
// Clear the block entities present - either the loader / saver has better, or we'll create empty ones:
2017-05-22 16:27:55 -04:00
for ( auto & KeyPair : m_BlockEntities )
2012-06-14 09:06:06 -04:00
{
2017-05-22 16:27:55 -04:00
delete KeyPair . second ;
2012-06-14 09:06:06 -04:00
}
2014-07-24 12:32:05 -04:00
m_BlockEntities . clear ( ) ;
std : : swap ( a_SetChunkData . GetBlockEntities ( ) , m_BlockEntities ) ;
2014-08-29 12:19:27 -04:00
// Check that all block entities have a valid blocktype at their respective coords (DEBUG-mode only):
# ifdef _DEBUG
2017-05-22 16:27:55 -04:00
for ( auto & KeyPair : m_BlockEntities )
2014-08-29 12:19:27 -04:00
{
2017-05-22 16:27:55 -04:00
cBlockEntity * Block = KeyPair . second ;
BLOCKTYPE EntityBlockType = Block - > GetBlockType ( ) ;
BLOCKTYPE WorldBlockType = GetBlock ( Block - > GetRelX ( ) , Block - > GetPosY ( ) , Block - > GetRelZ ( ) ) ;
2014-10-18 14:54:34 -04:00
ASSERT ( WorldBlockType = = EntityBlockType ) ;
2017-05-22 16:27:55 -04:00
} // for KeyPair - m_BlockEntities
2014-08-29 12:19:27 -04:00
# endif // _DEBUG
2016-01-17 09:09:25 -05:00
2013-04-06 17:21:57 -04:00
// Set all block entities' World variable:
2017-05-22 16:27:55 -04:00
for ( auto & KeyPair : m_BlockEntities )
2013-04-06 17:21:57 -04:00
{
2017-05-22 16:27:55 -04:00
KeyPair . second - > SetWorld ( m_World ) ;
2013-04-06 17:21:57 -04:00
}
2016-01-17 09:09:25 -05:00
2012-06-14 09:06:06 -04:00
// Create block entities that the loader didn't load; fill them with defaults
CreateBlockEntities ( ) ;
2016-01-17 09:09:25 -05:00
2013-03-14 16:02:52 -04:00
// Set the chunk data as valid. This may be needed for some simulators that perform actions upon block adding (Vaporize)
2014-09-05 16:16:48 -04:00
SetPresence ( cpPresent ) ;
2016-01-17 09:09:25 -05:00
2013-03-03 10:33:55 -05:00
// Wake up all simulators for their respective blocks:
WakeUpSimulators ( ) ;
2012-06-14 09:06:06 -04:00
m_HasLoadFailed = false ;
}
void cChunk : : SetLight (
const cChunkDef : : BlockNibbles & a_BlockLight ,
const cChunkDef : : BlockNibbles & a_SkyLight
)
{
// TODO: We might get cases of wrong lighting when a chunk changes in the middle of a lighting calculation.
// Postponing until we see how bad it is :)
2014-04-06 18:30:21 -04:00
2014-05-24 08:33:40 -04:00
m_ChunkData . SetBlockLight ( a_BlockLight ) ;
2014-04-06 18:30:21 -04:00
2014-05-21 15:08:34 -04:00
m_ChunkData . SetSkyLight ( a_SkyLight ) ;
2014-04-06 18:30:21 -04:00
2012-06-14 09:06:06 -04:00
m_IsLightValid = true ;
}
void cChunk : : GetBlockTypes ( BLOCKTYPE * a_BlockTypes )
{
2014-05-29 12:25:08 -04:00
m_ChunkData . CopyBlockTypes ( a_BlockTypes ) ;
2012-06-14 09:06:06 -04:00
}
2012-10-06 12:58:31 -04:00
void cChunk : : WriteBlockArea ( cBlockArea & a_Area , int a_MinBlockX , int a_MinBlockY , int a_MinBlockZ , int a_DataTypes )
{
if ( ( a_DataTypes & ( cBlockArea : : baTypes | cBlockArea : : baMetas ) ) ! = ( cBlockArea : : baTypes | cBlockArea : : baMetas ) )
{
LOGWARNING ( " cChunk::WriteBlockArea(): unsupported datatype request, can write only types + metas (0x%x), requested 0x%x. Ignoring. " ,
( cBlockArea : : baTypes | cBlockArea : : baMetas ) , a_DataTypes & ( cBlockArea : : baTypes | cBlockArea : : baMetas )
) ;
return ;
}
2016-01-17 09:09:25 -05:00
2012-10-06 12:58:31 -04:00
// SizeX, SizeZ are the dimensions of the block data to copy to the chunk (size of the geometric union)
int BlockStartX = std : : max ( a_MinBlockX , m_PosX * cChunkDef : : Width ) ;
int BlockEndX = std : : min ( a_MinBlockX + a_Area . GetSizeX ( ) , ( m_PosX + 1 ) * cChunkDef : : Width ) ;
int BlockStartZ = std : : max ( a_MinBlockZ , m_PosZ * cChunkDef : : Width ) ;
int BlockEndZ = std : : min ( a_MinBlockZ + a_Area . GetSizeZ ( ) , ( m_PosZ + 1 ) * cChunkDef : : Width ) ;
int SizeX = BlockEndX - BlockStartX ;
int SizeZ = BlockEndZ - BlockStartZ ;
int OffX = BlockStartX - m_PosX * cChunkDef : : Width ;
int OffZ = BlockStartZ - m_PosZ * cChunkDef : : Width ;
int BaseX = BlockStartX - a_MinBlockX ;
int BaseZ = BlockStartZ - a_MinBlockZ ;
2015-06-30 08:08:21 -04:00
int SizeY = std : : min ( a_Area . GetSizeY ( ) , cChunkDef : : Height - a_MinBlockY ) ;
2012-10-06 12:58:31 -04:00
// TODO: Improve this by not calling FastSetBlock() and doing the processing here
// so that the heightmap is touched only once for each column.
BLOCKTYPE * AreaBlockTypes = a_Area . GetBlockTypes ( ) ;
NIBBLETYPE * AreaBlockMetas = a_Area . GetBlockMetas ( ) ;
for ( int y = 0 ; y < SizeY ; y + + )
{
int ChunkY = a_MinBlockY + y ;
int AreaY = y ;
for ( int z = 0 ; z < SizeZ ; z + + )
{
int ChunkZ = OffZ + z ;
int AreaZ = BaseZ + z ;
for ( int x = 0 ; x < SizeX ; x + + )
{
int ChunkX = OffX + x ;
int AreaX = BaseX + x ;
int idx = a_Area . MakeIndex ( AreaX , AreaY , AreaZ ) ;
BLOCKTYPE BlockType = AreaBlockTypes [ idx ] ;
NIBBLETYPE BlockMeta = AreaBlockMetas [ idx ] ;
FastSetBlock ( ChunkX , ChunkY , ChunkZ , BlockType , BlockMeta ) ;
} // for x
} // for z
} // for y
}
2012-06-14 09:06:06 -04:00
bool cChunk : : HasBlockEntityAt ( int a_BlockX , int a_BlockY , int a_BlockZ )
{
2017-05-22 16:27:55 -04:00
return ( GetBlockEntity ( a_BlockX , a_BlockY , a_BlockZ ) ! = nullptr ) ;
2012-06-14 09:06:06 -04:00
}
void cChunk : : Stay ( bool a_Stay )
{
m_StayCount + = ( a_Stay ? 1 : - 1 ) ;
ASSERT ( m_StayCount > = 0 ) ;
}
2015-05-09 05:16:56 -04:00
void cChunk : : CollectMobCensus ( cMobCensus & toFill )
2013-09-07 16:19:56 -04:00
{
2013-09-08 06:25:07 -04:00
toFill . CollectSpawnableChunk ( * this ) ;
2015-09-16 12:04:05 -04:00
std : : vector < Vector3d > PlayerPositions ;
PlayerPositions . reserve ( m_LoadedByClient . size ( ) ) ;
for ( auto ClientHandle : m_LoadedByClient )
2013-09-07 16:19:56 -04:00
{
2015-09-16 12:04:05 -04:00
const cPlayer * currentPlayer = ClientHandle - > GetPlayer ( ) ;
PlayerPositions . push_back ( currentPlayer - > GetPosition ( ) ) ;
2014-07-17 10:33:09 -04:00
}
2013-09-07 16:19:56 -04:00
Vector3d currentPosition ;
2015-09-16 12:04:05 -04:00
for ( auto entity : m_Entities )
2013-09-07 16:19:56 -04:00
{
2014-07-17 13:13:23 -04:00
// LOGD("Counting entity #%i (%s)", (*itr)->GetUniqueID(), (*itr)->GetClass());
2015-09-16 12:04:05 -04:00
if ( entity - > IsMob ( ) )
2013-09-07 16:19:56 -04:00
{
2015-09-16 12:04:05 -04:00
auto & Monster = reinterpret_cast < cMonster & > ( * entity ) ;
2013-09-09 12:45:39 -04:00
currentPosition = Monster . GetPosition ( ) ;
2015-09-24 10:04:44 -04:00
for ( const auto & PlayerPos : PlayerPositions )
2013-09-07 16:19:56 -04:00
{
2015-09-16 12:04:05 -04:00
toFill . CollectMob ( Monster , * this , ( currentPosition - PlayerPos ) . SqrLength ( ) ) ;
2013-09-07 16:19:56 -04:00
}
}
} // for itr - m_Entitites[]
}
2014-07-19 08:53:41 -04:00
void cChunk : : GetThreeRandomNumbers ( int & a_X , int & a_Y , int & a_Z , int a_MaxX , int a_MaxY , int a_MaxZ )
2013-09-07 18:11:38 -04:00
{
2017-06-13 15:35:30 -04:00
ASSERT (
( a_MaxX > 0 ) & & ( a_MaxY > 0 ) & & ( a_MaxZ > 0 ) & &
( a_MaxX < = std : : numeric_limits < int > : : max ( ) / a_MaxY ) & & // a_MaxX * a_MaxY doesn't overflow
( a_MaxX * a_MaxY < = std : : numeric_limits < int > : : max ( ) / a_MaxZ ) // a_MaxX * a_MaxY * a_MaxZ doesn't overflow
) ;
// MTRand gives an inclusive range [0, Max] but this gives the exclusive range [0, Max)
int OverallMax = ( a_MaxX - 1 ) * ( a_MaxY - 1 ) * ( a_MaxZ - 1 ) ;
int Random = m_World - > GetTickRandomNumber ( OverallMax ) ;
a_X = Random % a_MaxX ;
a_Y = ( Random / a_MaxX ) % a_MaxY ;
a_Z = ( ( Random / a_MaxX ) / a_MaxY ) % a_MaxZ ;
2013-09-07 18:11:38 -04:00
}
2014-07-10 17:04:33 -04:00
void cChunk : : GetRandomBlockCoords ( int & a_X , int & a_Y , int & a_Z )
2013-09-07 18:11:38 -04:00
{
// MG TODO : check if this kind of optimization (only one random call) is still needed
// MG TODO : if so propagate it
2014-07-10 17:04:33 -04:00
GetThreeRandomNumbers ( a_X , a_Y , a_Z , Width , Height - 2 , Width ) ;
2013-09-07 18:11:38 -04:00
a_Y + + ;
}
2015-05-09 05:16:56 -04:00
void cChunk : : SpawnMobs ( cMobSpawner & a_MobSpawner )
2013-09-07 18:11:38 -04:00
{
2014-07-10 17:04:33 -04:00
int CenterX , CenterY , CenterZ ;
GetRandomBlockCoords ( CenterX , CenterY , CenterZ ) ;
2013-10-21 11:41:48 -04:00
2014-07-10 17:04:33 -04:00
BLOCKTYPE PackCenterBlock = GetBlock ( CenterX , CenterY , CenterZ ) ;
if ( ! a_MobSpawner . CheckPackCenter ( PackCenterBlock ) )
2013-09-07 20:47:02 -04:00
{
2014-07-10 17:04:33 -04:00
return ;
2013-09-07 20:47:02 -04:00
}
2016-01-17 09:09:25 -05:00
2014-07-10 17:04:33 -04:00
a_MobSpawner . NewPack ( ) ;
int NumberOfTries = 0 ;
int NumberOfSuccess = 0 ;
int MaxNbOfSuccess = 4 ; // This can be changed during the process for Wolves and Ghasts
while ( ( NumberOfTries < 12 ) & & ( NumberOfSuccess < MaxNbOfSuccess ) )
{
const int HorizontalRange = 20 ; // MG TODO : relocate
const int VerticalRange = 0 ; // MG TODO : relocate
int TryX , TryY , TryZ ;
GetThreeRandomNumbers ( TryX , TryY , TryZ , 2 * HorizontalRange + 1 , 2 * VerticalRange + 1 , 2 * HorizontalRange + 1 ) ;
TryX - = HorizontalRange ;
TryY - = VerticalRange ;
TryZ - = HorizontalRange ;
TryX + = CenterX ;
TryY + = CenterY ;
TryZ + = CenterZ ;
ASSERT ( TryY > 0 ) ;
ASSERT ( TryY < cChunkDef : : Height - 1 ) ;
2016-01-17 09:09:25 -05:00
2014-07-10 17:04:33 -04:00
EMCSBiome Biome = m_ChunkMap - > GetBiomeAt ( TryX , TryZ ) ;
// MG TODO :
// Moon cycle (for slime)
// check player and playerspawn presence < 24 blocks
// check mobs presence on the block
// MG TODO : check that "Level" really means Y
2016-01-17 09:09:25 -05:00
2014-07-10 17:04:33 -04:00
/*
NIBBLETYPE SkyLight = 0 ;
NIBBLETYPE BlockLight = 0 ;
*/
2013-09-07 20:47:02 -04:00
2014-07-10 17:04:33 -04:00
NumberOfTries + + ;
if ( ! IsLightValid ( ) )
2013-09-07 20:47:02 -04:00
{
2014-07-10 17:04:33 -04:00
continue ;
2013-09-07 20:47:02 -04:00
}
2016-01-17 09:09:25 -05:00
2014-07-10 17:04:33 -04:00
cEntity * newMob = a_MobSpawner . TryToSpawnHere ( this , TryX , TryY , TryZ , Biome , MaxNbOfSuccess ) ;
2014-10-20 16:55:07 -04:00
if ( newMob = = nullptr )
2014-07-10 17:04:33 -04:00
{
continue ;
}
int WorldX , WorldY , WorldZ ;
PositionToWorldPosition ( TryX , TryY , TryZ , WorldX , WorldY , WorldZ ) ;
double ActualX = WorldX + 0.5 ;
double ActualZ = WorldZ + 0.5 ;
newMob - > SetPosition ( ActualX , WorldY , ActualZ ) ;
LOGD ( " Spawning %s #%i at {%d, %d, %d} " , newMob - > GetClass ( ) , newMob - > GetUniqueID ( ) , WorldX , WorldY , WorldZ ) ;
NumberOfSuccess + + ;
} // while (retry)
2013-09-07 18:11:38 -04:00
}
2012-06-14 09:06:06 -04:00
2015-01-11 16:12:26 -05:00
void cChunk : : Tick ( std : : chrono : : milliseconds a_Dt )
2012-06-14 09:06:06 -04:00
{
2016-04-05 04:45:09 -04:00
// If we are not valid, tick players and bailout
if ( ! IsValid ( ) )
{
for ( auto Entity : m_Entities )
{
if ( Entity - > IsPlayer ( ) )
{
Entity - > Tick ( a_Dt , * this ) ;
}
}
return ;
}
2012-08-25 13:52:08 -04:00
BroadcastPendingBlockChanges ( ) ;
2012-06-14 09:06:06 -04:00
CheckBlocks ( ) ;
2016-01-17 09:09:25 -05:00
2013-03-01 14:35:29 -05:00
// Tick simulators:
m_World - > GetSimulatorManager ( ) - > SimulateChunk ( a_Dt , m_PosX , m_PosZ , this ) ;
2016-01-17 09:09:25 -05:00
2013-04-13 17:02:10 -04:00
TickBlocks ( ) ;
2012-06-14 09:06:06 -04:00
2013-04-01 16:56:25 -04:00
// Tick all block entities in this chunk:
2017-05-22 16:27:55 -04:00
for ( auto & KeyPair : m_BlockEntities )
2012-06-14 09:06:06 -04:00
{
2017-05-22 16:27:55 -04:00
m_IsDirty = KeyPair . second - > Tick ( a_Dt , * this ) | m_IsDirty ;
2012-06-14 09:06:06 -04:00
}
2016-01-17 09:09:25 -05:00
2014-07-22 12:26:48 -04:00
for ( cEntityList : : iterator itr = m_Entities . begin ( ) ; itr ! = m_Entities . end ( ) ; )
2014-08-13 19:03:30 -04:00
{
2016-02-07 12:07:14 -05:00
// Do not tick mobs that are detached from the world. They're either scheduled for teleportation or for removal.
if ( ! ( * itr ) - > IsTicking ( ) )
{
+ + itr ;
continue ;
}
2016-04-06 05:16:14 -04:00
if ( ! ( ( * itr ) - > IsMob ( ) ) ) // Mobs are ticked inside cWorld::TickMobs() (as we don't have to tick them if they are far away from players)
{
// Tick all entities in this chunk (except mobs):
ASSERT ( ( * itr ) - > GetParentChunk ( ) = = this ) ;
( * itr ) - > Tick ( a_Dt , * this ) ;
ASSERT ( ( * itr ) - > GetParentChunk ( ) = = this ) ;
}
2014-07-22 12:26:48 -04:00
2016-02-07 12:07:14 -05:00
// Do not move mobs that are detached from the world to neighbors. They're either scheduled for teleportation or for removal.
2016-03-27 13:43:30 -04:00
// Because the schedulded destruction is going to look for them in this chunk. See cEntity::destroy.
2016-02-07 12:07:14 -05:00
if ( ! ( * itr ) - > IsTicking ( ) )
2014-05-31 17:28:51 -04:00
{
2016-02-07 12:07:14 -05:00
+ + itr ;
continue ;
2014-06-21 17:07:38 -04:00
}
2016-02-07 12:07:14 -05:00
if ( ( ( ( * itr ) - > GetChunkX ( ) ! = m_PosX ) | |
( ( * itr ) - > GetChunkZ ( ) ! = m_PosZ ) )
2013-04-13 17:02:10 -04:00
)
{
2016-02-07 12:07:14 -05:00
// This block is very similar to RemoveEntity, except it uses an iterator to avoid scanning the whole m_Entities
2014-08-13 19:03:30 -04:00
// The entity moved out of the chunk, move it to the neighbor
2016-02-07 12:07:14 -05:00
( * itr ) - > SetParentChunk ( nullptr ) ;
2013-04-13 17:02:10 -04:00
MoveEntityToNewChunk ( * itr ) ;
2016-02-07 12:07:14 -05:00
// Mark as dirty if it was a server-generated entity:
if ( ! ( * itr ) - > IsPlayer ( ) )
{
MarkDirty ( ) ;
}
2016-02-20 13:39:01 -05:00
itr = m_Entities . erase ( itr ) ;
2013-04-13 17:02:10 -04:00
}
else
{
2014-06-12 10:21:07 -04:00
+ + itr ;
2013-04-13 17:02:10 -04:00
}
2014-05-31 17:28:51 -04:00
} // for itr - m_Entitites[]
2016-01-17 09:09:25 -05:00
2013-04-01 16:56:25 -04:00
ApplyWeatherToTop ( ) ;
2012-06-14 09:06:06 -04:00
}
2013-11-30 09:58:27 -05:00
void cChunk : : TickBlock ( int a_RelX , int a_RelY , int a_RelZ )
{
2014-04-26 13:50:23 -04:00
cBlockHandler * Handler = BlockHandler ( GetBlock ( a_RelX , a_RelY , a_RelZ ) ) ;
2014-10-20 16:55:07 -04:00
ASSERT ( Handler ! = nullptr ) ; // Happenned on server restart, FS #243
2014-02-02 09:49:37 -05:00
cChunkInterface ChunkInterface ( this - > GetWorld ( ) - > GetChunkMap ( ) ) ;
cBlockInServerPluginInterface PluginInterface ( * this - > GetWorld ( ) ) ;
2014-07-19 08:53:41 -04:00
Handler - > OnUpdate ( ChunkInterface , * this - > GetWorld ( ) , PluginInterface , * this , a_RelX , a_RelY , a_RelZ ) ;
2013-11-30 09:58:27 -05:00
}
2013-04-13 17:02:10 -04:00
void cChunk : : MoveEntityToNewChunk ( cEntity * a_Entity )
{
2013-05-19 07:49:01 -04:00
cChunk * Neighbor = GetNeighborChunk ( a_Entity - > GetChunkX ( ) * cChunkDef : : Width , a_Entity - > GetChunkZ ( ) * cChunkDef : : Width ) ;
2014-10-20 16:55:07 -04:00
if ( Neighbor = = nullptr )
2013-04-13 17:02:10 -04:00
{
2016-03-29 14:23:53 -04:00
Neighbor = m_ChunkMap - > GetChunk ( a_Entity - > GetChunkX ( ) , a_Entity - > GetChunkZ ( ) ) ;
if ( Neighbor = = nullptr ) // This will assert inside GetChunk in debug builds
2013-05-06 14:32:22 -04:00
{
LOGWARNING ( " %s: Failed to move entity, destination chunk unreachable. Entity lost " , __FUNCTION__ ) ;
return ;
}
2013-04-13 17:02:10 -04:00
}
2013-05-19 07:49:01 -04:00
ASSERT ( Neighbor ! = this ) ; // Moving into the same chunk? wtf?
2013-04-13 17:02:10 -04:00
Neighbor - > AddEntity ( a_Entity ) ;
class cMover :
public cClientDiffCallback
{
virtual void Removed ( cClientHandle * a_Client ) override
{
a_Client - > SendDestroyEntity ( * m_Entity ) ;
}
virtual void Added ( cClientHandle * a_Client ) override
{
m_Entity - > SpawnOn ( * a_Client ) ;
}
cEntity * m_Entity ;
public :
2015-05-19 14:32:10 -04:00
cMover ( cEntity * a_CallbackEntity ) :
m_Entity ( a_CallbackEntity )
2013-04-13 17:02:10 -04:00
{ }
} Mover ( a_Entity ) ;
2016-01-17 09:09:25 -05:00
2013-04-13 17:02:10 -04:00
m_ChunkMap - > CompareChunkClients ( this , Neighbor , Mover ) ;
}
2012-08-25 13:52:08 -04:00
void cChunk : : BroadcastPendingBlockChanges ( void )
{
2013-08-19 15:55:03 -04:00
if ( m_PendingSendBlocks . empty ( ) )
{
return ;
}
2015-03-09 17:32:12 -04:00
if ( m_PendingSendBlocks . size ( ) > = 10240 )
{
// Resend the full chunk
2015-09-16 12:04:05 -04:00
for ( auto ClientHandle : m_LoadedByClient )
2015-03-09 17:32:12 -04:00
{
2015-09-16 12:04:05 -04:00
m_World - > ForceSendChunkTo ( m_PosX , m_PosZ , cChunkSender : : E_CHUNK_PRIORITY_MEDIUM , ClientHandle ) ;
2015-03-09 17:32:12 -04:00
}
}
else
2012-08-25 13:52:08 -04:00
{
2015-03-09 17:32:12 -04:00
// Only send block changes
2015-09-16 12:04:05 -04:00
for ( auto ClientHandle : m_LoadedByClient )
2015-03-09 17:32:12 -04:00
{
2015-09-16 12:04:05 -04:00
ClientHandle - > SendBlockChanges ( m_PosX , m_PosZ , m_PendingSendBlocks ) ;
2015-03-09 17:32:12 -04:00
}
2012-08-25 13:52:08 -04:00
}
2013-08-19 03:13:19 -04:00
m_PendingSendBlocks . clear ( ) ;
2012-08-25 13:52:08 -04:00
}
2014-01-31 18:17:41 -05:00
void cChunk : : CheckBlocks ( )
2012-06-14 09:06:06 -04:00
{
2014-04-24 16:49:56 -04:00
if ( m_ToTickBlocks . empty ( ) )
2012-06-14 09:06:06 -04:00
{
return ;
}
2014-04-26 13:50:23 -04:00
std : : vector < Vector3i > ToTickBlocks ;
2013-03-15 16:18:11 -04:00
std : : swap ( m_ToTickBlocks , ToTickBlocks ) ;
2016-01-17 09:09:25 -05:00
2014-01-31 18:17:41 -05:00
cChunkInterface ChunkInterface ( m_World - > GetChunkMap ( ) ) ;
2014-02-02 09:49:37 -05:00
cBlockInServerPluginInterface PluginInterface ( * m_World ) ;
2016-01-17 09:09:25 -05:00
2014-04-26 13:50:23 -04:00
for ( std : : vector < Vector3i > : : const_iterator itr = ToTickBlocks . begin ( ) , end = ToTickBlocks . end ( ) ; itr ! = end ; + + itr )
2012-06-14 09:06:06 -04:00
{
2014-04-26 13:50:23 -04:00
Vector3i Pos = ( * itr ) ;
2012-06-14 09:06:06 -04:00
2014-04-26 13:50:23 -04:00
cBlockHandler * Handler = BlockHandler ( GetBlock ( Pos ) ) ;
Handler - > Check ( ChunkInterface , PluginInterface , Pos . x , Pos . y , Pos . z , * this ) ;
2012-06-14 09:06:06 -04:00
} // for itr - ToTickBlocks[]
}
2013-04-13 17:02:10 -04:00
void cChunk : : TickBlocks ( void )
2012-06-14 09:06:06 -04:00
{
// Tick dem blocks
2016-01-30 19:25:03 -05:00
// _X: We must limit the random number or else we get a nasty int overflow bug - https://forum.cuberite.org/thread-457.html
2013-04-13 17:02:10 -04:00
int RandomX = m_World - > GetTickRandomNumber ( 0x00ffffff ) ;
int RandomY = m_World - > GetTickRandomNumber ( 0x00ffffff ) ;
int RandomZ = m_World - > GetTickRandomNumber ( 0x00ffffff ) ;
2012-06-14 09:06:06 -04:00
int TickX = m_BlockTickX ;
int TickY = m_BlockTickY ;
int TickZ = m_BlockTickZ ;
2016-01-17 09:09:25 -05:00
2014-02-02 09:49:37 -05:00
cChunkInterface ChunkInterface ( this - > GetWorld ( ) - > GetChunkMap ( ) ) ;
cBlockInServerPluginInterface PluginInterface ( * this - > GetWorld ( ) ) ;
2012-06-14 09:06:06 -04:00
// This for loop looks disgusting, but it actually does a simple thing - first processes m_BlockTick, then adds random to it
// This is so that SetNextBlockTick() works
for ( int i = 0 ; i < 50 ; i + + ,
2016-01-17 09:09:25 -05:00
2012-06-14 09:06:06 -04:00
// This weird construct (*2, then /2) is needed,
// otherwise the blocktick distribution is too biased towards even coords!
2016-01-17 09:09:25 -05:00
2012-06-14 09:06:06 -04:00
TickX = ( TickX + RandomX ) % ( Width * 2 ) ,
TickY = ( TickY + RandomY ) % ( Height * 2 ) ,
TickZ = ( TickZ + RandomZ ) % ( Width * 2 ) ,
m_BlockTickX = TickX / 2 ,
m_BlockTickY = TickY / 2 ,
m_BlockTickZ = TickZ / 2
)
{
if ( m_BlockTickY > cChunkDef : : GetHeight ( m_HeightMap , m_BlockTickX , m_BlockTickZ ) )
{
2014-07-17 13:13:23 -04:00
continue ; // It's all air up here
2012-06-14 09:06:06 -04:00
}
2014-04-26 13:50:23 -04:00
cBlockHandler * Handler = BlockHandler ( GetBlock ( m_BlockTickX , m_BlockTickY , m_BlockTickZ ) ) ;
2014-10-20 16:55:07 -04:00
ASSERT ( Handler ! = nullptr ) ; // Happenned on server restart, FS #243
2014-02-02 09:49:37 -05:00
Handler - > OnUpdate ( ChunkInterface , * this - > GetWorld ( ) , PluginInterface , * this , m_BlockTickX , m_BlockTickY , m_BlockTickZ ) ;
2012-10-07 06:08:57 -04:00
} // for i - tickblocks
2012-06-14 09:06:06 -04:00
}
2012-09-04 11:07:08 -04:00
2013-04-01 16:56:25 -04:00
void cChunk : : ApplyWeatherToTop ( )
2012-10-21 17:15:57 -04:00
{
if (
2017-06-13 15:35:30 -04:00
( GetRandomProvider ( ) . RandBool ( 0.99 ) ) | |
2012-10-21 17:15:57 -04:00
(
( m_World - > GetWeather ( ) ! = eWeather_Rain ) & &
( m_World - > GetWeather ( ) ! = eWeather_ThunderStorm )
)
)
{
// Not the right weather, or not at this tick; bail out
return ;
}
2016-01-17 09:09:25 -05:00
2013-04-01 16:56:25 -04:00
int X = m_World - > GetTickRandomNumber ( 15 ) ;
int Z = m_World - > GetTickRandomNumber ( 15 ) ;
2015-02-25 13:02:08 -05:00
// TODO: Check light levels, don't snow over when the BlockLight is higher than (7?)
int Height = GetHeight ( X , Z ) ;
if ( GetSnowStartHeight ( GetBiomeAt ( X , Z ) ) > Height )
{
return ;
}
BLOCKTYPE TopBlock = GetBlock ( X , Height , Z ) ;
NIBBLETYPE TopMeta = GetMeta ( X , Height , Z ) ;
if ( m_World - > IsDeepSnowEnabled ( ) & & ( TopBlock = = E_BLOCK_SNOW ) )
{
int MaxSize = 7 ;
BLOCKTYPE BlockType [ 4 ] ;
NIBBLETYPE BlockMeta [ 4 ] ;
UnboundedRelGetBlock ( X - 1 , Height , Z , BlockType [ 0 ] , BlockMeta [ 0 ] ) ;
UnboundedRelGetBlock ( X + 1 , Height , Z , BlockType [ 1 ] , BlockMeta [ 1 ] ) ;
UnboundedRelGetBlock ( X , Height , Z - 1 , BlockType [ 2 ] , BlockMeta [ 2 ] ) ;
UnboundedRelGetBlock ( X , Height , Z + 1 , BlockType [ 3 ] , BlockMeta [ 3 ] ) ;
for ( int i = 0 ; i < 4 ; i + + )
2012-10-21 17:15:57 -04:00
{
2015-02-25 13:02:08 -05:00
switch ( BlockType [ i ] )
2012-10-21 17:15:57 -04:00
{
2015-02-25 13:02:08 -05:00
case E_BLOCK_AIR :
2012-10-21 17:15:57 -04:00
{
2015-02-25 13:02:08 -05:00
MaxSize = 0 ;
break ;
2012-10-21 17:15:57 -04:00
}
2015-02-25 13:02:08 -05:00
case E_BLOCK_SNOW :
2012-10-21 17:15:57 -04:00
{
2015-02-25 13:02:08 -05:00
MaxSize = std : : min ( BlockMeta [ i ] + 1 , MaxSize ) ;
break ;
2012-10-21 17:15:57 -04:00
}
}
2015-02-25 13:02:08 -05:00
}
if ( TopMeta < MaxSize )
{
FastSetBlock ( X , Height , Z , E_BLOCK_SNOW , TopMeta + 1 ) ;
}
else if ( TopMeta > MaxSize )
2013-12-09 09:05:32 -05:00
{
2015-02-25 13:02:08 -05:00
FastSetBlock ( X , Height , Z , E_BLOCK_SNOW , TopMeta - 1 ) ;
2013-12-09 09:05:32 -05:00
}
2015-02-25 13:02:08 -05:00
}
2015-02-14 17:11:38 -05:00
else if ( cBlockInfo : : IsSnowable ( TopBlock ) & & ( Height < cChunkDef : : Height - 1 ) )
2015-02-25 13:02:08 -05:00
{
SetBlock ( X , Height + 1 , Z , E_BLOCK_SNOW , 0 ) ;
}
else if ( IsBlockWater ( TopBlock ) & & ( TopMeta = = 0 ) )
{
SetBlock ( X , Height , Z , E_BLOCK_ICE , 0 ) ;
}
else if (
( m_World - > IsDeepSnowEnabled ( ) ) & &
(
( TopBlock = = E_BLOCK_RED_ROSE ) | |
( TopBlock = = E_BLOCK_YELLOW_FLOWER ) | |
( TopBlock = = E_BLOCK_RED_MUSHROOM ) | |
( TopBlock = = E_BLOCK_BROWN_MUSHROOM )
)
)
{
SetBlock ( X , Height , Z , E_BLOCK_SNOW , 0 ) ;
}
2012-10-21 17:15:57 -04:00
}
2017-06-13 15:35:30 -04:00
bool cChunk : : GrowMelonPumpkin ( int a_RelX , int a_RelY , int a_RelZ , BLOCKTYPE a_BlockType )
2012-06-14 09:06:06 -04:00
{
2017-06-13 15:35:30 -04:00
auto & Random = GetRandomProvider ( ) ;
2012-06-14 09:06:06 -04:00
// Convert the stem BlockType into produce BlockType
BLOCKTYPE ProduceType ;
switch ( a_BlockType )
{
case E_BLOCK_MELON_STEM : ProduceType = E_BLOCK_MELON ; break ;
case E_BLOCK_PUMPKIN_STEM : ProduceType = E_BLOCK_PUMPKIN ; break ;
default :
{
ASSERT ( ! " Unhandled blocktype in TickMelonPumpkin() " ) ;
2016-05-29 04:30:47 -04:00
return false ;
2012-06-14 09:06:06 -04:00
}
}
// Check if there's another melon / pumpkin around that stem, if so, abort:
bool IsValid ;
BLOCKTYPE BlockType [ 4 ] ;
NIBBLETYPE BlockMeta ; // unused
IsValid = UnboundedRelGetBlock ( a_RelX + 1 , a_RelY , a_RelZ , BlockType [ 0 ] , BlockMeta ) ;
IsValid = IsValid & & UnboundedRelGetBlock ( a_RelX - 1 , a_RelY , a_RelZ , BlockType [ 1 ] , BlockMeta ) ;
IsValid = IsValid & & UnboundedRelGetBlock ( a_RelX , a_RelY , a_RelZ + 1 , BlockType [ 2 ] , BlockMeta ) ;
IsValid = IsValid & & UnboundedRelGetBlock ( a_RelX , a_RelY , a_RelZ - 1 , BlockType [ 3 ] , BlockMeta ) ;
if (
2014-07-17 10:33:09 -04:00
! IsValid | |
( BlockType [ 0 ] = = ProduceType ) | |
( BlockType [ 1 ] = = ProduceType ) | |
( BlockType [ 2 ] = = ProduceType ) | |
2012-06-14 09:06:06 -04:00
( BlockType [ 3 ] = = ProduceType )
)
{
// Neighbors not valid or already taken by the same produce
2016-05-29 04:30:47 -04:00
return false ;
2012-06-14 09:06:06 -04:00
}
2016-01-17 09:09:25 -05:00
2012-06-14 09:06:06 -04:00
// Pick a direction in which to place the produce:
int x = 0 , z = 0 ;
2017-06-13 15:35:30 -04:00
int CheckType = Random . RandInt ( 3 ) ; // The index to the neighbors array which should be checked for emptiness
2012-06-14 09:06:06 -04:00
switch ( CheckType )
{
case 0 : x = 1 ; break ;
case 1 : x = - 1 ; break ;
case 2 : z = 1 ; break ;
case 3 : z = - 1 ; break ;
}
2016-01-17 09:09:25 -05:00
2012-06-14 09:06:06 -04:00
// Check that the block in that direction is empty:
switch ( BlockType [ CheckType ] )
{
case E_BLOCK_AIR :
case E_BLOCK_SNOW :
case E_BLOCK_TALL_GRASS :
case E_BLOCK_DEAD_BUSH :
{
break ;
}
2016-05-29 04:30:47 -04:00
default : return false ;
2012-06-14 09:06:06 -04:00
}
2016-01-17 09:09:25 -05:00
2012-06-14 09:06:06 -04:00
// Check if there's soil under the neighbor. We already know the neighbors are valid. Place produce if ok
BLOCKTYPE Soil ;
2015-06-23 11:23:35 -04:00
VERIFY ( UnboundedRelGetBlock ( a_RelX + x , a_RelY - 1 , a_RelZ + z , Soil , BlockMeta ) ) ;
2012-06-14 09:06:06 -04:00
switch ( Soil )
{
case E_BLOCK_DIRT :
case E_BLOCK_GRASS :
case E_BLOCK_FARMLAND :
{
2015-06-23 11:23:35 -04:00
// Place a randomly-facing produce:
2017-06-13 15:35:30 -04:00
NIBBLETYPE Meta = ( ProduceType = = E_BLOCK_MELON ) ? 0 : static_cast < NIBBLETYPE > ( Random . RandInt ( 4 ) % 4 ) ;
2015-06-23 11:23:35 -04:00
LOGD ( " Growing melon / pumpkin at {%d, %d, %d} (<%d, %d> from stem), overwriting %s, growing on top of %s, meta %d " ,
a_RelX + x + m_PosX * cChunkDef : : Width , a_RelY , a_RelZ + z + m_PosZ * cChunkDef : : Width ,
x , z ,
2013-04-03 12:43:37 -04:00
ItemTypeToString ( BlockType [ CheckType ] ) . c_str ( ) ,
2015-06-23 11:23:35 -04:00
ItemTypeToString ( Soil ) . c_str ( ) ,
Meta
2013-04-03 12:43:37 -04:00
) ;
2015-06-23 11:23:35 -04:00
VERIFY ( UnboundedRelFastSetBlock ( a_RelX + x , a_RelY , a_RelZ + z , ProduceType , Meta ) ) ;
2015-06-23 16:16:38 -04:00
auto Absolute = RelativeToAbsolute ( Vector3i { a_RelX + x , a_RelY , a_RelZ + z } , m_PosX , m_PosZ ) ;
cChunkInterface ChunkInterface ( this - > GetWorld ( ) - > GetChunkMap ( ) ) ;
2015-07-01 04:40:16 -04:00
cBlockHandler : : NeighborChanged ( ChunkInterface , Absolute . x , Absolute . y - 1 , Absolute . z , BLOCK_FACE_YP ) ;
2012-06-14 09:06:06 -04:00
break ;
}
}
2016-05-29 04:30:47 -04:00
return true ;
2012-06-14 09:06:06 -04:00
}
2016-05-29 04:30:47 -04:00
int cChunk : : GrowSugarcane ( int a_RelX , int a_RelY , int a_RelZ , int a_NumBlocks )
2012-06-14 09:06:06 -04:00
{
// Check the total height of the sugarcane blocks here:
int Top = a_RelY + 1 ;
while (
( Top < cChunkDef : : Height ) & &
( GetBlock ( a_RelX , Top , a_RelZ ) = = E_BLOCK_SUGARCANE )
)
{
+ + Top ;
}
int Bottom = a_RelY - 1 ;
while (
( Bottom > 0 ) & &
( GetBlock ( a_RelX , Bottom , a_RelZ ) = = E_BLOCK_SUGARCANE )
)
{
- - Bottom ;
}
2016-01-17 09:09:25 -05:00
2012-06-14 09:06:06 -04:00
// Grow by at most a_NumBlocks, but no more than max height:
int ToGrow = std : : min ( a_NumBlocks , m_World - > GetMaxSugarcaneHeight ( ) + 1 - ( Top - Bottom ) ) ;
for ( int i = 0 ; i < ToGrow ; i + + )
{
BLOCKTYPE BlockType ;
NIBBLETYPE BlockMeta ;
if ( UnboundedRelGetBlock ( a_RelX , Top + i , a_RelZ , BlockType , BlockMeta ) & & ( BlockType = = E_BLOCK_AIR ) )
{
UnboundedRelFastSetBlock ( a_RelX , Top + i , a_RelZ , E_BLOCK_SUGARCANE , 0 ) ;
}
else
{
2016-05-29 04:30:47 -04:00
return i ;
2012-06-14 09:06:06 -04:00
}
} // for i
2016-05-29 04:30:47 -04:00
return ToGrow ;
2012-06-14 09:06:06 -04:00
}
2016-05-29 04:30:47 -04:00
int cChunk : : GrowCactus ( int a_RelX , int a_RelY , int a_RelZ , int a_NumBlocks )
2012-06-14 09:06:06 -04:00
{
// Check the total height of the sugarcane blocks here:
int Top = a_RelY + 1 ;
while (
( Top < cChunkDef : : Height ) & &
( GetBlock ( a_RelX , Top , a_RelZ ) = = E_BLOCK_CACTUS )
)
{
+ + Top ;
}
int Bottom = a_RelY - 1 ;
while (
( Bottom > 0 ) & &
( GetBlock ( a_RelX , Bottom , a_RelZ ) = = E_BLOCK_CACTUS )
)
{
- - Bottom ;
}
2016-01-17 09:09:25 -05:00
2012-06-14 09:06:06 -04:00
// Grow by at most a_NumBlocks, but no more than max height:
int ToGrow = std : : min ( a_NumBlocks , m_World - > GetMaxCactusHeight ( ) + 1 - ( Top - Bottom ) ) ;
for ( int i = 0 ; i < ToGrow ; i + + )
{
BLOCKTYPE BlockType ;
NIBBLETYPE BlockMeta ;
if ( UnboundedRelGetBlock ( a_RelX , Top + i , a_RelZ , BlockType , BlockMeta ) & & ( BlockType = = E_BLOCK_AIR ) )
{
// TODO: Check the surrounding blocks, if they aren't air, break the cactus block into pickups (and continue breaking blocks above in the next loop iterations)
UnboundedRelFastSetBlock ( a_RelX , Top + i , a_RelZ , E_BLOCK_CACTUS , 0 ) ;
}
else
{
2016-05-29 04:30:47 -04:00
return i ;
2012-06-14 09:06:06 -04:00
}
} // for i
2016-05-29 04:30:47 -04:00
return ToGrow ;
2012-06-14 09:06:06 -04:00
}
2016-05-29 16:10:35 -04:00
bool cChunk : : GrowTallGrass ( int a_RelX , int a_RelY , int a_RelZ )
{
if ( a_RelY > ( cChunkDef : : Height - 2 ) )
{
return false ;
}
BLOCKTYPE BlockType ;
NIBBLETYPE BlockMeta ;
if ( ! UnboundedRelGetBlock ( a_RelX , a_RelY , a_RelZ , BlockType , BlockMeta ) )
{
return false ;
}
if ( BlockType ! = E_BLOCK_TALL_GRASS )
{
return false ;
}
NIBBLETYPE LargeFlowerMeta ;
switch ( BlockMeta )
{
case E_META_TALL_GRASS_GRASS : LargeFlowerMeta = E_META_BIG_FLOWER_DOUBLE_TALL_GRASS ; break ;
case E_META_TALL_GRASS_FERN : LargeFlowerMeta = E_META_BIG_FLOWER_LARGE_FERN ; break ;
default : return false ;
}
return UnboundedRelFastSetBlock ( a_RelX , a_RelY , a_RelZ , E_BLOCK_BIG_FLOWER , LargeFlowerMeta ) & &
UnboundedRelFastSetBlock ( a_RelX , a_RelY + 1 , a_RelZ , E_BLOCK_BIG_FLOWER , 8 ) ;
}
2013-03-15 16:18:11 -04:00
bool cChunk : : UnboundedRelGetBlock ( int a_RelX , int a_RelY , int a_RelZ , BLOCKTYPE & a_BlockType , NIBBLETYPE & a_BlockMeta ) const
2012-06-14 09:06:06 -04:00
{
2015-11-10 17:06:29 -05:00
if ( ! cChunkDef : : IsValidHeight ( a_RelY ) )
2012-06-14 09:06:06 -04:00
{
2013-05-28 08:05:23 -04:00
LOGWARNING ( " %s: requesting a block with a_RelY out of range: %d " , __FUNCTION__ , a_RelY ) ;
2012-06-14 09:06:06 -04:00
return false ;
}
2013-10-22 11:54:09 -04:00
cChunk * Chunk = GetRelNeighborChunkAdjustCoords ( a_RelX , a_RelZ ) ;
2014-10-20 16:55:07 -04:00
if ( ( Chunk = = nullptr ) | | ! Chunk - > IsValid ( ) )
2012-12-14 17:38:30 -05:00
{
2013-10-22 11:54:09 -04:00
// The chunk is not available, bail out
return false ;
2012-12-14 17:38:30 -05:00
}
2013-10-22 11:54:09 -04:00
Chunk - > GetBlockTypeMeta ( a_RelX , a_RelY , a_RelZ , a_BlockType , a_BlockMeta ) ;
return true ;
2012-06-14 09:06:06 -04:00
}
2013-05-28 08:05:23 -04:00
bool cChunk : : UnboundedRelGetBlockType ( int a_RelX , int a_RelY , int a_RelZ , BLOCKTYPE & a_BlockType ) const
{
2015-11-10 17:06:29 -05:00
if ( ! cChunkDef : : IsValidHeight ( a_RelY ) )
2013-05-28 08:05:23 -04:00
{
LOGWARNING ( " %s: requesting a block with a_RelY out of range: %d " , __FUNCTION__ , a_RelY ) ;
return false ;
}
2013-10-22 11:54:09 -04:00
cChunk * Chunk = GetRelNeighborChunkAdjustCoords ( a_RelX , a_RelZ ) ;
2014-10-20 16:55:07 -04:00
if ( ( Chunk = = nullptr ) | | ! Chunk - > IsValid ( ) )
2013-05-28 08:05:23 -04:00
{
2013-10-22 11:54:09 -04:00
// The chunk is not available, bail out
return false ;
2013-05-28 08:05:23 -04:00
}
2013-10-22 11:54:09 -04:00
a_BlockType = Chunk - > GetBlock ( a_RelX , a_RelY , a_RelZ ) ;
return true ;
2013-05-28 08:05:23 -04:00
}
bool cChunk : : UnboundedRelGetBlockMeta ( int a_RelX , int a_RelY , int a_RelZ , NIBBLETYPE & a_BlockMeta ) const
{
2015-11-10 17:06:29 -05:00
if ( ! cChunkDef : : IsValidHeight ( a_RelY ) )
2013-05-28 08:05:23 -04:00
{
LOGWARNING ( " %s: requesting a block with a_RelY out of range: %d " , __FUNCTION__ , a_RelY ) ;
return false ;
}
2013-10-22 11:54:09 -04:00
cChunk * Chunk = GetRelNeighborChunkAdjustCoords ( a_RelX , a_RelZ ) ;
2014-10-20 16:55:07 -04:00
if ( ( Chunk = = nullptr ) | | ! Chunk - > IsValid ( ) )
2013-05-28 08:05:23 -04:00
{
2013-10-22 11:54:09 -04:00
// The chunk is not available, bail out
return false ;
2013-05-28 08:05:23 -04:00
}
2013-10-22 11:54:09 -04:00
a_BlockMeta = Chunk - > GetMeta ( a_RelX , a_RelY , a_RelZ ) ;
return true ;
}
2013-05-28 08:05:23 -04:00
2013-10-22 11:54:09 -04:00
bool cChunk : : UnboundedRelGetBlockBlockLight ( int a_RelX , int a_RelY , int a_RelZ , NIBBLETYPE & a_BlockBlockLight ) const
{
2015-11-10 17:06:29 -05:00
if ( ! cChunkDef : : IsValidHeight ( a_RelY ) )
2013-05-28 08:05:23 -04:00
{
2013-10-22 11:54:09 -04:00
LOGWARNING ( " %s: requesting a block with a_RelY out of range: %d " , __FUNCTION__ , a_RelY ) ;
return false ;
2013-05-28 08:05:23 -04:00
}
2013-10-22 11:54:09 -04:00
cChunk * Chunk = GetRelNeighborChunkAdjustCoords ( a_RelX , a_RelZ ) ;
2014-10-20 16:55:07 -04:00
if ( ( Chunk = = nullptr ) | | ! Chunk - > IsValid ( ) )
2013-05-28 08:05:23 -04:00
{
2013-10-22 11:54:09 -04:00
// The chunk is not available, bail out
return false ;
2013-05-28 08:05:23 -04:00
}
2013-10-22 11:54:09 -04:00
a_BlockBlockLight = Chunk - > GetBlockLight ( a_RelX , a_RelY , a_RelZ ) ;
return true ;
}
bool cChunk : : UnboundedRelGetBlockSkyLight ( int a_RelX , int a_RelY , int a_RelZ , NIBBLETYPE & a_BlockSkyLight ) const
{
2015-11-10 17:06:29 -05:00
if ( ! cChunkDef : : IsValidHeight ( a_RelY ) )
2013-05-28 08:05:23 -04:00
{
2013-10-22 11:54:09 -04:00
LOGWARNING ( " %s: requesting a block with a_RelY out of range: %d " , __FUNCTION__ , a_RelY ) ;
return false ;
2013-05-28 08:05:23 -04:00
}
2013-10-22 11:54:09 -04:00
cChunk * Chunk = GetRelNeighborChunkAdjustCoords ( a_RelX , a_RelZ ) ;
2014-10-20 16:55:07 -04:00
if ( ( Chunk = = nullptr ) | | ! Chunk - > IsValid ( ) )
2013-05-28 08:05:23 -04:00
{
2013-10-22 11:54:09 -04:00
// The chunk is not available, bail out
return false ;
2013-05-28 08:05:23 -04:00
}
2013-10-22 11:54:09 -04:00
a_BlockSkyLight = Chunk - > GetSkyLight ( a_RelX , a_RelY , a_RelZ ) ;
return true ;
2013-05-28 08:05:23 -04:00
}
2013-10-27 04:19:13 -04:00
bool cChunk : : UnboundedRelGetBlockLights ( int a_RelX , int a_RelY , int a_RelZ , NIBBLETYPE & a_BlockLight , NIBBLETYPE & a_SkyLight ) const
{
2015-11-10 17:06:29 -05:00
if ( ! cChunkDef : : IsValidHeight ( a_RelY ) )
2013-10-27 04:19:13 -04:00
{
LOGWARNING ( " %s: requesting a block with a_RelY out of range: %d " , __FUNCTION__ , a_RelY ) ;
return false ;
}
cChunk * Chunk = GetRelNeighborChunkAdjustCoords ( a_RelX , a_RelZ ) ;
2014-10-20 16:55:07 -04:00
if ( ( Chunk = = nullptr ) | | ! Chunk - > IsValid ( ) )
2013-10-27 04:19:13 -04:00
{
// The chunk is not available, bail out
return false ;
}
2014-04-26 13:50:23 -04:00
a_BlockLight = Chunk - > GetBlockLight ( a_RelX , a_RelY , a_RelZ ) ;
a_SkyLight = Chunk - > GetSkyLight ( a_RelX , a_RelY , a_RelZ ) ;
2013-10-27 04:19:13 -04:00
return true ;
}
2012-06-14 09:06:06 -04:00
bool cChunk : : UnboundedRelSetBlock ( int a_RelX , int a_RelY , int a_RelZ , BLOCKTYPE a_BlockType , NIBBLETYPE a_BlockMeta )
{
2015-11-10 17:06:29 -05:00
if ( ! cChunkDef : : IsValidHeight ( a_RelY ) )
2012-12-14 17:38:30 -05:00
{
LOGWARNING ( " UnboundedRelSetBlock(): requesting a block with a_RelY out of range: %d " , a_RelY ) ;
return false ;
}
2013-10-22 11:54:09 -04:00
cChunk * Chunk = GetRelNeighborChunkAdjustCoords ( a_RelX , a_RelZ ) ;
2014-10-20 16:55:07 -04:00
if ( ( Chunk = = nullptr ) | | ! Chunk - > IsValid ( ) )
2012-12-14 17:38:30 -05:00
{
2013-10-22 11:54:09 -04:00
// The chunk is not available, bail out
return false ;
2012-12-14 17:38:30 -05:00
}
2013-10-22 11:54:09 -04:00
Chunk - > SetBlock ( a_RelX , a_RelY , a_RelZ , a_BlockType , a_BlockMeta ) ;
return true ;
2014-07-17 10:33:09 -04:00
}
2012-06-14 09:06:06 -04:00
bool cChunk : : UnboundedRelFastSetBlock ( int a_RelX , int a_RelY , int a_RelZ , BLOCKTYPE a_BlockType , NIBBLETYPE a_BlockMeta )
{
2015-11-10 17:06:29 -05:00
if ( ! cChunkDef : : IsValidHeight ( a_RelY ) )
2012-12-14 17:38:30 -05:00
{
LOGWARNING ( " UnboundedRelFastSetBlock(): requesting a block with a_RelY out of range: %d " , a_RelY ) ;
return false ;
}
2013-10-22 11:54:09 -04:00
cChunk * Chunk = GetRelNeighborChunkAdjustCoords ( a_RelX , a_RelZ ) ;
2014-10-20 16:55:07 -04:00
if ( ( Chunk = = nullptr ) | | ! Chunk - > IsValid ( ) )
2012-12-14 17:38:30 -05:00
{
2013-10-22 11:54:09 -04:00
// The chunk is not available, bail out
return false ;
2012-12-14 17:38:30 -05:00
}
2013-10-22 11:54:09 -04:00
Chunk - > FastSetBlock ( a_RelX , a_RelY , a_RelZ , a_BlockType , a_BlockMeta ) ;
return true ;
2012-06-14 09:06:06 -04:00
}
2013-03-22 12:48:45 -04:00
void cChunk : : UnboundedQueueTickBlock ( int a_RelX , int a_RelY , int a_RelZ )
{
2015-11-10 17:06:29 -05:00
if ( ! cChunkDef : : IsValidHeight ( a_RelY ) )
2013-04-27 09:23:20 -04:00
{
// Outside of chunkmap
return ;
}
2013-10-22 11:54:09 -04:00
cChunk * Chunk = GetRelNeighborChunkAdjustCoords ( a_RelX , a_RelZ ) ;
2014-10-20 16:55:07 -04:00
if ( ( Chunk ! = nullptr ) & & Chunk - > IsValid ( ) )
2013-03-22 12:48:45 -04:00
{
2013-10-22 11:54:09 -04:00
Chunk - > QueueTickBlock ( a_RelX , a_RelY , a_RelZ ) ;
2013-03-22 12:48:45 -04:00
}
}
2013-10-22 11:54:09 -04:00
int cChunk : : GetHeight ( int a_X , int a_Z )
2012-06-14 09:06:06 -04:00
{
ASSERT ( ( a_X > = 0 ) & & ( a_X < Width ) & & ( a_Z > = 0 ) & & ( a_Z < Width ) ) ;
2016-01-17 09:09:25 -05:00
2012-06-14 09:06:06 -04:00
if ( ( a_X > = 0 ) & & ( a_X < Width ) & & ( a_Z > = 0 ) & & ( a_Z < Width ) )
{
return m_HeightMap [ a_X + a_Z * Width ] ;
}
return 0 ;
}
void cChunk : : CreateBlockEntities ( void )
{
for ( int x = 0 ; x < Width ; x + + )
{
for ( int z = 0 ; z < Width ; z + + )
{
for ( int y = 0 ; y < Height ; y + + )
{
2014-03-23 10:34:19 -04:00
BLOCKTYPE BlockType = GetBlock ( x , y , z ) ;
2013-04-06 17:21:57 -04:00
switch ( BlockType )
2012-06-14 09:06:06 -04:00
{
2014-04-11 18:01:15 -04:00
case E_BLOCK_BEACON :
2014-07-06 18:50:22 -04:00
case E_BLOCK_TRAPPED_CHEST :
2012-06-14 09:06:06 -04:00
case E_BLOCK_CHEST :
2014-01-18 08:16:47 -05:00
case E_BLOCK_COMMAND_BLOCK :
2012-12-19 16:19:36 -05:00
case E_BLOCK_DISPENSER :
2013-05-26 10:39:04 -04:00
case E_BLOCK_DROPPER :
2013-12-06 19:18:58 -05:00
case E_BLOCK_ENDER_CHEST :
2013-06-20 07:41:44 -04:00
case E_BLOCK_LIT_FURNACE :
2012-06-14 09:06:06 -04:00
case E_BLOCK_FURNACE :
2013-06-13 03:36:43 -04:00
case E_BLOCK_HOPPER :
2012-06-14 09:06:06 -04:00
case E_BLOCK_SIGN_POST :
case E_BLOCK_WALLSIGN :
2014-02-17 14:14:08 -05:00
case E_BLOCK_HEAD :
2012-08-26 13:50:42 -04:00
case E_BLOCK_NOTE_BLOCK :
2012-10-21 03:46:28 -04:00
case E_BLOCK_JUKEBOX :
2014-03-06 19:30:34 -05:00
case E_BLOCK_FLOWER_POT :
2014-09-19 17:00:54 -04:00
case E_BLOCK_MOB_SPAWNER :
2015-09-24 04:48:33 -04:00
case E_BLOCK_BREWING_STAND :
2012-10-21 03:46:28 -04:00
{
2014-09-06 13:01:30 -04:00
if ( ! HasBlockEntityAt ( x + m_PosX * Width , y , z + m_PosZ * Width ) )
2012-10-21 03:46:28 -04:00
{
2017-05-22 16:27:55 -04:00
AddBlockEntityClean ( cBlockEntity : : CreateByBlockType (
2013-11-14 09:37:09 -05:00
BlockType , GetMeta ( x , y , z ) ,
2014-09-06 13:01:30 -04:00
x + m_PosX * Width , y , z + m_PosZ * Width , m_World
2013-11-14 09:37:09 -05:00
) ) ;
2012-10-21 03:46:28 -04:00
}
break ;
}
2012-06-14 09:06:06 -04:00
} // switch (BlockType)
} // for y
} // for z
} // for x
}
2013-03-03 10:33:55 -05:00
void cChunk : : WakeUpSimulators ( void )
{
2014-10-25 16:54:00 -04:00
cSimulator * WaterSimulator = m_World - > GetWaterSimulator ( ) ;
cSimulator * LavaSimulator = m_World - > GetLavaSimulator ( ) ;
cSimulator * RedstoneSimulator = m_World - > GetRedstoneSimulator ( ) ;
2013-03-03 10:33:55 -05:00
int BaseX = m_PosX * cChunkDef : : Width ;
int BaseZ = m_PosZ * cChunkDef : : Width ;
for ( int x = 0 ; x < Width ; x + + )
{
int BlockX = x + BaseX ;
for ( int z = 0 ; z < Width ; z + + )
{
int BlockZ = z + BaseZ ;
for ( int y = GetHeight ( x , z ) ; y > = 0 ; y - - )
{
2014-03-23 10:34:19 -04:00
BLOCKTYPE Block = GetBlock ( x , y , z ) ;
2013-12-13 14:01:15 -05:00
// The redstone sim takes multiple blocks, use the inbuilt checker
if ( RedstoneSimulator - > IsAllowedBlock ( Block ) )
{
RedstoneSimulator - > AddBlock ( BlockX , y , BlockZ , this ) ;
continue ;
}
switch ( Block )
2013-03-03 10:33:55 -05:00
{
case E_BLOCK_WATER :
{
WaterSimulator - > AddBlock ( BlockX , y , BlockZ , this ) ;
break ;
}
case E_BLOCK_LAVA :
{
LavaSimulator - > AddBlock ( BlockX , y , BlockZ , this ) ;
break ;
}
2013-12-12 18:49:10 -05:00
default :
{
break ;
}
2013-03-03 10:33:55 -05:00
} // switch (BlockType)
} // for y
} // for z
} // for x
}
2014-04-07 15:57:14 -04:00
void cChunk : : CalculateHeightmap ( const BLOCKTYPE * a_BlockTypes )
2012-06-14 09:06:06 -04:00
{
for ( int x = 0 ; x < Width ; x + + )
{
for ( int z = 0 ; z < Width ; z + + )
{
for ( int y = Height - 1 ; y > - 1 ; y - - )
{
2014-07-21 09:19:48 -04:00
int index = MakeIndex ( x , y , z ) ;
2014-04-07 15:57:14 -04:00
if ( a_BlockTypes [ index ] ! = E_BLOCK_AIR )
2012-06-14 09:06:06 -04:00
{
2015-07-29 11:04:03 -04:00
m_HeightMap [ x + z * Width ] = static_cast < HEIGHTTYPE > ( y ) ;
2012-06-14 09:06:06 -04:00
break ;
}
} // for y
} // for z
} // for x
}
2014-05-25 08:46:34 -04:00
void cChunk : : SetBlock ( int a_RelX , int a_RelY , int a_RelZ , BLOCKTYPE a_BlockType , NIBBLETYPE a_BlockMeta , bool a_SendToClients )
2012-06-14 09:06:06 -04:00
{
2014-05-25 08:46:34 -04:00
FastSetBlock ( a_RelX , a_RelY , a_RelZ , a_BlockType , a_BlockMeta , a_SendToClients ) ;
2016-01-17 09:09:25 -05:00
2013-08-19 03:19:45 -04:00
// Tick this block and its neighbors:
2014-04-26 13:50:23 -04:00
m_ToTickBlocks . push_back ( Vector3i ( a_RelX , a_RelY , a_RelZ ) ) ;
2013-02-28 02:42:45 -05:00
QueueTickBlockNeighbors ( a_RelX , a_RelY , a_RelZ ) ;
2012-06-14 09:06:06 -04:00
2013-08-19 03:19:45 -04:00
// If there was a block entity, remove it:
2013-02-28 02:42:45 -05:00
Vector3i WorldPos = PositionToWorldPosition ( a_RelX , a_RelY , a_RelZ ) ;
cBlockEntity * BlockEntity = GetBlockEntity ( WorldPos ) ;
2014-10-20 16:55:07 -04:00
if ( BlockEntity ! = nullptr )
2012-06-14 09:06:06 -04:00
{
BlockEntity - > Destroy ( ) ;
2013-02-28 02:42:45 -05:00
RemoveBlockEntity ( BlockEntity ) ;
2012-06-14 09:06:06 -04:00
delete BlockEntity ;
2014-10-20 16:55:07 -04:00
BlockEntity = nullptr ;
2012-06-14 09:06:06 -04:00
}
2016-01-17 09:09:25 -05:00
2013-08-19 03:19:45 -04:00
// If the new block is a block entity, create the entity object:
2013-02-28 02:42:45 -05:00
switch ( a_BlockType )
2012-06-14 09:06:06 -04:00
{
2014-04-11 18:01:15 -04:00
case E_BLOCK_BEACON :
2014-07-06 18:50:22 -04:00
case E_BLOCK_TRAPPED_CHEST :
2012-06-14 09:06:06 -04:00
case E_BLOCK_CHEST :
2014-01-18 08:16:47 -05:00
case E_BLOCK_COMMAND_BLOCK :
2012-12-19 16:19:36 -05:00
case E_BLOCK_DISPENSER :
2013-05-26 10:39:04 -04:00
case E_BLOCK_DROPPER :
2013-12-06 19:18:58 -05:00
case E_BLOCK_ENDER_CHEST :
2013-06-20 07:41:44 -04:00
case E_BLOCK_LIT_FURNACE :
2012-06-14 09:06:06 -04:00
case E_BLOCK_FURNACE :
2013-06-13 03:36:43 -04:00
case E_BLOCK_HOPPER :
2012-06-14 09:06:06 -04:00
case E_BLOCK_SIGN_POST :
case E_BLOCK_WALLSIGN :
2014-02-17 14:14:08 -05:00
case E_BLOCK_HEAD :
2012-08-26 13:50:42 -04:00
case E_BLOCK_NOTE_BLOCK :
2012-10-21 03:46:28 -04:00
case E_BLOCK_JUKEBOX :
2014-03-06 19:30:34 -05:00
case E_BLOCK_FLOWER_POT :
2014-09-19 17:00:54 -04:00
case E_BLOCK_MOB_SPAWNER :
2015-09-24 04:48:33 -04:00
case E_BLOCK_BREWING_STAND :
2012-10-21 03:46:28 -04:00
{
2017-05-22 16:27:55 -04:00
// Fast set block has already marked dirty
AddBlockEntityClean ( cBlockEntity : : CreateByBlockType ( a_BlockType , a_BlockMeta , WorldPos . x , WorldPos . y , WorldPos . z , m_World ) ) ;
2012-10-21 03:46:28 -04:00
break ;
}
2012-06-14 09:06:06 -04:00
} // switch (a_BlockType)
}
2013-02-28 02:42:45 -05:00
void cChunk : : QueueTickBlock ( int a_RelX , int a_RelY , int a_RelZ )
2012-10-14 14:30:16 -04:00
{
2013-04-27 09:23:20 -04:00
ASSERT (
( a_RelX > = 0 ) & & ( a_RelX < Width ) & &
( a_RelY > = 0 ) & & ( a_RelY < Height ) & &
( a_RelZ > = 0 ) & & ( a_RelZ < Width )
) ; // Coords need to be valid
2013-02-28 02:42:45 -05:00
if ( ! IsValid ( ) )
2012-10-14 14:30:16 -04:00
{
2013-02-28 02:42:45 -05:00
return ;
2012-10-14 14:30:16 -04:00
}
2016-01-17 09:09:25 -05:00
2014-04-27 09:45:33 -04:00
m_ToTickBlocks . push_back ( Vector3i ( a_RelX , a_RelY , a_RelZ ) ) ;
2012-10-14 14:30:16 -04:00
}
2013-02-28 02:42:45 -05:00
void cChunk : : QueueTickBlockNeighbors ( int a_RelX , int a_RelY , int a_RelZ )
2012-10-14 14:30:16 -04:00
{
2013-02-28 02:42:45 -05:00
struct
2012-10-14 14:30:16 -04:00
{
2013-02-28 02:42:45 -05:00
int x , y , z ;
2012-10-14 14:30:16 -04:00
}
2013-02-28 02:42:45 -05:00
Coords [ ] =
{
{ 1 , 0 , 0 } ,
{ - 1 , 0 , 0 } ,
{ 0 , 1 , 0 } ,
{ 0 , - 1 , 0 } ,
{ 0 , 0 , 1 } ,
{ 0 , 0 , - 1 } ,
} ;
2013-12-20 10:15:39 -05:00
for ( size_t i = 0 ; i < ARRAYCOUNT ( Coords ) ; i + + )
2013-02-28 02:42:45 -05:00
{
2013-03-22 12:48:45 -04:00
UnboundedQueueTickBlock ( a_RelX + Coords [ i ] . x , a_RelY + Coords [ i ] . y , a_RelZ + Coords [ i ] . z ) ;
2013-02-28 02:42:45 -05:00
} // for i - Coords[]
2012-10-14 14:30:16 -04:00
}
2014-05-25 08:46:34 -04:00
void cChunk : : FastSetBlock ( int a_RelX , int a_RelY , int a_RelZ , BLOCKTYPE a_BlockType , BLOCKTYPE a_BlockMeta , bool a_SendToClients )
2012-06-14 09:06:06 -04:00
{
2013-03-14 17:56:01 -04:00
ASSERT ( ! ( ( a_RelX < 0 ) | | ( a_RelX > = Width ) | | ( a_RelY < 0 ) | | ( a_RelY > = Height ) | | ( a_RelZ < 0 ) | | ( a_RelZ > = Width ) ) ) ;
2012-06-14 09:06:06 -04:00
ASSERT ( IsValid ( ) ) ;
2016-01-17 09:09:25 -05:00
2014-04-26 13:50:23 -04:00
const BLOCKTYPE OldBlockType = GetBlock ( a_RelX , a_RelY , a_RelZ ) ;
2014-05-21 14:58:48 -04:00
const BLOCKTYPE OldBlockMeta = m_ChunkData . GetMeta ( a_RelX , a_RelY , a_RelZ ) ;
2013-03-14 17:56:01 -04:00
if ( ( OldBlockType = = a_BlockType ) & & ( OldBlockMeta = = a_BlockMeta ) )
2012-06-14 09:06:06 -04:00
{
return ;
}
2016-04-18 06:30:23 -04:00
bool ReplacingLiquids = (
( ( OldBlockType = = E_BLOCK_STATIONARY_WATER ) & & ( a_BlockType = = E_BLOCK_WATER ) ) | | // Replacing stationary water with water
( ( OldBlockType = = E_BLOCK_WATER ) & & ( a_BlockType = = E_BLOCK_STATIONARY_WATER ) ) | | // Replacing water with stationary water
( ( OldBlockType = = E_BLOCK_STATIONARY_LAVA ) & & ( a_BlockType = = E_BLOCK_LAVA ) ) | | // Replacing stationary lava with lava
( ( OldBlockType = = E_BLOCK_LAVA ) & & ( a_BlockType = = E_BLOCK_STATIONARY_LAVA ) ) // Replacing lava with stationary lava
) ;
if ( ! ReplacingLiquids )
{
MarkDirty ( ) ;
}
2014-03-23 10:34:19 -04:00
2014-05-21 14:58:48 -04:00
m_ChunkData . SetBlock ( a_RelX , a_RelY , a_RelZ , a_BlockType ) ;
2012-06-14 09:06:06 -04:00
2014-07-20 17:10:31 -04:00
// Queue block to be sent only if ...
if (
2014-07-17 13:13:23 -04:00
a_SendToClients & & // ... we are told to do so AND ...
2014-06-01 15:00:11 -04:00
(
2015-01-25 18:23:30 -05:00
! ( // ... the old and new blocktypes AREN'T leaves (because the client doesn't need meta updates)
( ( OldBlockType = = E_BLOCK_LEAVES ) & & ( a_BlockType = = E_BLOCK_LEAVES ) ) | |
( ( OldBlockType = = E_BLOCK_NEW_LEAVES ) & & ( a_BlockType = = E_BLOCK_NEW_LEAVES ) )
2015-01-25 19:13:55 -05:00
) & & // ... AND ...
(
2016-04-18 06:30:23 -04:00
( OldBlockMeta ! = a_BlockMeta ) | | ( ! ReplacingLiquids )
2014-06-01 15:00:11 -04:00
)
2013-03-22 12:48:45 -04:00
)
)
2012-06-14 09:06:06 -04:00
{
2013-03-14 17:56:01 -04:00
m_PendingSendBlocks . push_back ( sSetBlock ( m_PosX , m_PosZ , a_RelX , a_RelY , a_RelZ , a_BlockType , a_BlockMeta ) ) ;
2012-06-14 09:06:06 -04:00
}
2016-01-17 09:09:25 -05:00
2014-05-21 14:58:48 -04:00
m_ChunkData . SetMeta ( a_RelX , a_RelY , a_RelZ , a_BlockMeta ) ;
2012-06-14 09:06:06 -04:00
// ONLY recalculate lighting if it's necessary!
2014-03-01 14:34:19 -05:00
if (
( cBlockInfo : : GetLightValue ( OldBlockType ) ! = cBlockInfo : : GetLightValue ( a_BlockType ) ) | |
( cBlockInfo : : GetSpreadLightFalloff ( OldBlockType ) ! = cBlockInfo : : GetSpreadLightFalloff ( a_BlockType ) ) | |
( cBlockInfo : : IsTransparent ( OldBlockType ) ! = cBlockInfo : : IsTransparent ( a_BlockType ) )
2012-06-14 09:06:06 -04:00
)
{
m_IsLightValid = false ;
}
// Update heightmap, if needed:
2013-03-14 17:56:01 -04:00
if ( a_RelY > = m_HeightMap [ a_RelX + a_RelZ * Width ] )
2012-06-14 09:06:06 -04:00
{
if ( a_BlockType ! = E_BLOCK_AIR )
{
2015-07-29 11:04:03 -04:00
m_HeightMap [ a_RelX + a_RelZ * Width ] = static_cast < HEIGHTTYPE > ( a_RelY ) ;
2012-06-14 09:06:06 -04:00
}
else
{
2013-03-14 17:56:01 -04:00
for ( int y = a_RelY - 1 ; y > 0 ; - - y )
2012-06-14 09:06:06 -04:00
{
2014-04-27 09:45:33 -04:00
if ( GetBlock ( a_RelX , y , a_RelZ ) ! = E_BLOCK_AIR )
2012-06-14 09:06:06 -04:00
{
2015-07-29 11:04:03 -04:00
m_HeightMap [ a_RelX + a_RelZ * Width ] = static_cast < HEIGHTTYPE > ( y ) ;
2012-06-14 09:06:06 -04:00
break ;
}
} // for y - column in m_BlockData
}
}
}
2012-08-25 13:52:08 -04:00
void cChunk : : SendBlockTo ( int a_RelX , int a_RelY , int a_RelZ , cClientHandle * a_Client )
2012-06-14 09:06:06 -04:00
{
2014-10-20 16:55:07 -04:00
if ( a_Client = = nullptr )
2012-06-14 09:06:06 -04:00
{
2012-08-25 13:52:08 -04:00
// Queue the block for all clients in the chunk (will be sent in Tick())
2014-04-27 09:45:33 -04:00
m_PendingSendBlocks . push_back ( sSetBlock ( m_PosX , m_PosZ , a_RelX , a_RelY , a_RelZ , GetBlock ( a_RelX , a_RelY , a_RelZ ) , GetMeta ( a_RelX , a_RelY , a_RelZ ) ) ) ;
2012-08-25 13:52:08 -04:00
return ;
2012-06-14 09:06:06 -04:00
}
2012-08-25 13:52:08 -04:00
Vector3i wp = PositionToWorldPosition ( a_RelX , a_RelY , a_RelZ ) ;
2014-04-27 09:45:33 -04:00
a_Client - > SendBlockChange ( wp . x , wp . y , wp . z , GetBlock ( a_RelX , a_RelY , a_RelZ ) , GetMeta ( a_RelX , a_RelY , a_RelZ ) ) ;
2016-01-17 09:09:25 -05:00
2012-11-11 03:48:38 -05:00
// FS #268 - if a BlockEntity digging is cancelled by a plugin, the entire block entity must be re-sent to the client:
2017-05-22 16:27:55 -04:00
cBlockEntity * Block = GetBlockEntity ( wp . x , wp . y , wp . z ) ;
if ( Block ! = nullptr )
2012-11-11 03:48:38 -05:00
{
2017-05-22 16:27:55 -04:00
Block - > SendTo ( * a_Client ) ;
}
2012-06-14 09:06:06 -04:00
}
2013-03-01 14:35:29 -05:00
void cChunk : : AddBlockEntity ( cBlockEntity * a_BlockEntity )
2012-06-14 09:06:06 -04:00
{
2013-08-19 03:13:19 -04:00
MarkDirty ( ) ;
2017-05-22 16:27:55 -04:00
AddBlockEntityClean ( a_BlockEntity ) ;
}
void cChunk : : AddBlockEntityClean ( cBlockEntity * a_BlockEntity )
{
int Idx = MakeIndex ( a_BlockEntity - > GetRelX ( ) , a_BlockEntity - > GetPosY ( ) , a_BlockEntity - > GetRelZ ( ) ) ;
auto Result = m_BlockEntities . insert ( { Idx , a_BlockEntity } ) ;
UNUSED ( Result ) ;
ASSERT ( Result . second ) ; // No block entity already at this position
2012-06-14 09:06:06 -04:00
}
2013-03-01 14:35:29 -05:00
cBlockEntity * cChunk : : GetBlockEntity ( int a_BlockX , int a_BlockY , int a_BlockZ )
2012-06-14 09:06:06 -04:00
{
2014-06-24 03:46:38 -04:00
// Check that the query coords are within chunk bounds:
ASSERT ( a_BlockX > = m_PosX * cChunkDef : : Width ) ;
ASSERT ( a_BlockX < m_PosX * cChunkDef : : Width + cChunkDef : : Width ) ;
ASSERT ( a_BlockZ > = m_PosZ * cChunkDef : : Width ) ;
ASSERT ( a_BlockZ < m_PosZ * cChunkDef : : Width + cChunkDef : : Width ) ;
2017-05-22 16:27:55 -04:00
int RelX = a_BlockX - m_PosX * cChunkDef : : Width ;
int RelZ = a_BlockZ - m_PosZ * cChunkDef : : Width ;
2016-01-17 09:09:25 -05:00
2017-05-22 16:27:55 -04:00
auto itr = m_BlockEntities . find ( MakeIndex ( RelX , a_BlockY , RelZ ) ) ;
return ( itr = = m_BlockEntities . end ( ) ) ? nullptr : itr - > second ;
2012-06-14 09:06:06 -04:00
}
2014-07-10 12:18:32 -04:00
bool cChunk : : ShouldBeTicked ( void ) const
{
return ( HasAnyClients ( ) | | ( m_AlwaysTicked > 0 ) ) ;
}
void cChunk : : SetAlwaysTicked ( bool a_AlwaysTicked )
{
if ( a_AlwaysTicked )
{
m_AlwaysTicked + = 1 ;
2016-10-09 08:46:09 -04:00
Stay ( a_AlwaysTicked ) ;
2014-07-10 12:18:32 -04:00
}
else
{
m_AlwaysTicked - = 1 ;
2016-10-09 08:46:09 -04:00
Stay ( a_AlwaysTicked ) ;
2014-07-10 12:18:32 -04:00
}
}
2015-12-01 17:12:44 -05:00
bool cChunk : : UseBlockEntity ( cPlayer * a_Player , int a_X , int a_Y , int a_Z )
2012-06-14 09:06:06 -04:00
{
cBlockEntity * be = GetBlockEntity ( a_X , a_Y , a_Z ) ;
2014-10-20 16:55:07 -04:00
if ( be ! = nullptr )
2012-06-14 09:06:06 -04:00
{
2015-12-01 17:12:44 -05:00
return be - > UsedBy ( a_Player ) ;
2012-06-14 09:06:06 -04:00
}
2015-12-01 17:12:44 -05:00
return false ;
2012-06-14 09:06:06 -04:00
}
2014-02-18 07:06:18 -05:00
void cChunk : : SetBiomeAt ( int a_RelX , int a_RelZ , EMCSBiome a_Biome )
{
cChunkDef : : SetBiome ( m_BiomeMap , a_RelX , a_RelZ , a_Biome ) ;
MarkDirty ( ) ;
}
void cChunk : : SetAreaBiome ( int a_MinRelX , int a_MaxRelX , int a_MinRelZ , int a_MaxRelZ , EMCSBiome a_Biome )
{
for ( int z = a_MinRelZ ; z < = a_MaxRelZ ; z + + )
{
for ( int x = a_MinRelX ; x < = a_MaxRelX ; x + + )
{
cChunkDef : : SetBiome ( m_BiomeMap , x , z , a_Biome ) ;
}
}
MarkDirty ( ) ;
2016-01-17 09:09:25 -05:00
2014-02-18 07:06:18 -05:00
// Re-send the chunk to all clients:
2015-09-16 12:04:05 -04:00
for ( auto ClientHandle : m_LoadedByClient )
2014-02-18 07:06:18 -05:00
{
2015-09-16 12:04:05 -04:00
m_World - > ForceSendChunkTo ( m_PosX , m_PosZ , cChunkSender : : E_CHUNK_PRIORITY_MEDIUM , ClientHandle ) ;
2014-02-18 07:06:18 -05:00
} // for itr - m_LoadedByClient[]
}
2014-10-15 13:01:55 -04:00
void cChunk : : CollectPickupsByPlayer ( cPlayer & a_Player )
2012-06-14 09:06:06 -04:00
{
2014-10-15 13:01:55 -04:00
double PosX = a_Player . GetPosX ( ) ;
double PosY = a_Player . GetPosY ( ) ;
double PosZ = a_Player . GetPosZ ( ) ;
2016-01-17 09:09:25 -05:00
2012-06-14 09:06:06 -04:00
for ( cEntityList : : iterator itr = m_Entities . begin ( ) ; itr ! = m_Entities . end ( ) ; + + itr )
{
2013-11-12 16:43:20 -05:00
if ( ( ! ( * itr ) - > IsPickup ( ) ) & & ( ! ( * itr ) - > IsProjectile ( ) ) )
2012-06-14 09:06:06 -04:00
{
2014-07-17 13:13:23 -04:00
continue ; // Only pickups and projectiles can be picked up
2012-06-14 09:06:06 -04:00
}
2015-07-29 11:04:03 -04:00
float DiffX = static_cast < float > ( ( * itr ) - > GetPosX ( ) - PosX ) ;
float DiffY = static_cast < float > ( ( * itr ) - > GetPosY ( ) - PosY ) ;
float DiffZ = static_cast < float > ( ( * itr ) - > GetPosZ ( ) - PosZ ) ;
2012-06-14 09:06:06 -04:00
float SqrDist = DiffX * DiffX + DiffY * DiffY + DiffZ * DiffZ ;
if ( SqrDist < 1.5f * 1.5f ) // 1.5 block
{
2013-05-17 10:30:18 -04:00
/*
2013-05-13 16:26:45 -04:00
LOG ( " Pickup %d being collected by player \" %s \" , distance %f " ,
( * itr ) - > GetUniqueID ( ) , a_Player - > GetName ( ) . c_str ( ) , SqrDist
) ;
2013-05-17 10:30:18 -04:00
*/
2012-06-14 09:06:06 -04:00
MarkDirty ( ) ;
2013-11-12 16:43:20 -05:00
if ( ( * itr ) - > IsPickup ( ) )
{
( reinterpret_cast < cPickup * > ( * itr ) ) - > CollectedBy ( a_Player ) ;
}
else
{
( reinterpret_cast < cProjectileEntity * > ( * itr ) ) - > CollectedBy ( a_Player ) ;
}
2012-06-14 09:06:06 -04:00
}
2013-05-13 16:26:45 -04:00
else if ( SqrDist < 5 * 5 )
{
2013-05-17 10:30:18 -04:00
/*
2013-05-13 16:26:45 -04:00
LOG ( " Pickup %d close to player \" %s \" , but still too far to collect: %f " ,
( * itr ) - > GetUniqueID ( ) , a_Player - > GetName ( ) . c_str ( ) , SqrDist
) ;
2013-05-17 10:30:18 -04:00
*/
2013-05-13 16:26:45 -04:00
}
2012-06-14 09:06:06 -04:00
}
}
2013-06-12 03:14:06 -04:00
bool cChunk : : SetSignLines ( int a_PosX , int a_PosY , int a_PosZ , const AString & a_Line1 , const AString & a_Line2 , const AString & a_Line3 , const AString & a_Line4 )
2012-06-14 09:06:06 -04:00
{
// Also sends update packets to all clients in the chunk
2017-05-22 16:27:55 -04:00
auto Entity = GetBlockEntity ( a_PosX , a_PosY , a_PosZ ) ;
if ( Entity = = nullptr )
2012-06-14 09:06:06 -04:00
{
2017-05-22 16:27:55 -04:00
return false ; // Not a block entity
}
if (
( Entity - > GetBlockType ( ) ! = E_BLOCK_WALLSIGN ) & &
( Entity - > GetBlockType ( ) ! = E_BLOCK_SIGN_POST )
)
{
return false ; // Not a sign
}
MarkDirty ( ) ;
auto Sign = static_cast < cSignEntity * > ( Entity ) ;
Sign - > SetLines ( a_Line1 , a_Line2 , a_Line3 , a_Line4 ) ;
m_World - > BroadcastBlockEntity ( a_PosX , a_PosY , a_PosZ ) ;
return true ;
2012-06-14 09:06:06 -04:00
}
2015-05-09 05:16:56 -04:00
void cChunk : : RemoveBlockEntity ( cBlockEntity * a_BlockEntity )
2012-06-14 09:06:06 -04:00
{
MarkDirty ( ) ;
2017-05-22 16:27:55 -04:00
ASSERT ( a_BlockEntity ! = nullptr ) ;
int Idx = MakeIndex ( a_BlockEntity - > GetRelX ( ) , a_BlockEntity - > GetPosY ( ) , a_BlockEntity - > GetRelZ ( ) ) ;
m_BlockEntities . erase ( Idx ) ;
2012-06-14 09:06:06 -04:00
}
2014-05-31 17:28:51 -04:00
bool cChunk : : AddClient ( cClientHandle * a_Client )
2012-06-14 09:06:06 -04:00
{
2015-09-16 12:04:05 -04:00
if ( std : : find ( m_LoadedByClient . begin ( ) , m_LoadedByClient . end ( ) , a_Client ) ! = m_LoadedByClient . end ( ) )
2012-06-14 09:06:06 -04:00
{
2015-09-16 12:04:05 -04:00
// Already there, nothing needed
return false ;
2012-06-14 09:06:06 -04:00
}
2015-09-16 12:04:05 -04:00
m_LoadedByClient . push_back ( a_Client ) ;
2012-06-14 09:06:06 -04:00
2014-07-21 09:19:48 -04:00
for ( cEntityList : : iterator itr = m_Entities . begin ( ) ; itr ! = m_Entities . end ( ) ; + + itr )
2012-06-14 09:06:06 -04:00
{
2014-01-16 14:31:06 -05:00
/*
// DEBUG:
LOGD ( " cChunk: Entity #%d (%s) at [%i, %i, %i] spawning for player \" %s \" " ,
( * itr ) - > GetUniqueID ( ) , ( * itr ) - > GetClass ( ) ,
m_PosX , m_PosY , m_PosZ ,
a_Client - > GetUsername ( ) . c_str ( )
) ;
*/
2012-08-24 03:58:26 -04:00
( * itr ) - > SpawnOn ( * a_Client ) ;
2012-06-14 09:06:06 -04:00
}
return true ;
}
2014-05-31 17:28:51 -04:00
void cChunk : : RemoveClient ( cClientHandle * a_Client )
2012-06-14 09:06:06 -04:00
{
2016-04-06 05:16:14 -04:00
auto itr = std : : remove ( m_LoadedByClient . begin ( ) , m_LoadedByClient . end ( ) , a_Client ) ;
2015-09-19 12:22:57 -04:00
// We should always remove at most one client.
2016-04-06 05:16:14 -04:00
ASSERT ( std : : distance ( itr , m_LoadedByClient . end ( ) ) < = 1 ) ;
2015-09-19 12:22:57 -04:00
// Note: itr can equal m_LoadedByClient.end()
2016-04-06 05:16:14 -04:00
m_LoadedByClient . erase ( itr , m_LoadedByClient . end ( ) ) ;
2015-09-16 12:04:05 -04:00
if ( ! a_Client - > IsDestroyed ( ) )
2012-06-14 09:06:06 -04:00
{
2015-09-16 12:04:05 -04:00
for ( auto Entity : m_Entities )
2012-06-14 09:06:06 -04:00
{
2015-09-16 12:04:05 -04:00
/*
// DEBUG:
LOGD ( " chunk [%i, %i] destroying entity #%i for player \" %s \" " ,
m_PosX , m_PosZ ,
( * itr ) - > GetUniqueID ( ) , a_Client - > GetUsername ( ) . c_str ( )
) ;
*/
a_Client - > SendDestroyEntity ( * Entity ) ;
2012-06-14 09:06:06 -04:00
}
2015-09-16 12:04:05 -04:00
}
2014-05-31 17:28:51 -04:00
2015-09-16 12:04:05 -04:00
return ;
2012-06-14 09:06:06 -04:00
}
2014-07-22 05:24:28 -04:00
bool cChunk : : HasClient ( cClientHandle * a_Client )
2012-06-14 09:06:06 -04:00
{
2015-09-16 12:04:05 -04:00
return std : : find ( m_LoadedByClient . begin ( ) , m_LoadedByClient . end ( ) , a_Client ) ! = m_LoadedByClient . end ( ) ;
2012-06-14 09:06:06 -04:00
}
2014-07-10 12:18:32 -04:00
bool cChunk : : HasAnyClients ( void ) const
2012-06-14 09:06:06 -04:00
{
return ! m_LoadedByClient . empty ( ) ;
}
2012-12-21 07:52:14 -05:00
void cChunk : : AddEntity ( cEntity * a_Entity )
2012-06-14 09:06:06 -04:00
{
2012-12-21 07:52:14 -05:00
if ( ! a_Entity - > IsPlayer ( ) )
2012-06-14 09:06:06 -04:00
{
MarkDirty ( ) ;
}
2014-05-31 17:28:51 -04:00
2014-07-22 12:26:48 -04:00
ASSERT ( std : : find ( m_Entities . begin ( ) , m_Entities . end ( ) , a_Entity ) = = m_Entities . end ( ) ) ; // Not there already
2014-05-31 17:28:51 -04:00
2012-12-21 07:52:14 -05:00
m_Entities . push_back ( a_Entity ) ;
2016-02-07 12:07:14 -05:00
ASSERT ( a_Entity - > GetParentChunk ( ) = = nullptr ) ;
a_Entity - > SetParentChunk ( this ) ;
2012-06-14 09:06:06 -04:00
}
void cChunk : : RemoveEntity ( cEntity * a_Entity )
{
2016-02-07 12:07:14 -05:00
ASSERT ( a_Entity - > GetParentChunk ( ) = = this ) ;
a_Entity - > SetParentChunk ( nullptr ) ;
2014-06-21 17:07:38 -04:00
m_Entities . remove ( a_Entity ) ;
2016-02-02 10:37:21 -05:00
// Mark as dirty if it was a server-generated entity:
if ( ! a_Entity - > IsPlayer ( ) )
{
MarkDirty ( ) ;
}
}
2015-03-21 10:18:17 -04:00
bool cChunk : : HasEntity ( UInt32 a_EntityID )
2013-04-13 17:02:10 -04:00
{
for ( cEntityList : : const_iterator itr = m_Entities . begin ( ) , end = m_Entities . end ( ) ; itr ! = end ; + + itr )
{
if ( ( * itr ) - > GetUniqueID ( ) = = a_EntityID )
{
return true ;
}
} // for itr - m_Entities[]
return false ;
}
2012-06-16 04:35:07 -04:00
bool cChunk : : ForEachEntity ( cEntityCallback & a_Callback )
{
// The entity list is locked by the parent chunkmap's CS
2012-06-19 16:31:21 -04:00
for ( cEntityList : : iterator itr = m_Entities . begin ( ) , itr2 = itr ; itr ! = m_Entities . end ( ) ; itr = itr2 )
2012-06-16 04:35:07 -04:00
{
2012-06-19 16:31:21 -04:00
+ + itr2 ;
2016-02-07 12:07:14 -05:00
if ( ! ( * itr ) - > IsTicking ( ) )
2016-02-01 15:49:34 -05:00
{
continue ;
}
2012-06-16 04:35:07 -04:00
if ( a_Callback . Item ( * itr ) )
{
return false ;
}
} // for itr - m_Entitites[]
return true ;
}
2014-09-03 11:00:26 -04:00
bool cChunk : : ForEachEntityInBox ( const cBoundingBox & a_Box , cEntityCallback & a_Callback )
{
// The entity list is locked by the parent chunkmap's CS
for ( cEntityList : : iterator itr = m_Entities . begin ( ) , itr2 = itr ; itr ! = m_Entities . end ( ) ; itr = itr2 )
{
+ + itr2 ;
2016-02-07 12:07:14 -05:00
if ( ! ( * itr ) - > IsTicking ( ) )
2016-02-01 15:49:34 -05:00
{
continue ;
}
2014-09-03 11:00:26 -04:00
cBoundingBox EntBox ( ( * itr ) - > GetPosition ( ) , ( * itr ) - > GetWidth ( ) / 2 , ( * itr ) - > GetHeight ( ) ) ;
if ( ! EntBox . DoesIntersect ( a_Box ) )
{
// The entity is not in the specified box
continue ;
}
if ( a_Callback . Item ( * itr ) )
{
return false ;
}
} // for itr - m_Entitites[]
return true ;
}
2015-03-21 10:18:17 -04:00
bool cChunk : : DoWithEntityByID ( UInt32 a_EntityID , cEntityCallback & a_Callback , bool & a_CallbackResult )
2016-10-12 08:38:45 -04:00
{
return DoWithEntityByID ( a_EntityID , std : : bind ( & cEntityCallback : : Item , & a_Callback , std : : placeholders : : _1 ) , a_CallbackResult ) ;
}
bool cChunk : : DoWithEntityByID ( UInt32 a_EntityID , cLambdaEntityCallback a_Callback , bool & a_CallbackResult )
2013-04-13 17:02:10 -04:00
{
// The entity list is locked by the parent chunkmap's CS
for ( cEntityList : : iterator itr = m_Entities . begin ( ) , end = m_Entities . end ( ) ; itr ! = end ; + + itr )
{
2016-02-07 12:07:14 -05:00
if ( ( ( * itr ) - > GetUniqueID ( ) = = a_EntityID ) & & ( ( * itr ) - > IsTicking ( ) ) )
2013-04-13 17:02:10 -04:00
{
2016-10-12 08:38:45 -04:00
a_CallbackResult = a_Callback ( * itr ) ;
2013-04-13 17:02:10 -04:00
return true ;
}
} // for itr - m_Entitites[]
return false ;
}
2017-05-22 16:27:55 -04:00
template < class tyEntity , BLOCKTYPE . . . tBlocktype >
bool cChunk : : GenericForEachBlockEntity ( cItemCallback < tyEntity > & a_Callback )
2013-11-20 15:53:29 -05:00
{
// The blockentity list is locked by the parent chunkmap's CS
2017-05-22 16:27:55 -04:00
for ( auto & KeyPair : m_BlockEntities )
2013-11-20 15:53:29 -05:00
{
2017-05-22 16:27:55 -04:00
cBlockEntity * Block = KeyPair . second ;
if (
( sizeof . . . ( tBlocktype ) = = 0 ) | | // Let empty list mean all block entities
( IsOneOf < tBlocktype . . . > ( Block - > GetBlockType ( ) ) )
)
2013-11-20 15:53:29 -05:00
{
2017-05-22 16:27:55 -04:00
if ( a_Callback . Item ( static_cast < tyEntity * > ( Block ) ) )
{
return false ;
}
2013-11-20 15:53:29 -05:00
}
2017-05-22 16:27:55 -04:00
} // for KeyPair - m_BlockEntitites[]
2013-11-20 15:53:29 -05:00
return true ;
}
2017-05-22 16:27:55 -04:00
bool cChunk : : ForEachBlockEntity ( cBlockEntityCallback & a_Callback )
{
return GenericForEachBlockEntity < cBlockEntity > ( a_Callback ) ;
}
2015-09-24 04:48:33 -04:00
bool cChunk : : ForEachBrewingstand ( cBrewingstandCallback & a_Callback )
{
2017-05-22 16:27:55 -04:00
return GenericForEachBlockEntity < cBrewingstandEntity ,
E_BLOCK_BREWING_STAND
> ( a_Callback ) ;
2015-09-24 04:48:33 -04:00
}
2012-06-17 15:58:39 -04:00
bool cChunk : : ForEachChest ( cChestCallback & a_Callback )
{
2017-05-22 16:27:55 -04:00
return GenericForEachBlockEntity < cChestEntity ,
E_BLOCK_CHEST
> ( a_Callback ) ;
2012-06-17 15:58:39 -04:00
}
2012-12-26 12:16:33 -05:00
bool cChunk : : ForEachDispenser ( cDispenserCallback & a_Callback )
{
2017-05-22 16:27:55 -04:00
return GenericForEachBlockEntity < cDispenserEntity ,
E_BLOCK_DISPENSER
> ( a_Callback ) ;
2012-12-26 12:16:33 -05:00
}
2013-05-26 10:39:04 -04:00
bool cChunk : : ForEachDropper ( cDropperCallback & a_Callback )
{
2017-05-22 16:27:55 -04:00
return GenericForEachBlockEntity < cDropperEntity ,
E_BLOCK_DROPPER
> ( a_Callback ) ;
2013-05-26 10:39:04 -04:00
}
bool cChunk : : ForEachDropSpenser ( cDropSpenserCallback & a_Callback )
{
2017-05-22 16:27:55 -04:00
return GenericForEachBlockEntity < cDropSpenserEntity ,
E_BLOCK_DISPENSER ,
E_BLOCK_DROPPER
> ( a_Callback ) ;
2013-05-26 10:39:04 -04:00
}
2012-06-17 15:58:39 -04:00
bool cChunk : : ForEachFurnace ( cFurnaceCallback & a_Callback )
{
2017-05-22 16:27:55 -04:00
return GenericForEachBlockEntity < cFurnaceEntity ,
E_BLOCK_FURNACE ,
E_BLOCK_LIT_FURNACE
> ( a_Callback ) ;
2012-06-17 15:58:39 -04:00
}
2017-05-22 16:27:55 -04:00
template < class tyEntity , BLOCKTYPE . . . tBlocktype >
bool cChunk : : GenericDoWithBlockEntityAt ( int a_BlockX , int a_BlockY , int a_BlockZ , cItemCallback < tyEntity > & a_Callback )
2013-11-20 15:53:29 -05:00
{
// The blockentity list is locked by the parent chunkmap's CS
2017-05-22 16:27:55 -04:00
cBlockEntity * Block = GetBlockEntity ( a_BlockX , a_BlockY , a_BlockZ ) ;
if ( Block = = nullptr )
2013-11-20 15:53:29 -05:00
{
2017-05-22 16:27:55 -04:00
return false ; // No block entity here
}
if (
( sizeof . . . ( tBlocktype ) ! = 0 ) & & // Let empty list mean all block entities
( ! IsOneOf < tBlocktype . . . > ( Block - > GetBlockType ( ) ) )
)
{
return false ; // Not any of the given tBlocktypes
}
return ! a_Callback . Item ( static_cast < tyEntity * > ( Block ) ) ;
}
2016-01-17 09:09:25 -05:00
2017-05-22 16:27:55 -04:00
bool cChunk : : DoWithBlockEntityAt ( int a_BlockX , int a_BlockY , int a_BlockZ , cBlockEntityCallback & a_Callback )
{
return GenericDoWithBlockEntityAt < cBlockEntity > ( a_BlockX , a_BlockY , a_BlockZ , a_Callback ) ;
2013-11-20 15:53:29 -05:00
}
2014-07-30 16:19:51 -04:00
bool cChunk : : DoWithBeaconAt ( int a_BlockX , int a_BlockY , int a_BlockZ , cBeaconCallback & a_Callback )
{
2017-05-22 16:27:55 -04:00
return GenericDoWithBlockEntityAt < cBeaconEntity ,
E_BLOCK_BEACON
> ( a_BlockX , a_BlockY , a_BlockZ , a_Callback ) ;
2014-07-30 16:19:51 -04:00
}
2015-09-24 04:48:33 -04:00
bool cChunk : : DoWithBrewingstandAt ( int a_BlockX , int a_BlockY , int a_BlockZ , cBrewingstandCallback & a_Callback )
{
2017-05-22 16:27:55 -04:00
return GenericDoWithBlockEntityAt < cBrewingstandEntity ,
E_BLOCK_BREWING_STAND
> ( a_BlockX , a_BlockY , a_BlockZ , a_Callback ) ;
2015-09-24 04:48:33 -04:00
}
2012-06-17 15:58:39 -04:00
bool cChunk : : DoWithChestAt ( int a_BlockX , int a_BlockY , int a_BlockZ , cChestCallback & a_Callback )
{
2017-05-22 16:27:55 -04:00
return GenericDoWithBlockEntityAt < cChestEntity ,
E_BLOCK_CHEST ,
E_BLOCK_TRAPPED_CHEST
> ( a_BlockX , a_BlockY , a_BlockZ , a_Callback ) ;
2012-06-17 15:58:39 -04:00
}
2012-12-26 12:16:33 -05:00
bool cChunk : : DoWithDispenserAt ( int a_BlockX , int a_BlockY , int a_BlockZ , cDispenserCallback & a_Callback )
{
2017-05-22 16:27:55 -04:00
return GenericDoWithBlockEntityAt < cDispenserEntity ,
E_BLOCK_DISPENSER
> ( a_BlockX , a_BlockY , a_BlockZ , a_Callback ) ;
2012-12-26 12:16:33 -05:00
}
2013-05-26 10:39:04 -04:00
bool cChunk : : DoWithDropperAt ( int a_BlockX , int a_BlockY , int a_BlockZ , cDropperCallback & a_Callback )
{
2017-05-22 16:27:55 -04:00
return GenericDoWithBlockEntityAt < cDropperEntity ,
E_BLOCK_DROPPER
> ( a_BlockX , a_BlockY , a_BlockZ , a_Callback ) ;
2013-05-26 10:39:04 -04:00
}
bool cChunk : : DoWithDropSpenserAt ( int a_BlockX , int a_BlockY , int a_BlockZ , cDropSpenserCallback & a_Callback )
{
2017-05-22 16:27:55 -04:00
return GenericDoWithBlockEntityAt < cDropSpenserEntity ,
E_BLOCK_DISPENSER ,
E_BLOCK_DROPPER
> ( a_BlockX , a_BlockY , a_BlockZ , a_Callback ) ;
2013-05-26 10:39:04 -04:00
}
2012-06-17 15:58:39 -04:00
bool cChunk : : DoWithFurnaceAt ( int a_BlockX , int a_BlockY , int a_BlockZ , cFurnaceCallback & a_Callback )
{
2017-05-22 16:27:55 -04:00
return GenericDoWithBlockEntityAt < cFurnaceEntity ,
E_BLOCK_FURNACE ,
E_BLOCK_LIT_FURNACE
> ( a_BlockX , a_BlockY , a_BlockZ , a_Callback ) ;
2012-06-17 15:58:39 -04:00
}
2013-12-14 11:52:22 -05:00
bool cChunk : : DoWithNoteBlockAt ( int a_BlockX , int a_BlockY , int a_BlockZ , cNoteBlockCallback & a_Callback )
{
2017-05-22 16:27:55 -04:00
return GenericDoWithBlockEntityAt < cNoteEntity ,
E_BLOCK_NOTE_BLOCK
> ( a_BlockX , a_BlockY , a_BlockZ , a_Callback ) ;
2013-12-14 11:52:22 -05:00
}
2014-01-18 08:16:47 -05:00
bool cChunk : : DoWithCommandBlockAt ( int a_BlockX , int a_BlockY , int a_BlockZ , cCommandBlockCallback & a_Callback )
{
2017-05-22 16:27:55 -04:00
return GenericDoWithBlockEntityAt < cCommandBlockEntity ,
E_BLOCK_COMMAND_BLOCK
> ( a_BlockX , a_BlockY , a_BlockZ , a_Callback ) ;
2014-01-18 08:16:47 -05:00
}
2014-03-07 05:44:16 -05:00
bool cChunk : : DoWithMobHeadAt ( int a_BlockX , int a_BlockY , int a_BlockZ , cMobHeadCallback & a_Callback )
2014-02-18 15:40:02 -05:00
{
2017-05-22 16:27:55 -04:00
return GenericDoWithBlockEntityAt < cMobHeadEntity ,
E_BLOCK_HEAD
> ( a_BlockX , a_BlockY , a_BlockZ , a_Callback ) ;
2014-02-18 15:40:02 -05:00
}
2014-03-06 19:30:34 -05:00
bool cChunk : : DoWithFlowerPotAt ( int a_BlockX , int a_BlockY , int a_BlockZ , cFlowerPotCallback & a_Callback )
{
2017-05-22 16:27:55 -04:00
return GenericDoWithBlockEntityAt < cFlowerPotEntity ,
E_BLOCK_FLOWER_POT
> ( a_BlockX , a_BlockY , a_BlockZ , a_Callback ) ;
2014-03-06 19:30:34 -05:00
}
2012-06-17 15:58:39 -04:00
bool cChunk : : GetSignLines ( int a_BlockX , int a_BlockY , int a_BlockZ , AString & a_Line1 , AString & a_Line2 , AString & a_Line3 , AString & a_Line4 )
{
// The blockentity list is locked by the parent chunkmap's CS
2017-05-22 16:27:55 -04:00
auto Entity = GetBlockEntity ( a_BlockX , a_BlockY , a_BlockZ ) ;
if ( Entity = = nullptr )
2012-06-17 15:58:39 -04:00
{
2017-05-22 16:27:55 -04:00
return false ; // Not a block entity
}
if (
( Entity - > GetBlockType ( ) ! = E_BLOCK_WALLSIGN ) & &
( Entity - > GetBlockType ( ) ! = E_BLOCK_SIGN_POST )
)
{
return false ; // Not a sign
}
2016-01-17 09:09:25 -05:00
2017-05-22 16:27:55 -04:00
auto Sign = static_cast < cSignEntity * > ( Entity ) ;
a_Line1 = Sign - > GetLine ( 0 ) ;
a_Line2 = Sign - > GetLine ( 1 ) ;
a_Line3 = Sign - > GetLine ( 2 ) ;
a_Line4 = Sign - > GetLine ( 3 ) ;
return true ;
2012-06-17 15:58:39 -04:00
}
2013-03-15 16:18:11 -04:00
BLOCKTYPE cChunk : : GetBlock ( int a_RelX , int a_RelY , int a_RelZ ) const
2012-06-14 09:06:06 -04:00
{
2014-05-21 14:58:48 -04:00
return m_ChunkData . GetBlock ( a_RelX , a_RelY , a_RelZ ) ;
2012-06-14 09:06:06 -04:00
}
2014-10-19 09:11:53 -04:00
void cChunk : : GetBlockTypeMeta ( int a_RelX , int a_RelY , int a_RelZ , BLOCKTYPE & a_BlockType , NIBBLETYPE & a_BlockMeta ) const
2012-06-14 09:06:06 -04:00
{
2014-04-26 13:50:23 -04:00
a_BlockType = GetBlock ( a_RelX , a_RelY , a_RelZ ) ;
2014-05-21 14:58:48 -04:00
a_BlockMeta = m_ChunkData . GetMeta ( a_RelX , a_RelY , a_RelZ ) ;
2012-06-14 09:06:06 -04:00
}
2012-10-20 07:40:34 -04:00
void cChunk : : GetBlockInfo ( int a_RelX , int a_RelY , int a_RelZ , BLOCKTYPE & a_BlockType , NIBBLETYPE & a_Meta , NIBBLETYPE & a_SkyLight , NIBBLETYPE & a_BlockLight )
{
2014-04-27 09:45:33 -04:00
a_BlockType = GetBlock ( a_RelX , a_RelY , a_RelZ ) ;
2014-05-21 14:58:48 -04:00
a_Meta = m_ChunkData . GetMeta ( a_RelX , a_RelY , a_RelZ ) ;
a_SkyLight = m_ChunkData . GetSkyLight ( a_RelX , a_RelY , a_RelZ ) ;
a_BlockLight = m_ChunkData . GetBlockLight ( a_RelX , a_RelY , a_RelZ ) ;
2012-10-20 07:40:34 -04:00
}
2016-01-17 09:09:25 -05:00
bool cChunk : : GetChunkAndRelByAbsolute ( const Vector3d & a_Position , cChunk * * a_Chunk , Vector3i & a_Rel )
{
return GetChunkAndRelByAbsolute ( Vector3i ( FloorC ( a_Position . x ) , FloorC ( a_Position . y ) , FloorC ( a_Position . z ) ) , a_Chunk , a_Rel ) ;
}
bool cChunk : : GetChunkAndRelByAbsolute ( const Vector3i & a_Position , cChunk * * a_Chunk , Vector3i & a_Rel )
{
* a_Chunk = this - > GetNeighborChunk ( a_Position . x , a_Position . z ) ;
if ( ( * a_Chunk = = nullptr ) | | ! ( * a_Chunk ) - > IsValid ( ) )
{
return false ;
}
a_Rel . x = a_Position . x - ( * a_Chunk ) - > GetPosX ( ) * cChunkDef : : Width ;
a_Rel . y = a_Position . y ;
a_Rel . z = a_Position . z - ( * a_Chunk ) - > GetPosZ ( ) * cChunkDef : : Width ;
return true ;
}
2013-03-01 14:35:29 -05:00
cChunk * cChunk : : GetNeighborChunk ( int a_BlockX , int a_BlockZ )
2013-02-28 02:42:45 -05:00
{
// Convert coords to relative, then call the relative version:
a_BlockX - = m_PosX * cChunkDef : : Width ;
a_BlockZ - = m_PosZ * cChunkDef : : Width ;
2013-03-01 14:35:29 -05:00
return GetRelNeighborChunk ( a_BlockX , a_BlockZ ) ;
2013-02-28 02:42:45 -05:00
}
2013-03-01 14:35:29 -05:00
cChunk * cChunk : : GetRelNeighborChunk ( int a_RelX , int a_RelZ )
2013-02-28 02:42:45 -05:00
{
2014-03-27 13:13:52 -04:00
// If the relative coords are too far away, use the parent's chunk lookup instead:
2014-04-08 11:57:35 -04:00
if ( ( a_RelX < - 128 ) | | ( a_RelX > 128 ) | | ( a_RelZ < - 128 ) | | ( a_RelZ > 128 ) )
2014-03-27 13:13:52 -04:00
{
int BlockX = m_PosX * cChunkDef : : Width + a_RelX ;
int BlockZ = m_PosZ * cChunkDef : : Width + a_RelZ ;
2014-07-06 18:50:22 -04:00
int ChunkX , ChunkZ ;
BlockToChunk ( BlockX , BlockZ , ChunkX , ChunkZ ) ;
2014-08-28 05:36:35 -04:00
return m_ChunkMap - > GetChunkNoLoad ( ChunkX , ChunkZ ) ;
2014-03-27 13:13:52 -04:00
}
// Walk the neighbors:
2013-02-28 02:42:45 -05:00
bool ReturnThis = true ;
if ( a_RelX < 0 )
{
2014-10-20 16:55:07 -04:00
if ( m_NeighborXM ! = nullptr )
2013-02-28 02:42:45 -05:00
{
2013-03-01 14:35:29 -05:00
cChunk * Candidate = m_NeighborXM - > GetRelNeighborChunk ( a_RelX + cChunkDef : : Width , a_RelZ ) ;
2014-10-20 16:55:07 -04:00
if ( Candidate ! = nullptr )
2013-02-28 02:42:45 -05:00
{
return Candidate ;
}
}
// Going X first failed, but if the request is crossing Z as well, let's try the Z first later on.
ReturnThis = false ;
}
else if ( a_RelX > = cChunkDef : : Width )
{
2014-10-20 16:55:07 -04:00
if ( m_NeighborXP ! = nullptr )
2013-02-28 02:42:45 -05:00
{
2013-03-01 14:35:29 -05:00
cChunk * Candidate = m_NeighborXP - > GetRelNeighborChunk ( a_RelX - cChunkDef : : Width , a_RelZ ) ;
2014-10-20 16:55:07 -04:00
if ( Candidate ! = nullptr )
2013-02-28 02:42:45 -05:00
{
return Candidate ;
}
}
// Going X first failed, but if the request is crossing Z as well, let's try the Z first later on.
ReturnThis = false ;
}
2016-01-17 09:09:25 -05:00
2013-02-28 02:42:45 -05:00
if ( a_RelZ < 0 )
{
2014-10-20 16:55:07 -04:00
if ( m_NeighborZM ! = nullptr )
2013-02-28 02:42:45 -05:00
{
2013-03-01 14:35:29 -05:00
return m_NeighborZM - > GetRelNeighborChunk ( a_RelX , a_RelZ + cChunkDef : : Width ) ;
2013-02-28 02:42:45 -05:00
// For requests crossing both X and Z, the X-first way has been already tried
}
2014-10-20 16:55:07 -04:00
return nullptr ;
2013-03-03 14:05:11 -05:00
}
else if ( a_RelZ > = cChunkDef : : Width )
{
2014-10-20 16:55:07 -04:00
if ( m_NeighborZP ! = nullptr )
2013-02-28 02:42:45 -05:00
{
2013-03-01 14:35:29 -05:00
return m_NeighborZP - > GetRelNeighborChunk ( a_RelX , a_RelZ - cChunkDef : : Width ) ;
2013-02-28 02:42:45 -05:00
// For requests crossing both X and Z, the X-first way has been already tried
}
2014-10-20 16:55:07 -04:00
return nullptr ;
2013-02-28 02:42:45 -05:00
}
2016-01-17 09:09:25 -05:00
2014-10-20 16:55:07 -04:00
return ( ReturnThis ? this : nullptr ) ;
2013-02-28 02:42:45 -05:00
}
2013-10-22 11:54:09 -04:00
cChunk * cChunk : : GetRelNeighborChunkAdjustCoords ( int & a_RelX , int & a_RelZ ) const
2013-05-28 08:18:03 -04:00
{
2013-10-22 11:54:09 -04:00
cChunk * ToReturn = const_cast < cChunk * > ( this ) ;
// The most common case: inside this chunk:
if (
( a_RelX > = 0 ) & & ( a_RelX < Width ) & &
( a_RelZ > = 0 ) & & ( a_RelZ < Width )
)
{
return ToReturn ;
}
2016-01-17 09:09:25 -05:00
2013-10-22 11:54:09 -04:00
// Request for a different chunk, calculate chunk offset:
int RelX = a_RelX ; // Make a local copy of the coords (faster access)
2013-05-28 08:18:03 -04:00
int RelZ = a_RelZ ;
2014-10-20 16:55:07 -04:00
while ( ( RelX > = Width ) & & ( ToReturn ! = nullptr ) )
2013-05-28 08:18:03 -04:00
{
2013-10-22 11:54:09 -04:00
RelX - = Width ;
ToReturn = ToReturn - > m_NeighborXP ;
2013-05-28 08:18:03 -04:00
}
2014-10-20 16:55:07 -04:00
while ( ( RelX < 0 ) & & ( ToReturn ! = nullptr ) )
2013-05-28 08:18:03 -04:00
{
2013-10-22 11:54:09 -04:00
RelX + = Width ;
ToReturn = ToReturn - > m_NeighborXM ;
2013-05-28 08:18:03 -04:00
}
2014-10-20 16:55:07 -04:00
while ( ( RelZ > = Width ) & & ( ToReturn ! = nullptr ) )
2013-05-28 08:18:03 -04:00
{
2013-10-22 11:54:09 -04:00
RelZ - = Width ;
ToReturn = ToReturn - > m_NeighborZP ;
2013-05-28 08:18:03 -04:00
}
2014-10-20 16:55:07 -04:00
while ( ( RelZ < 0 ) & & ( ToReturn ! = nullptr ) )
2013-05-28 08:18:03 -04:00
{
2013-10-22 11:54:09 -04:00
RelZ + = Width ;
ToReturn = ToReturn - > m_NeighborZM ;
}
2014-10-20 16:55:07 -04:00
if ( ToReturn ! = nullptr )
2013-10-22 11:54:09 -04:00
{
a_RelX = RelX ;
a_RelZ = RelZ ;
return ToReturn ;
2013-05-28 08:18:03 -04:00
}
2016-01-17 09:09:25 -05:00
2013-10-22 11:54:09 -04:00
// The chunk cannot be walked through neighbors, find it through the chunkmap:
int AbsX = a_RelX + m_PosX * Width ;
int AbsZ = a_RelZ + m_PosZ * Width ;
int DstChunkX , DstChunkZ ;
BlockToChunk ( AbsX , AbsZ , DstChunkX , DstChunkZ ) ;
ToReturn = m_ChunkMap - > FindChunk ( DstChunkX , DstChunkZ ) ;
a_RelX = AbsX - DstChunkX * Width ;
a_RelZ = AbsZ - DstChunkZ * Width ;
return ToReturn ;
2013-05-28 08:18:03 -04:00
}
1.9 / 1.9.2 / 1.9.3 / 1.9.4 protocol support (#3135)
* Semistable update to 15w31a
I'm going through snapshots in a sequential order since it should make things easier, and since protocol version history is written.
* Update to 15w34b protocol
Also, fix an issue with the Entity Equipment packet from the past version. Clients are able to connect and do stuff!
* Partially update to 15w35e
Chunk data doesn't work, but the client joins. I'm waiting to do chunk data because chunk data has an incomplete format until 15w36d.
* Add '/blk' debug command
This command lets one see what block they are looking at, and makes figuring out what's supposed to be where in a highly broken chunk possible.
* Fix CRLF normalization in CheckBasicStyle.lua
Normally, this doesn't cause an issue, but when running from cygwin, it detects the CR as whitespace and creates thousands of violations for every single line. Lua, when run on windows, will normalize automatically, but when run via cygwin, it won't.
The bug was simply that gsub was returning a replaced version, but not changing the parameter, so the replaced version was ignored.
* Update to 15w40b
This includes chunk serialization. Fully functional chunk serialization for 1.9.
I'm not completely happy with the chunk serialization as-is (correct use of palettes would be great), but cuberite also doesn't skip sending empty chunks so this performance optimization should probably come later. The creation of a full buffer is suboptimal, but it's the easiest way to implement this code.
* Write long-by-long rather than creating a buffer
This is a bit faster and should be equivalent. However, the code still doesn't look too good.
* Update to 15w41a protocol
This includes the new set passengers packet, which works off of the ridden entity, not the rider. That means, among other things, that information about the previously ridden vehicle is needed when detaching. So a new method with that info was added.
* Update to 15w45a
* 15w51b protocol
* Update to 1.9.0 protocol
Closes #3067. There are still a few things that need to be worked out (picking up items, effects, particles, and most importantly inventory), but in general this should work. I'll make a few more changes tomorrow to get the rest of the protocol set up, along with 1.9.1/1.9.2 (which did make a few changes). Chunks, however, _are_ working, along with most other parts of the game (placing/breaking blocks).
* Fix item pickup packet not working
That was a silly mistake, but at least it was an easy one.
* 1.9.2 protocol support
* Fix version info found in server list ping
Thus, the client reports that it can connect rather than saying that the server is out of date. This required creating separate classes for 1.9.1 and 1.9.2, unfortunately.
* Fix build errors generated by clang
These didn't happen in MSVC.
* Add protocol19x.cpp and protocol19x.h to CMakeLists
* Ignore warnings in protocol19x that are ignored in protocol18x
* Document BLOCK_FACE and DIG_STATUS constants
* Fix BLOCK_FACE links and add separate section for DIG_STATUS
* Fix bat animation and object spawning
The causes of both of these are explained in #3135, but the gist is that both were typos.
* Implement Use Item packet
This means that buckets, bows, fishing rods, and several other similar items now work when not looking at a block.
* Handle DIG_STATUS_SWAP_ITEM_IN_HAND
* Add support for spawn eggs and potions
The items are transformed from the 1.9 version to the 1.8 version when reading and transformed back when sending.
* Remove spammy potion debug logging
* Fix wolf collar color metadata
The wrong type was being used, causing several clientside issues (including the screen going black).
* Fix 1.9 chunk sending in the nether
The nether and the end don't send skylight.
* Fix clang build errors
* Fix water bottles becoming mundane potions
This happened because the can become splash potion bit got set incorrectly. Water bottles and mundane potions are only differentiated by the fact that water bottles have a metadata of 0, so setting that bit made it a mundane potion.
Also add missing break statements to the read item NBT switch, which would otherwise break items with custom names and also cause incorrect "Unimplemented NBT data when parsing!" logging.
* Copy Protocol18x as Protocol19x
Aditionally, method and class names have been swapped to clean up other diffs. This commit is only added to make the following diffs more readable; it doesn't make any other changes (beyond class names).
* Make thrown potions use the correct appearence
This was caused by potions now using metadata.
* Add missing api doc for cSplashPotionEntity::GetItem
* Fix compile error in SplashPotionEntity.cpp
* Fix fix of cSplashPotionEntity API doc
* Temporarilly disable fall damage particles
These were causing issues in 1.9 due to the changed effect ID.
* Properly send a kick packet when connecting with an invalid version
This means that the client no longer waits on the server screen with no indication whatsoever. However, right now the server list ping isn't implemented for unknown versions, so it'll only load "Old" on the ping.
I also added a GetVarIntSize method to cByteBuffer. This helps clean up part of the code here (and I think it could clean up other parts), but it may make sense for it to be moved elsewhere (or declared in a different way).
* Handle server list pings from unrecognized versions
This isn't the cleanest way of writing it (it feels odd to use ProtocolRecognizer to send packets, and the addition of m_InPingForUnrecognizedVersion feels like the wrong technique), but it works and I can't think of a better way (apart from creating a full separate protocol class to handle only the ping... which would be worse).
* Use cPacketizer for the disconnect packet
This also should fix clang build errors.
* Add 1.9.3 / 1.9.4 support
* Fix incorrect indentation in APIDesc
2016-05-14 15:12:42 -04:00
void cChunk : : BroadcastAttachEntity ( const cEntity & a_Entity , const cEntity & a_Vehicle )
2013-03-03 14:05:11 -05:00
{
2015-09-16 12:04:05 -04:00
for ( auto ClientHandle : m_LoadedByClient )
2013-03-03 14:05:11 -05:00
{
2015-09-16 12:04:05 -04:00
ClientHandle - > SendAttachEntity ( a_Entity , a_Vehicle ) ;
2013-03-03 14:05:11 -05:00
} // for itr - LoadedByClient[]
}
2013-07-07 09:06:06 -04:00
void cChunk : : BroadcastBlockAction ( int a_BlockX , int a_BlockY , int a_BlockZ , char a_Byte1 , char a_Byte2 , BLOCKTYPE a_BlockType , const cClientHandle * a_Exclude )
2012-08-19 07:51:17 -04:00
{
2015-09-16 12:04:05 -04:00
for ( auto itr = m_LoadedByClient . begin ( ) ; itr ! = m_LoadedByClient . end ( ) ; + + itr )
2012-08-19 07:51:17 -04:00
{
if ( * itr = = a_Exclude )
{
continue ;
}
2013-07-07 09:06:06 -04:00
( * itr ) - > SendBlockAction ( a_BlockX , a_BlockY , a_BlockZ , a_Byte1 , a_Byte2 , a_BlockType ) ;
2012-08-19 07:51:17 -04:00
} // for itr - LoadedByClient[]
}
2015-03-21 11:11:57 -04:00
void cChunk : : BroadcastBlockBreakAnimation ( UInt32 a_EntityID , int a_BlockX , int a_BlockY , int a_BlockZ , char a_Stage , const cClientHandle * a_Exclude )
2012-08-19 07:51:17 -04:00
{
2015-09-16 12:04:05 -04:00
for ( auto itr = m_LoadedByClient . begin ( ) ; itr ! = m_LoadedByClient . end ( ) ; + + itr )
2012-08-19 07:51:17 -04:00
{
if ( * itr = = a_Exclude )
{
continue ;
}
2015-03-21 11:11:57 -04:00
( * itr ) - > SendBlockBreakAnim ( a_EntityID , a_BlockX , a_BlockY , a_BlockZ , a_Stage ) ;
2012-08-19 07:51:17 -04:00
} // for itr - LoadedByClient[]
2013-03-17 22:51:55 -04:00
}
2013-07-07 09:06:06 -04:00
void cChunk : : BroadcastBlockEntity ( int a_BlockX , int a_BlockY , int a_BlockZ , const cClientHandle * a_Exclude )
2013-03-17 22:51:55 -04:00
{
2013-07-07 09:06:06 -04:00
// We can operate on entity pointers, we're inside the ChunkMap's CS lock which guards the list
cBlockEntity * Entity = GetBlockEntity ( a_BlockX , a_BlockY , a_BlockZ ) ;
2014-10-20 16:55:07 -04:00
if ( Entity = = nullptr )
2013-07-07 09:06:06 -04:00
{
return ;
}
2015-09-16 12:04:05 -04:00
for ( auto itr = m_LoadedByClient . begin ( ) ; itr ! = m_LoadedByClient . end ( ) ; + + itr )
2013-03-17 22:51:55 -04:00
{
if ( * itr = = a_Exclude )
{
continue ;
}
2013-07-07 09:06:06 -04:00
Entity - > SendTo ( * ( * itr ) ) ;
2013-03-17 22:51:55 -04:00
} // for itr - LoadedByClient[]
2012-08-19 07:51:17 -04:00
}
2016-12-15 14:21:43 -05:00
void cChunk : : BroadcastCollectEntity ( const cEntity & a_Entity , const cPlayer & a_Player , int a_Count , const cClientHandle * a_Exclude )
2012-08-19 15:42:32 -04:00
{
2015-09-16 12:04:05 -04:00
for ( auto itr = m_LoadedByClient . begin ( ) ; itr ! = m_LoadedByClient . end ( ) ; + + itr )
2012-08-19 15:42:32 -04:00
{
if ( * itr = = a_Exclude )
{
continue ;
}
2016-12-15 14:21:43 -05:00
( * itr ) - > SendCollectEntity ( a_Entity , a_Player , a_Count ) ;
2012-08-19 15:42:32 -04:00
} // for itr - LoadedByClient[]
}
2013-07-07 09:06:06 -04:00
void cChunk : : BroadcastDestroyEntity ( const cEntity & a_Entity , const cClientHandle * a_Exclude )
2012-08-19 15:42:32 -04:00
{
2015-09-16 12:04:05 -04:00
for ( auto itr = m_LoadedByClient . begin ( ) ; itr ! = m_LoadedByClient . end ( ) ; + + itr )
2012-08-19 15:42:32 -04:00
{
if ( * itr = = a_Exclude )
{
continue ;
}
2013-07-07 09:06:06 -04:00
( * itr ) - > SendDestroyEntity ( a_Entity ) ;
2012-08-19 15:42:32 -04:00
} // for itr - LoadedByClient[]
}
1.9 / 1.9.2 / 1.9.3 / 1.9.4 protocol support (#3135)
* Semistable update to 15w31a
I'm going through snapshots in a sequential order since it should make things easier, and since protocol version history is written.
* Update to 15w34b protocol
Also, fix an issue with the Entity Equipment packet from the past version. Clients are able to connect and do stuff!
* Partially update to 15w35e
Chunk data doesn't work, but the client joins. I'm waiting to do chunk data because chunk data has an incomplete format until 15w36d.
* Add '/blk' debug command
This command lets one see what block they are looking at, and makes figuring out what's supposed to be where in a highly broken chunk possible.
* Fix CRLF normalization in CheckBasicStyle.lua
Normally, this doesn't cause an issue, but when running from cygwin, it detects the CR as whitespace and creates thousands of violations for every single line. Lua, when run on windows, will normalize automatically, but when run via cygwin, it won't.
The bug was simply that gsub was returning a replaced version, but not changing the parameter, so the replaced version was ignored.
* Update to 15w40b
This includes chunk serialization. Fully functional chunk serialization for 1.9.
I'm not completely happy with the chunk serialization as-is (correct use of palettes would be great), but cuberite also doesn't skip sending empty chunks so this performance optimization should probably come later. The creation of a full buffer is suboptimal, but it's the easiest way to implement this code.
* Write long-by-long rather than creating a buffer
This is a bit faster and should be equivalent. However, the code still doesn't look too good.
* Update to 15w41a protocol
This includes the new set passengers packet, which works off of the ridden entity, not the rider. That means, among other things, that information about the previously ridden vehicle is needed when detaching. So a new method with that info was added.
* Update to 15w45a
* 15w51b protocol
* Update to 1.9.0 protocol
Closes #3067. There are still a few things that need to be worked out (picking up items, effects, particles, and most importantly inventory), but in general this should work. I'll make a few more changes tomorrow to get the rest of the protocol set up, along with 1.9.1/1.9.2 (which did make a few changes). Chunks, however, _are_ working, along with most other parts of the game (placing/breaking blocks).
* Fix item pickup packet not working
That was a silly mistake, but at least it was an easy one.
* 1.9.2 protocol support
* Fix version info found in server list ping
Thus, the client reports that it can connect rather than saying that the server is out of date. This required creating separate classes for 1.9.1 and 1.9.2, unfortunately.
* Fix build errors generated by clang
These didn't happen in MSVC.
* Add protocol19x.cpp and protocol19x.h to CMakeLists
* Ignore warnings in protocol19x that are ignored in protocol18x
* Document BLOCK_FACE and DIG_STATUS constants
* Fix BLOCK_FACE links and add separate section for DIG_STATUS
* Fix bat animation and object spawning
The causes of both of these are explained in #3135, but the gist is that both were typos.
* Implement Use Item packet
This means that buckets, bows, fishing rods, and several other similar items now work when not looking at a block.
* Handle DIG_STATUS_SWAP_ITEM_IN_HAND
* Add support for spawn eggs and potions
The items are transformed from the 1.9 version to the 1.8 version when reading and transformed back when sending.
* Remove spammy potion debug logging
* Fix wolf collar color metadata
The wrong type was being used, causing several clientside issues (including the screen going black).
* Fix 1.9 chunk sending in the nether
The nether and the end don't send skylight.
* Fix clang build errors
* Fix water bottles becoming mundane potions
This happened because the can become splash potion bit got set incorrectly. Water bottles and mundane potions are only differentiated by the fact that water bottles have a metadata of 0, so setting that bit made it a mundane potion.
Also add missing break statements to the read item NBT switch, which would otherwise break items with custom names and also cause incorrect "Unimplemented NBT data when parsing!" logging.
* Copy Protocol18x as Protocol19x
Aditionally, method and class names have been swapped to clean up other diffs. This commit is only added to make the following diffs more readable; it doesn't make any other changes (beyond class names).
* Make thrown potions use the correct appearence
This was caused by potions now using metadata.
* Add missing api doc for cSplashPotionEntity::GetItem
* Fix compile error in SplashPotionEntity.cpp
* Fix fix of cSplashPotionEntity API doc
* Temporarilly disable fall damage particles
These were causing issues in 1.9 due to the changed effect ID.
* Properly send a kick packet when connecting with an invalid version
This means that the client no longer waits on the server screen with no indication whatsoever. However, right now the server list ping isn't implemented for unknown versions, so it'll only load "Old" on the ping.
I also added a GetVarIntSize method to cByteBuffer. This helps clean up part of the code here (and I think it could clean up other parts), but it may make sense for it to be moved elsewhere (or declared in a different way).
* Handle server list pings from unrecognized versions
This isn't the cleanest way of writing it (it feels odd to use ProtocolRecognizer to send packets, and the addition of m_InPingForUnrecognizedVersion feels like the wrong technique), but it works and I can't think of a better way (apart from creating a full separate protocol class to handle only the ping... which would be worse).
* Use cPacketizer for the disconnect packet
This also should fix clang build errors.
* Add 1.9.3 / 1.9.4 support
* Fix incorrect indentation in APIDesc
2016-05-14 15:12:42 -04:00
void cChunk : : BroadcastDetachEntity ( const cEntity & a_Entity , const cEntity & a_PreviousVehicle )
{
for ( auto ClientHandle : m_LoadedByClient )
{
ClientHandle - > SendDetachEntity ( a_Entity , a_PreviousVehicle ) ;
} // for itr - LoadedByClient[]
}
2013-12-15 04:51:46 -05:00
void cChunk : : BroadcastEntityEffect ( const cEntity & a_Entity , int a_EffectID , int a_Amplifier , short a_Duration , const cClientHandle * a_Exclude )
{
2015-09-16 12:04:05 -04:00
for ( auto itr = m_LoadedByClient . begin ( ) ; itr ! = m_LoadedByClient . end ( ) ; + + itr )
2013-12-15 04:51:46 -05:00
{
if ( * itr = = a_Exclude )
{
continue ;
}
( * itr ) - > SendEntityEffect ( a_Entity , a_EffectID , a_Amplifier , a_Duration ) ;
} // for itr - LoadedByClient[]
}
2013-07-07 09:06:06 -04:00
void cChunk : : BroadcastEntityEquipment ( const cEntity & a_Entity , short a_SlotNum , const cItem & a_Item , const cClientHandle * a_Exclude )
2012-08-19 15:42:32 -04:00
{
2015-09-16 12:04:05 -04:00
for ( auto itr = m_LoadedByClient . begin ( ) ; itr ! = m_LoadedByClient . end ( ) ; + + itr )
2012-08-19 15:42:32 -04:00
{
if ( * itr = = a_Exclude )
{
continue ;
}
2013-07-07 09:06:06 -04:00
( * itr ) - > SendEntityEquipment ( a_Entity , a_SlotNum , a_Item ) ;
2012-08-19 15:42:32 -04:00
} // for itr - LoadedByClient[]
}
2013-07-07 09:06:06 -04:00
void cChunk : : BroadcastEntityHeadLook ( const cEntity & a_Entity , const cClientHandle * a_Exclude )
2012-08-19 15:42:32 -04:00
{
2015-09-16 12:04:05 -04:00
for ( auto itr = m_LoadedByClient . begin ( ) ; itr ! = m_LoadedByClient . end ( ) ; + + itr )
2012-08-19 15:42:32 -04:00
{
if ( * itr = = a_Exclude )
{
continue ;
}
2013-07-07 09:06:06 -04:00
( * itr ) - > SendEntityHeadLook ( a_Entity ) ;
2012-08-19 15:42:32 -04:00
} // for itr - LoadedByClient[]
}
2013-07-07 09:06:06 -04:00
void cChunk : : BroadcastEntityLook ( const cEntity & a_Entity , const cClientHandle * a_Exclude )
2012-08-19 15:42:32 -04:00
{
2015-09-16 12:04:05 -04:00
for ( auto itr = m_LoadedByClient . begin ( ) ; itr ! = m_LoadedByClient . end ( ) ; + + itr )
2012-08-19 15:42:32 -04:00
{
if ( * itr = = a_Exclude )
{
continue ;
}
2013-07-07 09:06:06 -04:00
( * itr ) - > SendEntityLook ( a_Entity ) ;
2012-08-19 15:42:32 -04:00
} // for itr - LoadedByClient[]
}
2013-07-07 09:06:06 -04:00
void cChunk : : BroadcastEntityMetadata ( const cEntity & a_Entity , const cClientHandle * a_Exclude )
2012-08-19 17:14:45 -04:00
{
2015-09-16 12:04:05 -04:00
for ( auto itr = m_LoadedByClient . begin ( ) ; itr ! = m_LoadedByClient . end ( ) ; + + itr )
2012-08-19 17:14:45 -04:00
{
if ( * itr = = a_Exclude )
{
continue ;
}
2013-07-07 09:06:06 -04:00
( * itr ) - > SendEntityMetadata ( a_Entity ) ;
2012-08-19 17:14:45 -04:00
} // for itr - LoadedByClient[]
}
2013-07-07 09:06:06 -04:00
void cChunk : : BroadcastEntityRelMove ( const cEntity & a_Entity , char a_RelX , char a_RelY , char a_RelZ , const cClientHandle * a_Exclude )
2012-08-19 17:14:45 -04:00
{
2015-09-16 12:04:05 -04:00
for ( auto itr = m_LoadedByClient . begin ( ) ; itr ! = m_LoadedByClient . end ( ) ; + + itr )
2012-08-19 17:14:45 -04:00
{
if ( * itr = = a_Exclude )
{
continue ;
}
2013-07-07 09:06:06 -04:00
( * itr ) - > SendEntityRelMove ( a_Entity , a_RelX , a_RelY , a_RelZ ) ;
2012-08-19 17:14:45 -04:00
} // for itr - LoadedByClient[]
}
2013-07-07 09:06:06 -04:00
void cChunk : : BroadcastEntityRelMoveLook ( const cEntity & a_Entity , char a_RelX , char a_RelY , char a_RelZ , const cClientHandle * a_Exclude )
2012-08-24 03:58:26 -04:00
{
2015-09-16 12:04:05 -04:00
for ( auto itr = m_LoadedByClient . begin ( ) ; itr ! = m_LoadedByClient . end ( ) ; + + itr )
2012-08-24 03:58:26 -04:00
{
if ( * itr = = a_Exclude )
{
continue ;
}
2013-07-07 09:06:06 -04:00
( * itr ) - > SendEntityRelMoveLook ( a_Entity , a_RelX , a_RelY , a_RelZ ) ;
2012-08-24 03:58:26 -04:00
} // for itr - LoadedByClient[]
}
2013-07-07 09:06:06 -04:00
void cChunk : : BroadcastEntityStatus ( const cEntity & a_Entity , char a_Status , const cClientHandle * a_Exclude )
2012-08-24 05:49:00 -04:00
{
2015-09-16 12:04:05 -04:00
for ( auto itr = m_LoadedByClient . begin ( ) ; itr ! = m_LoadedByClient . end ( ) ; + + itr )
2012-08-24 05:49:00 -04:00
{
if ( * itr = = a_Exclude )
{
continue ;
}
2013-07-07 09:06:06 -04:00
( * itr ) - > SendEntityStatus ( a_Entity , a_Status ) ;
2012-08-24 05:49:00 -04:00
} // for itr - LoadedByClient[]
}
2013-07-07 09:06:06 -04:00
void cChunk : : BroadcastEntityVelocity ( const cEntity & a_Entity , const cClientHandle * a_Exclude )
2012-08-25 17:46:18 -04:00
{
2015-09-16 12:04:05 -04:00
for ( auto itr = m_LoadedByClient . begin ( ) ; itr ! = m_LoadedByClient . end ( ) ; + + itr )
2012-08-25 17:46:18 -04:00
{
if ( * itr = = a_Exclude )
{
continue ;
}
2013-07-07 09:06:06 -04:00
( * itr ) - > SendEntityVelocity ( a_Entity ) ;
2012-08-25 17:46:18 -04:00
} // for itr - LoadedByClient[]
}
2013-12-06 18:47:07 -05:00
void cChunk : : BroadcastEntityAnimation ( const cEntity & a_Entity , char a_Animation , const cClientHandle * a_Exclude )
2012-09-11 08:01:34 -04:00
{
2015-09-16 12:04:05 -04:00
for ( auto itr = m_LoadedByClient . begin ( ) ; itr ! = m_LoadedByClient . end ( ) ; + + itr )
2012-09-11 08:01:34 -04:00
{
if ( * itr = = a_Exclude )
{
continue ;
}
2013-12-06 18:47:07 -05:00
( * itr ) - > SendEntityAnimation ( a_Entity , a_Animation ) ;
2012-09-11 08:01:34 -04:00
} // for itr - LoadedByClient[]
}
2014-09-11 11:03:09 -04:00
void cChunk : : BroadcastParticleEffect ( const AString & a_ParticleName , float a_SrcX , float a_SrcY , float a_SrcZ , float a_OffsetX , float a_OffsetY , float a_OffsetZ , float a_ParticleData , int a_ParticleAmount , cClientHandle * a_Exclude )
2013-12-22 08:45:25 -05:00
{
2015-09-16 12:04:05 -04:00
for ( auto itr = m_LoadedByClient . begin ( ) ; itr ! = m_LoadedByClient . end ( ) ; + + itr )
2013-12-22 08:45:25 -05:00
{
if ( * itr = = a_Exclude )
{
continue ;
}
2014-09-11 11:03:09 -04:00
( * itr ) - > SendParticleEffect ( a_ParticleName , a_SrcX , a_SrcY , a_SrcZ , a_OffsetX , a_OffsetY , a_OffsetZ , a_ParticleData , a_ParticleAmount ) ;
2013-12-22 08:45:25 -05:00
} // for itr - LoadedByClient[]
}
2013-12-15 04:51:46 -05:00
void cChunk : : BroadcastRemoveEntityEffect ( const cEntity & a_Entity , int a_EffectID , const cClientHandle * a_Exclude )
{
2015-09-16 12:04:05 -04:00
for ( auto itr = m_LoadedByClient . begin ( ) ; itr ! = m_LoadedByClient . end ( ) ; + + itr )
2013-12-15 04:51:46 -05:00
{
if ( * itr = = a_Exclude )
{
continue ;
}
( * itr ) - > SendRemoveEntityEffect ( a_Entity , a_EffectID ) ;
} // for itr - LoadedByClient[]
}
2014-07-12 20:08:02 -04:00
void cChunk : : BroadcastSoundEffect ( const AString & a_SoundName , double a_X , double a_Y , double a_Z , float a_Volume , float a_Pitch , const cClientHandle * a_Exclude )
2012-10-21 03:46:28 -04:00
{
2015-09-16 12:04:05 -04:00
for ( auto itr = m_LoadedByClient . begin ( ) ; itr ! = m_LoadedByClient . end ( ) ; + + itr )
2012-10-21 03:46:28 -04:00
{
if ( * itr = = a_Exclude )
{
continue ;
}
2014-07-12 20:08:02 -04:00
( * itr ) - > SendSoundEffect ( a_SoundName , a_X , a_Y , a_Z , a_Volume , a_Pitch ) ;
2012-10-21 03:46:28 -04:00
} // for itr - LoadedByClient[]
}
2015-11-23 18:39:19 -05:00
void cChunk : : BroadcastSoundParticleEffect ( const EffectID a_EffectID , int a_SrcX , int a_SrcY , int a_SrcZ , int a_Data , const cClientHandle * a_Exclude )
2012-09-25 05:54:36 -04:00
{
2015-09-16 12:04:05 -04:00
for ( auto itr = m_LoadedByClient . begin ( ) ; itr ! = m_LoadedByClient . end ( ) ; + + itr )
2012-09-25 05:54:36 -04:00
{
if ( * itr = = a_Exclude )
{
continue ;
}
2013-07-07 09:06:06 -04:00
( * itr ) - > SendSoundParticleEffect ( a_EffectID , a_SrcX , a_SrcY , a_SrcZ , a_Data ) ;
2012-09-25 05:54:36 -04:00
} // for itr - LoadedByClient[]
}
2013-07-07 09:06:06 -04:00
void cChunk : : BroadcastSpawnEntity ( cEntity & a_Entity , const cClientHandle * a_Exclude )
2012-09-29 16:43:42 -04:00
{
2015-09-16 12:04:05 -04:00
for ( auto itr = m_LoadedByClient . begin ( ) ; itr ! = m_LoadedByClient . end ( ) ; + + itr )
2012-09-29 16:43:42 -04:00
{
2013-07-07 09:06:06 -04:00
if ( * itr = = a_Exclude )
{
continue ;
}
a_Entity . SpawnOn ( * ( * itr ) ) ;
2012-09-29 16:43:42 -04:00
} // for itr - LoadedByClient[]
}
2013-07-07 09:06:06 -04:00
void cChunk : : BroadcastThunderbolt ( int a_BlockX , int a_BlockY , int a_BlockZ , const cClientHandle * a_Exclude )
2012-08-26 17:01:07 -04:00
{
2015-09-16 12:04:05 -04:00
for ( auto itr = m_LoadedByClient . begin ( ) ; itr ! = m_LoadedByClient . end ( ) ; + + itr )
2012-08-26 17:01:07 -04:00
{
if ( * itr = = a_Exclude )
{
continue ;
}
2013-07-07 09:06:06 -04:00
( * itr ) - > SendThunderbolt ( a_BlockX , a_BlockY , a_BlockZ ) ;
2012-08-26 17:01:07 -04:00
} // for itr - LoadedByClient[]
}
2014-07-21 09:19:48 -04:00
void cChunk : : BroadcastUseBed ( const cEntity & a_Entity , int a_BlockX , int a_BlockY , int a_BlockZ )
2012-08-24 03:58:26 -04:00
{
2015-09-16 12:04:05 -04:00
for ( auto itr = m_LoadedByClient . begin ( ) ; itr ! = m_LoadedByClient . end ( ) ; + + itr )
2012-08-24 03:58:26 -04:00
{
2013-07-07 09:06:06 -04:00
( * itr ) - > SendUseBed ( a_Entity , a_BlockX , a_BlockY , a_BlockZ ) ;
2012-08-24 03:58:26 -04:00
} // for itr - LoadedByClient[]
}
void cChunk : : SendBlockEntity ( int a_BlockX , int a_BlockY , int a_BlockZ , cClientHandle & a_Client )
{
cBlockEntity * Entity = GetBlockEntity ( a_BlockX , a_BlockY , a_BlockZ ) ;
2014-10-20 16:55:07 -04:00
if ( Entity = = nullptr )
2012-08-24 03:58:26 -04:00
{
return ;
}
Entity - > SendTo ( a_Client ) ;
}
2012-08-25 13:52:08 -04:00
void cChunk : : PositionToWorldPosition ( int a_RelX , int a_RelY , int a_RelZ , int & a_BlockX , int & a_BlockY , int & a_BlockZ )
2012-06-14 09:06:06 -04:00
{
2012-08-25 13:52:08 -04:00
a_BlockY = a_RelY ;
a_BlockX = m_PosX * Width + a_RelX ;
a_BlockZ = m_PosZ * Width + a_RelZ ;
2012-06-14 09:06:06 -04:00
}
2012-08-25 13:52:08 -04:00
Vector3i cChunk : : PositionToWorldPosition ( int a_RelX , int a_RelY , int a_RelZ )
2012-06-14 09:06:06 -04:00
{
2014-09-06 13:01:30 -04:00
return Vector3i ( m_PosX * Width + a_RelX , a_RelY , m_PosZ * Width + a_RelZ ) ;
2012-06-14 09:06:06 -04:00
}
2013-10-29 12:44:51 -04:00
NIBBLETYPE cChunk : : GetTimeAlteredLight ( NIBBLETYPE a_Skylight ) const
{
a_Skylight - = m_World - > GetSkyDarkness ( ) ;
2013-10-30 18:14:42 -04:00
// Because NIBBLETYPE is unsigned, we clamp it to 0 .. 15 by checking for values above 15
2013-10-29 12:44:51 -04:00
return ( a_Skylight < 16 ) ? a_Skylight : 0 ;
}