2013-07-29 07:13:03 -04:00
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include "BlockEntity.h"
|
2013-08-05 04:43:43 -04:00
|
|
|
#include "../BlockArea.h"
|
2013-08-19 05:39:13 -04:00
|
|
|
#include "../Entities/Player.h"
|
2014-03-25 17:26:13 -04:00
|
|
|
#include "MetaRotator.h"
|
2013-07-29 07:13:03 -04:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class cBlockChestHandler :
|
2014-03-25 17:26:13 -04:00
|
|
|
public cMetaRotator<cBlockEntityHandler, 0x07, 0x02, 0x05, 0x03, 0x04>
|
2013-07-29 07:13:03 -04:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
cBlockChestHandler(BLOCKTYPE a_BlockType)
|
2014-03-25 17:26:13 -04:00
|
|
|
: cMetaRotator<cBlockEntityHandler, 0x07, 0x02, 0x05, 0x03, 0x04>(a_BlockType)
|
2013-07-29 07:13:03 -04:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
virtual bool GetPlacementBlockTypeMeta(
|
2014-02-01 08:06:32 -05:00
|
|
|
cChunkInterface & a_ChunkInterface, cPlayer * a_Player,
|
2014-07-17 16:50:58 -04:00
|
|
|
int a_BlockX, int a_BlockY, int a_BlockZ, eBlockFace a_BlockFace,
|
2013-07-29 07:13:03 -04:00
|
|
|
int a_CursorX, int a_CursorY, int a_CursorZ,
|
|
|
|
BLOCKTYPE & a_BlockType, NIBBLETYPE & a_BlockMeta
|
|
|
|
) override
|
|
|
|
{
|
|
|
|
a_BlockType = m_BlockType;
|
|
|
|
|
|
|
|
// Is there a doublechest already next to this block?
|
2014-01-26 09:20:39 -05:00
|
|
|
if (!CanBeAt(a_ChunkInterface, a_BlockX, a_BlockY, a_BlockZ))
|
2013-07-29 07:13:03 -04:00
|
|
|
{
|
|
|
|
// Yup, cannot form a triple-chest, refuse:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check if this forms a doublechest, if so, need to adjust the meta:
|
|
|
|
cBlockArea Area;
|
2014-02-01 08:06:32 -05:00
|
|
|
if (!Area.Read(&a_ChunkInterface, a_BlockX - 1, a_BlockX + 1, a_BlockY, a_BlockY, a_BlockZ - 1, a_BlockZ + 1))
|
2013-07-29 07:13:03 -04:00
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2014-01-17 05:11:17 -05:00
|
|
|
double yaw = a_Player->GetYaw();
|
2013-07-29 07:13:03 -04:00
|
|
|
if (
|
2014-07-06 18:50:22 -04:00
|
|
|
(Area.GetRelBlockType(0, 0, 1) == m_BlockType) ||
|
|
|
|
(Area.GetRelBlockType(2, 0, 1) == m_BlockType)
|
2013-07-29 07:13:03 -04:00
|
|
|
)
|
|
|
|
{
|
2014-01-17 05:11:17 -05:00
|
|
|
a_BlockMeta = ((yaw >= -90) && (yaw < 90)) ? 2 : 3;
|
2013-07-29 07:13:03 -04:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if (
|
2014-07-06 18:50:22 -04:00
|
|
|
(Area.GetRelBlockType(0, 0, 1) == m_BlockType) ||
|
|
|
|
(Area.GetRelBlockType(2, 0, 1) == m_BlockType)
|
2013-07-29 07:13:03 -04:00
|
|
|
)
|
|
|
|
{
|
2014-04-18 15:09:44 -04:00
|
|
|
// FIXME: This is unreachable, as the condition is the same as the above one
|
2014-01-17 05:11:17 -05:00
|
|
|
a_BlockMeta = (yaw < 0) ? 4 : 5;
|
2013-07-29 07:13:03 -04:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Single chest, get meta from rotation only
|
2014-01-17 05:11:17 -05:00
|
|
|
a_BlockMeta = RotationToMetaData(yaw);
|
2013-07-29 07:13:03 -04:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
virtual void OnPlacedByPlayer(
|
2014-07-17 16:50:58 -04:00
|
|
|
cChunkInterface & a_ChunkInterface, cWorldInterface & a_WorldInterface, cPlayer * a_Player,
|
2014-02-04 13:59:05 -05:00
|
|
|
int a_BlockX, int a_BlockY, int a_BlockZ, eBlockFace a_BlockFace,
|
2013-07-29 07:13:03 -04:00
|
|
|
int a_CursorX, int a_CursorY, int a_CursorZ,
|
|
|
|
BLOCKTYPE a_BlockType, NIBBLETYPE a_BlockMeta
|
|
|
|
) override
|
|
|
|
{
|
|
|
|
// Check if this forms a doublechest, if so, need to adjust the meta:
|
|
|
|
cBlockArea Area;
|
2014-02-01 08:06:32 -05:00
|
|
|
if (!Area.Read(&a_ChunkInterface, a_BlockX - 1, a_BlockX + 1, a_BlockY, a_BlockY, a_BlockZ - 1, a_BlockZ + 1))
|
2013-07-29 07:13:03 -04:00
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-01-17 05:11:17 -05:00
|
|
|
double rot = a_Player->GetYaw(); // FIXME: Rename rot to yaw
|
2013-07-29 07:13:03 -04:00
|
|
|
// Choose meta from player rotation, choose only between 2 or 3
|
|
|
|
NIBBLETYPE NewMeta = ((rot >= -90) && (rot < 90)) ? 2 : 3;
|
|
|
|
if (
|
2014-01-26 09:20:39 -05:00
|
|
|
CheckAndAdjustNeighbor(a_ChunkInterface, Area, 0, 1, NewMeta) ||
|
|
|
|
CheckAndAdjustNeighbor(a_ChunkInterface, Area, 2, 1, NewMeta)
|
2013-07-29 07:13:03 -04:00
|
|
|
)
|
|
|
|
{
|
|
|
|
// Forming a double chest in the X direction
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// Choose meta from player rotation, choose only between 4 or 5
|
|
|
|
NewMeta = (rot < 0) ? 4 : 5;
|
|
|
|
if (
|
2014-01-26 09:20:39 -05:00
|
|
|
CheckAndAdjustNeighbor(a_ChunkInterface, Area, 1, 0, NewMeta) ||
|
|
|
|
CheckAndAdjustNeighbor(a_ChunkInterface, Area, 2, 2, NewMeta)
|
2013-07-29 07:13:03 -04:00
|
|
|
)
|
|
|
|
{
|
|
|
|
// Forming a double chest in the Z direction
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Single chest, no further processing needed
|
|
|
|
}
|
|
|
|
|
2014-02-04 12:38:10 -05:00
|
|
|
virtual bool CanBeAt(cChunkInterface & a_ChunkInterface, int a_RelX, int a_RelY, int a_RelZ, const cChunk & a_Chunk) override
|
2014-01-31 18:17:41 -05:00
|
|
|
{
|
2014-02-04 12:38:10 -05:00
|
|
|
int BlockX = a_RelX + a_Chunk.GetPosX() * cChunkDef::Width;
|
|
|
|
int BlockZ = a_RelZ + a_Chunk.GetPosZ() * cChunkDef::Width;
|
|
|
|
return CanBeAt(a_ChunkInterface, BlockX, a_RelY, BlockZ);
|
2014-01-31 18:17:41 -05:00
|
|
|
}
|
|
|
|
|
2013-07-29 07:13:03 -04:00
|
|
|
|
2014-02-01 08:06:32 -05:00
|
|
|
virtual bool CanBeAt(cChunkInterface & a_ChunkInterface, int a_BlockX, int a_BlockY, int a_BlockZ)
|
2013-07-29 07:13:03 -04:00
|
|
|
{
|
|
|
|
cBlockArea Area;
|
2014-02-01 08:06:32 -05:00
|
|
|
if (!Area.Read(&a_ChunkInterface, a_BlockX - 2, a_BlockX + 2, a_BlockY, a_BlockY, a_BlockZ - 2, a_BlockZ + 2))
|
2013-07-29 07:13:03 -04:00
|
|
|
{
|
|
|
|
// Cannot read the surroundings, probably at the edge of loaded chunks. Disallow.
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
int NumChestNeighbors = 0;
|
2014-07-06 18:50:22 -04:00
|
|
|
if (Area.GetRelBlockType(1, 0, 2) == m_BlockType)
|
2013-07-29 07:13:03 -04:00
|
|
|
{
|
|
|
|
if (
|
2014-07-06 18:50:22 -04:00
|
|
|
(Area.GetRelBlockType(0, 0, 2) == m_BlockType) ||
|
|
|
|
(Area.GetRelBlockType(1, 0, 1) == m_BlockType) ||
|
|
|
|
(Area.GetRelBlockType(1, 0, 3) == m_BlockType)
|
2013-07-29 07:13:03 -04:00
|
|
|
)
|
|
|
|
{
|
|
|
|
// Already a doublechest neighbor, disallow:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
NumChestNeighbors += 1;
|
|
|
|
}
|
2014-07-06 18:50:22 -04:00
|
|
|
if (Area.GetRelBlockType(3, 0, 2) == m_BlockType)
|
2013-07-29 07:13:03 -04:00
|
|
|
{
|
|
|
|
if (
|
2014-07-06 18:50:22 -04:00
|
|
|
(Area.GetRelBlockType(4, 0, 2) == m_BlockType) ||
|
|
|
|
(Area.GetRelBlockType(3, 0, 1) == m_BlockType) ||
|
|
|
|
(Area.GetRelBlockType(3, 0, 3) == m_BlockType)
|
2013-07-29 07:13:03 -04:00
|
|
|
)
|
|
|
|
{
|
|
|
|
// Already a doublechest neighbor, disallow:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
NumChestNeighbors += 1;
|
|
|
|
}
|
2014-07-06 18:50:22 -04:00
|
|
|
if (Area.GetRelBlockType(2, 0, 1) == m_BlockType)
|
2013-07-29 07:13:03 -04:00
|
|
|
{
|
|
|
|
if (
|
2014-07-06 18:50:22 -04:00
|
|
|
(Area.GetRelBlockType(2, 0, 0) == m_BlockType) ||
|
|
|
|
(Area.GetRelBlockType(1, 0, 1) == m_BlockType) ||
|
|
|
|
(Area.GetRelBlockType(3, 0, 1) == m_BlockType)
|
2013-07-29 07:13:03 -04:00
|
|
|
)
|
|
|
|
{
|
|
|
|
// Already a doublechest neighbor, disallow:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
NumChestNeighbors += 1;
|
|
|
|
}
|
2014-07-06 18:50:22 -04:00
|
|
|
if (Area.GetRelBlockType(2, 0, 3) == m_BlockType)
|
2013-07-29 07:13:03 -04:00
|
|
|
{
|
|
|
|
if (
|
2014-07-06 18:50:22 -04:00
|
|
|
(Area.GetRelBlockType(2, 0, 4) == m_BlockType) ||
|
|
|
|
(Area.GetRelBlockType(1, 0, 3) == m_BlockType) ||
|
|
|
|
(Area.GetRelBlockType(3, 0, 3) == m_BlockType)
|
2013-07-29 07:13:03 -04:00
|
|
|
)
|
|
|
|
{
|
|
|
|
// Already a doublechest neighbor, disallow:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
NumChestNeighbors += 1;
|
|
|
|
}
|
|
|
|
return (NumChestNeighbors < 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// Translates player rotation when placing a chest into the chest block metadata. Valid for single chests only
|
|
|
|
static NIBBLETYPE RotationToMetaData(double a_Rotation)
|
|
|
|
{
|
|
|
|
a_Rotation += 90 + 45; // So its not aligned with axis
|
|
|
|
|
|
|
|
if (a_Rotation > 360.f)
|
|
|
|
{
|
|
|
|
a_Rotation -= 360.f;
|
|
|
|
}
|
|
|
|
if ((a_Rotation >= 0.f) && (a_Rotation < 90.f))
|
|
|
|
{
|
|
|
|
return 0x4;
|
|
|
|
}
|
|
|
|
else if ((a_Rotation >= 180) && (a_Rotation < 270))
|
|
|
|
{
|
|
|
|
return 0x5;
|
|
|
|
}
|
|
|
|
else if ((a_Rotation >= 90) && (a_Rotation < 180))
|
|
|
|
{
|
|
|
|
return 0x2;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return 0x3;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// If there's a chest in the a_Area in the specified coords, modifies its meta to a_NewMeta and returns true.
|
2014-02-01 08:06:32 -05:00
|
|
|
bool CheckAndAdjustNeighbor(cChunkInterface & a_ChunkInterface, const cBlockArea & a_Area, int a_RelX, int a_RelZ, NIBBLETYPE a_NewMeta)
|
2013-07-29 07:13:03 -04:00
|
|
|
{
|
2014-07-06 18:50:22 -04:00
|
|
|
if (a_Area.GetRelBlockType(a_RelX, 0, a_RelZ) != m_BlockType)
|
2013-07-29 07:13:03 -04:00
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2014-02-01 08:06:32 -05:00
|
|
|
a_ChunkInterface.SetBlockMeta(a_Area.GetOriginX() + a_RelX, a_Area.GetOriginY(), a_Area.GetOriginZ() + a_RelZ, a_NewMeta);
|
2013-07-29 07:13:03 -04:00
|
|
|
return true;
|
|
|
|
}
|
2014-02-02 17:08:57 -05:00
|
|
|
|
|
|
|
|
|
|
|
virtual void ConvertToPickups(cItems & a_Pickups, NIBBLETYPE a_BlockMeta) override
|
|
|
|
{
|
2014-07-06 18:50:22 -04:00
|
|
|
a_Pickups.push_back(cItem(m_BlockType, 1, 0));
|
2014-02-02 17:08:57 -05:00
|
|
|
}
|
2013-07-29 07:13:03 -04:00
|
|
|
} ;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|