2013-07-29 07:13:03 -04:00
|
|
|
#pragma once
|
|
|
|
|
2015-08-19 12:45:53 -04:00
|
|
|
#include "BlockPlant.h"
|
2020-04-03 02:57:01 -04:00
|
|
|
#include "../BlockInfo.h"
|
2013-07-29 07:13:03 -04:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class cBlockCactusHandler :
|
2019-10-11 05:02:53 -04:00
|
|
|
public cClearMetaOnDrop<cBlockPlant<false>>
|
2013-07-29 07:13:03 -04:00
|
|
|
{
|
2020-04-13 12:38:06 -04:00
|
|
|
using Super = cClearMetaOnDrop<cBlockPlant<false>>;
|
2019-10-16 04:06:34 -04:00
|
|
|
|
2013-07-29 07:13:03 -04:00
|
|
|
public:
|
2019-10-16 04:06:34 -04:00
|
|
|
|
|
|
|
cBlockCactusHandler(BLOCKTYPE a_BlockType):
|
2020-04-13 12:38:06 -04:00
|
|
|
Super(a_BlockType)
|
2013-07-29 07:13:03 -04:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2019-10-16 04:06:34 -04:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2014-02-01 08:06:32 -05:00
|
|
|
virtual bool CanBeAt(cChunkInterface & a_ChunkInterface, int a_RelX, int a_RelY, int a_RelZ, const cChunk & a_Chunk) override
|
2013-07-29 07:13:03 -04:00
|
|
|
{
|
|
|
|
if (a_RelY <= 0)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
BLOCKTYPE Surface = a_Chunk.GetBlock(a_RelX, a_RelY - 1, a_RelZ);
|
|
|
|
if ((Surface != E_BLOCK_SAND) && (Surface != E_BLOCK_CACTUS))
|
|
|
|
{
|
|
|
|
// Cactus can only be placed on sand and itself
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-07-18 09:14:52 -04:00
|
|
|
// Check surroundings. Cacti may ONLY be surrounded by non-solid blocks
|
2013-07-29 07:13:03 -04:00
|
|
|
static const struct
|
|
|
|
{
|
|
|
|
int x, z;
|
|
|
|
} Coords[] =
|
|
|
|
{
|
|
|
|
{-1, 0},
|
|
|
|
{ 1, 0},
|
|
|
|
{ 0, -1},
|
|
|
|
{ 0, 1},
|
|
|
|
} ;
|
2013-12-20 10:01:34 -05:00
|
|
|
for (size_t i = 0; i < ARRAYCOUNT(Coords); i++)
|
2013-07-29 07:13:03 -04:00
|
|
|
{
|
|
|
|
BLOCKTYPE BlockType;
|
|
|
|
NIBBLETYPE BlockMeta;
|
|
|
|
if (
|
|
|
|
a_Chunk.UnboundedRelGetBlock(a_RelX + Coords[i].x, a_RelY, a_RelZ + Coords[i].z, BlockType, BlockMeta) &&
|
2017-07-18 09:14:52 -04:00
|
|
|
(
|
|
|
|
cBlockInfo::IsSolid(BlockType) ||
|
|
|
|
(BlockType == E_BLOCK_LAVA) ||
|
|
|
|
(BlockType == E_BLOCK_STATIONARY_LAVA)
|
|
|
|
)
|
2013-07-29 07:13:03 -04:00
|
|
|
)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
} // for i - Coords[]
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-10-11 05:02:53 -04:00
|
|
|
|
|
|
|
|
|
|
|
|
2015-06-30 10:50:15 -04:00
|
|
|
|
|
|
|
virtual ColourID GetMapBaseColourID(NIBBLETYPE a_Meta) override
|
|
|
|
{
|
|
|
|
UNUSED(a_Meta);
|
|
|
|
return 7;
|
|
|
|
}
|
2015-08-19 12:45:53 -04:00
|
|
|
|
|
|
|
|
2019-10-11 05:02:53 -04:00
|
|
|
|
|
|
|
|
|
|
|
virtual int Grow(cChunk & a_Chunk, Vector3i a_RelPos, int a_NumStages = 1) override
|
2015-08-19 12:45:53 -04:00
|
|
|
{
|
2019-10-11 05:02:53 -04:00
|
|
|
// Check the total height of the cacti blocks here:
|
|
|
|
int top = a_RelPos.y + 1;
|
|
|
|
while (
|
|
|
|
(top < cChunkDef::Height) &&
|
|
|
|
(a_Chunk.GetBlock({a_RelPos.x, top, a_RelPos.z}) == E_BLOCK_CACTUS)
|
|
|
|
)
|
|
|
|
{
|
|
|
|
++top;
|
|
|
|
}
|
|
|
|
int bottom = a_RelPos.y - 1;
|
|
|
|
while (
|
|
|
|
(bottom > 0) &&
|
|
|
|
(a_Chunk.GetBlock({a_RelPos.x, bottom, a_RelPos.z}) == E_BLOCK_CACTUS)
|
|
|
|
)
|
|
|
|
{
|
|
|
|
--bottom;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Refuse if already too high:
|
|
|
|
auto numToGrow = std::min(a_NumStages, a_Chunk.GetWorld()->GetMaxCactusHeight() + 1 - (top - bottom));
|
|
|
|
if (numToGrow <= 0)
|
2015-08-19 12:45:53 -04:00
|
|
|
{
|
2019-10-11 05:02:53 -04:00
|
|
|
return 0;
|
2015-08-19 12:45:53 -04:00
|
|
|
}
|
|
|
|
|
2019-10-11 05:02:53 -04:00
|
|
|
BLOCKTYPE blockType;
|
|
|
|
for (int i = 0; i < numToGrow; ++i)
|
|
|
|
{
|
|
|
|
Vector3i pos(a_RelPos.x, top + i, a_RelPos.z);
|
|
|
|
if (!a_Chunk.UnboundedRelGetBlockType(pos, blockType) || (blockType != E_BLOCK_AIR))
|
|
|
|
{
|
|
|
|
// Cannot grow there
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
|
|
|
a_Chunk.UnboundedRelFastSetBlock(pos, E_BLOCK_CACTUS, 0);
|
|
|
|
|
|
|
|
// Check surroundings. Cacti may ONLY be surrounded by non-solid blocks; if they aren't, drop as pickup and bail out the growing
|
|
|
|
static const Vector3i neighborOffsets[] =
|
|
|
|
{
|
|
|
|
{-1, 0, 0},
|
|
|
|
{ 1, 0, 0},
|
|
|
|
{ 0, 0, -1},
|
|
|
|
{ 0, 0, 1},
|
|
|
|
} ;
|
|
|
|
for (const auto & ofs: neighborOffsets)
|
|
|
|
{
|
|
|
|
if (
|
|
|
|
a_Chunk.UnboundedRelGetBlockType(pos + ofs, blockType) &&
|
|
|
|
(
|
|
|
|
cBlockInfo::IsSolid(blockType) ||
|
|
|
|
(blockType == E_BLOCK_LAVA) ||
|
|
|
|
(blockType == E_BLOCK_STATIONARY_LAVA)
|
|
|
|
)
|
|
|
|
)
|
|
|
|
{
|
|
|
|
// Remove the cactus
|
|
|
|
auto absPos = a_Chunk.RelativeToAbsolute(pos);
|
|
|
|
a_Chunk.GetWorld()->DropBlockAsPickups(absPos);
|
|
|
|
return i + 1;
|
|
|
|
}
|
|
|
|
} // for neighbor
|
|
|
|
} // for i - numToGrow
|
|
|
|
return numToGrow;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
protected:
|
|
|
|
|
|
|
|
virtual PlantAction CanGrow(cChunk & a_Chunk, Vector3i a_RelPos) override
|
|
|
|
{
|
|
|
|
// Only allow growing if there's an air block above:
|
|
|
|
if (((a_RelPos.y + 1) < cChunkDef::Height) && (a_Chunk.GetBlock(a_RelPos.addedY(1)) == E_BLOCK_AIR))
|
|
|
|
{
|
2020-04-13 12:38:06 -04:00
|
|
|
return Super::CanGrow(a_Chunk, a_RelPos);
|
2019-10-11 05:02:53 -04:00
|
|
|
}
|
|
|
|
return paStay;
|
2015-08-19 12:45:53 -04:00
|
|
|
}
|
2013-07-29 07:13:03 -04:00
|
|
|
} ;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|