summaryrefslogtreecommitdiffstats
path: root/src/Blocks/BlockSlab.h
blob: b18bf7ef36f5ca540f9a5b72dde7c942942bd22c (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198

// BlockSlab.h

// Declares cBlockSlabHandler and cBlockDoubleSlabHandler classes





#pragma once

#include "BlockHandler.h"
#include "../Items/ItemHandler.h"



class cBlockSlabHandler :
	public cBlockHandler
{
public:
	cBlockSlabHandler(BLOCKTYPE a_BlockType)
		: cBlockHandler(a_BlockType)
	{
	}


	virtual void ConvertToPickups(cItems & a_Pickups, NIBBLETYPE a_BlockMeta) override
	{
		a_Pickups.push_back(cItem(m_BlockType, 1, a_BlockMeta & 0x7));
	}


	virtual bool GetPlacementBlockTypeMeta(
		cChunkInterface & a_ChunkInterface, cPlayer * a_Player,
		int a_BlockX, int a_BlockY, int a_BlockZ, eBlockFace a_BlockFace, 
		int a_CursorX, int a_CursorY, int a_CursorZ,
		BLOCKTYPE & a_BlockType, NIBBLETYPE & a_BlockMeta
	) override
	{
		a_BlockType = m_BlockType;
		BLOCKTYPE  Type = (BLOCKTYPE) (a_Player->GetEquippedItem().m_ItemType);
		NIBBLETYPE Meta = (NIBBLETYPE) a_Player->GetEquippedItem().m_ItemDamage;
		
		// HandlePlaceBlock wants a cItemHandler pointer thing, so let's give it one
		cItemHandler * ItemHandler = cItemHandler::GetItemHandler(GetDoubleSlabType(Type));

		// Check if the block at the coordinates is a slab. Eligibility for combining has already been processed in ClientHandle
		if (IsAnySlabType(a_ChunkInterface.GetBlock(a_BlockX, a_BlockY, a_BlockZ)))
		{
			// Call the function in ClientHandle that places a block when the client sends the packet,
			// so that plugins may interfere with the placement.
			
			if ((a_BlockFace == BLOCK_FACE_TOP) || (a_BlockFace == BLOCK_FACE_BOTTOM))
			{
				// Top and bottom faces need no parameter modification
				a_Player->GetClientHandle()->HandlePlaceBlock(a_BlockX, a_BlockY, a_BlockZ, a_BlockFace, a_CursorX, a_CursorY, a_CursorZ, *ItemHandler);
			}
			else
			{
				// The other faces need to distinguish between top and bottom cursor positions
				if (a_CursorY > 7)
				{
					// Edit the call to use BLOCK_FACE_BOTTOM, otherwise it places incorrectly
					a_Player->GetClientHandle()->HandlePlaceBlock(a_BlockX, a_BlockY, a_BlockZ, BLOCK_FACE_TOP, a_CursorX, a_CursorY, a_CursorZ, *ItemHandler);
				}
				else
				{
					// Edit the call to use BLOCK_FACE_TOP, otherwise it places incorrectly
					a_Player->GetClientHandle()->HandlePlaceBlock(a_BlockX, a_BlockY, a_BlockZ, BLOCK_FACE_BOTTOM, a_CursorX, a_CursorY, a_CursorZ, *ItemHandler);
				}
			}
			return false;  // Cancel the event, because dblslabs were already placed, nothing else needed
		}
		
		// Place the single-slab with correct metas:
		switch (a_BlockFace)
		{
			case BLOCK_FACE_TOP:
			{
				// Bottom half slab block
				a_BlockMeta = Meta & 0x7;
				break;
			}
			case BLOCK_FACE_BOTTOM:
			{
				// Top half slab block
				a_BlockMeta = Meta | 0x8;
				break;
			}
			case BLOCK_FACE_EAST:
			case BLOCK_FACE_NORTH:
			case BLOCK_FACE_SOUTH:
			case BLOCK_FACE_WEST:
			{
				if (a_CursorY > 7)
				{
					// Cursor at top half of block, place top slab
					a_BlockMeta = Meta | 0x8; break;
				}
				else
				{
					// Cursor at bottom half of block, place bottom slab
					a_BlockMeta = Meta & 0x7; break;
				}
			}
		}  // switch (a_BlockFace)
		return true;
	}
	
	
	virtual const char * GetStepSound(void) override
	{
		switch (m_BlockType)
		{
			case E_BLOCK_WOODEN_SLAB: return "step.wood";
			case E_BLOCK_STONE_SLAB:  return "step.stone";
		}
		ASSERT(!"Unhandled slab type!");
		return "";
	}

	
	/// Returns true if the specified blocktype is one of the slabs handled by this handler
	static bool IsAnySlabType(BLOCKTYPE a_BlockType)
	{
		return ((a_BlockType == E_BLOCK_WOODEN_SLAB) || (a_BlockType == E_BLOCK_STONE_SLAB));
	}
	
	
	/// 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;
		}
		ASSERT(!"Unhandled slab type!");
		return E_BLOCK_AIR;
	}
	
} ;





class cBlockDoubleSlabHandler :
	public cBlockHandler
{
public:
	cBlockDoubleSlabHandler(BLOCKTYPE a_BlockType)
		: cBlockHandler(a_BlockType)
	{
	}


	virtual void ConvertToPickups(cItems & a_Pickups, NIBBLETYPE a_BlockMeta) override
	{
		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)
		{
			case E_BLOCK_DOUBLE_STONE_SLAB:  return E_BLOCK_STONE_SLAB;
			case E_BLOCK_DOUBLE_WOODEN_SLAB: return E_BLOCK_WOODEN_SLAB;
		}
		ASSERT(!"Unhandled double slab type!");
		return a_BlockType;
	}
	
	virtual const char * GetStepSound(void) override
	{
		switch (m_BlockType)
		{
			case E_BLOCK_DOUBLE_STONE_SLAB:  return "step.stone";
			case E_BLOCK_DOUBLE_WOODEN_SLAB: return "step.wood";
		}
		ASSERT(!"Unhandled double slab type!");
		return "";
	}


	virtual NIBBLETYPE MetaMirrorXZ(NIBBLETYPE a_Meta) override
	{
		NIBBLETYPE OtherMeta = a_Meta & 0x07;  // Contains unrelate meta data.

		// 8th bit is up/down.  1 right-side-up, 0 is up-side-down.
		return (a_Meta & 0x08) ? 0x00 + OtherMeta : 0x01 + OtherMeta;
	}
} ;