2013-07-29 07:13:03 -04:00
|
|
|
|
2013-08-25 10:11:19 -04:00
|
|
|
// BlockSlab.h
|
|
|
|
|
|
|
|
// Declares cBlockSlabHandler and cBlockDoubleSlabHandler classes
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2013-07-29 07:13:03 -04:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include "BlockHandler.h"
|
2013-08-25 10:11:19 -04:00
|
|
|
#include "../Items/ItemHandler.h"
|
2014-03-31 15:34:11 -04:00
|
|
|
#include "Root.h"
|
2014-09-26 13:13:19 -04:00
|
|
|
#include "ChunkInterface.h"
|
2014-10-19 09:01:01 -04:00
|
|
|
#include "../Entities/Player.h"
|
2013-07-29 07:13:03 -04:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class cBlockSlabHandler :
|
|
|
|
public cBlockHandler
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
cBlockSlabHandler(BLOCKTYPE a_BlockType)
|
|
|
|
: cBlockHandler(a_BlockType)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
virtual void ConvertToPickups(cItems & a_Pickups, NIBBLETYPE a_BlockMeta) override
|
|
|
|
{
|
2014-02-28 09:26:32 -05:00
|
|
|
a_Pickups.push_back(cItem(m_BlockType, 1, a_BlockMeta & 0x7));
|
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;
|
2014-02-28 09:26:32 -05:00
|
|
|
NIBBLETYPE Meta = (NIBBLETYPE) a_Player->GetEquippedItem().m_ItemDamage;
|
2013-08-23 14:38:39 -04:00
|
|
|
|
2014-03-31 15:34:11 -04:00
|
|
|
// Set the correct metadata based on player equipped item (i.e. a_BlockMeta not initialised yet)
|
2013-07-29 07:13:03 -04:00
|
|
|
switch (a_BlockFace)
|
|
|
|
{
|
2013-08-24 13:46:19 -04:00
|
|
|
case BLOCK_FACE_TOP:
|
|
|
|
{
|
2013-08-25 10:11:19 -04:00
|
|
|
// Bottom half slab block
|
|
|
|
a_BlockMeta = Meta & 0x7;
|
|
|
|
break;
|
2013-08-24 13:46:19 -04:00
|
|
|
}
|
2013-08-23 14:38:39 -04:00
|
|
|
case BLOCK_FACE_BOTTOM:
|
|
|
|
{
|
2013-08-25 10:11:19 -04:00
|
|
|
// Top half slab block
|
|
|
|
a_BlockMeta = Meta | 0x8;
|
|
|
|
break;
|
2013-08-23 14:38:39 -04:00
|
|
|
}
|
2013-07-29 07:13:03 -04:00
|
|
|
case BLOCK_FACE_EAST:
|
|
|
|
case BLOCK_FACE_NORTH:
|
|
|
|
case BLOCK_FACE_SOUTH:
|
|
|
|
case BLOCK_FACE_WEST:
|
|
|
|
{
|
|
|
|
if (a_CursorY > 7)
|
|
|
|
{
|
2013-08-24 13:46:19 -04:00
|
|
|
// Cursor at top half of block, place top slab
|
|
|
|
a_BlockMeta = Meta | 0x8; break;
|
2013-07-29 07:13:03 -04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2013-08-24 13:46:19 -04:00
|
|
|
// Cursor at bottom half of block, place bottom slab
|
|
|
|
a_BlockMeta = Meta & 0x7; break;
|
2013-07-29 07:13:03 -04:00
|
|
|
}
|
|
|
|
}
|
2014-04-01 08:55:46 -04:00
|
|
|
case BLOCK_FACE_NONE: return false;
|
2013-07-29 07:13:03 -04:00
|
|
|
} // switch (a_BlockFace)
|
2014-03-31 15:34:11 -04:00
|
|
|
|
|
|
|
// Check if the block at the coordinates is a single slab. Eligibility for combining has already been processed in ClientHandle
|
|
|
|
// Changed to-be-placed to a double slab if we are clicking on a single slab, as opposed to placing one for the first time
|
|
|
|
if (IsAnySlabType(a_ChunkInterface.GetBlock(a_BlockX, a_BlockY, a_BlockZ)))
|
|
|
|
{
|
|
|
|
a_BlockType = GetDoubleSlabType(m_BlockType);
|
2014-06-16 17:35:30 -04:00
|
|
|
a_BlockMeta = a_BlockMeta & 0x7;
|
2014-03-31 15:34:11 -04:00
|
|
|
}
|
|
|
|
|
2013-07-29 07:13:03 -04:00
|
|
|
return true;
|
|
|
|
}
|
2013-08-25 10:11:19 -04:00
|
|
|
|
2014-04-06 15:41:01 -04:00
|
|
|
|
2014-05-07 06:59:48 -04:00
|
|
|
virtual bool CanDirtGrowGrass(NIBBLETYPE a_Meta) override
|
2014-04-06 15:41:01 -04:00
|
|
|
{
|
2014-05-08 14:16:35 -04:00
|
|
|
return ((a_Meta & 0x8) != 0);
|
2014-04-06 15:41:01 -04:00
|
|
|
}
|
|
|
|
|
2013-08-25 10:11:19 -04:00
|
|
|
|
|
|
|
/// Returns true if the specified blocktype is one of the slabs handled by this handler
|
|
|
|
static bool IsAnySlabType(BLOCKTYPE a_BlockType)
|
|
|
|
{
|
2014-09-10 15:50:46 -04:00
|
|
|
return ((a_BlockType == E_BLOCK_WOODEN_SLAB) || (a_BlockType == E_BLOCK_STONE_SLAB) || (a_BlockType == E_BLOCK_NEW_STONE_SLAB));
|
2013-08-25 10:11:19 -04:00
|
|
|
}
|
2014-08-08 12:55:05 -04:00
|
|
|
|
|
|
|
|
|
|
|
virtual void OnCancelRightClick(cChunkInterface & a_ChunkInterface, cWorldInterface & a_WorldInterface, cPlayer * a_Player, int a_BlockX, int a_BlockY, int a_BlockZ, eBlockFace a_BlockFace) override
|
|
|
|
{
|
|
|
|
if ((a_BlockFace == BLOCK_FACE_NONE) || (a_Player->GetEquippedItem().m_ItemType != (short)m_BlockType))
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-08-10 11:12:08 -04:00
|
|
|
// Sends the slab back to the client. It's to refuse a doubleslab placement.
|
2014-08-08 12:55:05 -04:00
|
|
|
a_Player->GetWorld()->SendBlockTo(a_BlockX, a_BlockY, a_BlockZ, a_Player);
|
|
|
|
}
|
2013-08-25 10:11:19 -04:00
|
|
|
|
|
|
|
|
|
|
|
/// Converts the single-slab blocktype to its equivalent double-slab blocktype
|
|
|
|
static BLOCKTYPE GetDoubleSlabType(BLOCKTYPE a_SingleSlabBlockType)
|
|
|
|
{
|
|
|
|
switch (a_SingleSlabBlockType)
|
|
|
|
{
|
|
|
|
case E_BLOCK_STONE_SLAB: return E_BLOCK_DOUBLE_STONE_SLAB;
|
|
|
|
case E_BLOCK_WOODEN_SLAB: return E_BLOCK_DOUBLE_WOODEN_SLAB;
|
2014-09-10 15:50:46 -04:00
|
|
|
case E_BLOCK_NEW_STONE_SLAB: return E_BLOCK_DOUBLE_NEW_STONE_SLAB;
|
2013-08-25 10:11:19 -04:00
|
|
|
}
|
|
|
|
ASSERT(!"Unhandled slab type!");
|
|
|
|
return E_BLOCK_AIR;
|
|
|
|
}
|
|
|
|
|
2014-06-22 18:15:22 -04:00
|
|
|
|
|
|
|
virtual NIBBLETYPE MetaMirrorXZ(NIBBLETYPE a_Meta) override
|
|
|
|
{
|
|
|
|
// Toggle the 4th bit - up / down:
|
|
|
|
return (a_Meta ^ 0x08);
|
|
|
|
}
|
2013-08-25 10:11:19 -04:00
|
|
|
} ;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class cBlockDoubleSlabHandler :
|
|
|
|
public cBlockHandler
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
cBlockDoubleSlabHandler(BLOCKTYPE a_BlockType)
|
|
|
|
: cBlockHandler(a_BlockType)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
virtual void ConvertToPickups(cItems & a_Pickups, NIBBLETYPE a_BlockMeta) override
|
|
|
|
{
|
2014-02-28 09:26:32 -05:00
|
|
|
BLOCKTYPE Block = GetSingleSlabType(m_BlockType);
|
|
|
|
a_Pickups.push_back(cItem(Block, 2, a_BlockMeta & 0x7));
|
|
|
|
}
|
|
|
|
|
|
|
|
inline static BLOCKTYPE GetSingleSlabType(BLOCKTYPE a_BlockType)
|
|
|
|
{
|
|
|
|
switch (a_BlockType)
|
2013-08-25 10:11:19 -04:00
|
|
|
{
|
2014-02-28 09:26:32 -05:00
|
|
|
case E_BLOCK_DOUBLE_STONE_SLAB: return E_BLOCK_STONE_SLAB;
|
|
|
|
case E_BLOCK_DOUBLE_WOODEN_SLAB: return E_BLOCK_WOODEN_SLAB;
|
2014-09-10 15:50:46 -04:00
|
|
|
case E_BLOCK_DOUBLE_NEW_STONE_SLAB: return E_BLOCK_NEW_STONE_SLAB;
|
2013-08-25 10:11:19 -04:00
|
|
|
}
|
2014-02-28 09:26:32 -05:00
|
|
|
ASSERT(!"Unhandled double slab type!");
|
|
|
|
return a_BlockType;
|
2013-08-25 10:11:19 -04:00
|
|
|
}
|
2013-07-29 07:13:03 -04:00
|
|
|
} ;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|