1
0

Adding more customize options to mob spawners and improving the way to look for surrounding entities (#4955)

* added nearly any customize option

* fixed unnecessary diff
added comments

* removed unnecessary const qualifier

* fixed build

* changed to ForEachEntityInBox

* added docs

* updated lua api description

* checkstyle

* added changes suggested by @peterbell10
And now the player may break the server by setting ridiculous ranges

* updated docs
changed cast to static cast

* fixed clang

* fixed clang on WSSAnvil.cpp

Co-authored-by: 12xx12 <12xx12100@gmail.com>
This commit is contained in:
12xx12 2020-10-10 21:31:44 +02:00 committed by GitHub
parent b0b1ccddd1
commit c0711407e9
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 241 additions and 87 deletions

View File

@ -1353,7 +1353,7 @@ World:ForEachChestInChunk(Player:GetChunkX(), Player:GetChunkZ(),
Type = "number", Type = "number",
}, },
}, },
Notes = "Returns the amount of this monster type in a 8-block radius (Y: 4-block radius).", Notes = "Returns the amount of this monster type in a radius defined by SetSpawnRange (Y: 4-block radius).",
}, },
GetNearbyPlayersNum = GetNearbyPlayersNum =
{ {
@ -1365,6 +1365,22 @@ World:ForEachChestInChunk(Player:GetChunkX(), Player:GetChunkZ(),
}, },
Notes = "Returns the amount of the nearby players in a 16-block radius.", Notes = "Returns the amount of the nearby players in a 16-block radius.",
}, },
GetSpawnCount =
{
Returns =
{
Type = "number",
},
Notes = "Returns the number of entities the spawner will try to spawn on each activation.",
},
GetSpawnRange =
{
Returns =
{
Type = "number",
},
Notes = "Returns half the length of the square the spawner tries to spawn entities in.",
},
GetSpawnDelay = GetSpawnDelay =
{ {
Returns = Returns =
@ -1375,6 +1391,38 @@ World:ForEachChestInChunk(Player:GetChunkX(), Player:GetChunkZ(),
}, },
Notes = "Returns the spawn delay. This is the tick delay that is needed to spawn new monsters.", Notes = "Returns the spawn delay. This is the tick delay that is needed to spawn new monsters.",
}, },
GetMinSpawnDelay =
{
Returns =
{
Type = "number",
},
Notes = "Returns the minimum number of ticks the spawner waits until spawning new entities automatically.",
},
GetMaxSpawnDelay =
{
Returns =
{
Type = "number",
},
Notes = "Returns the maximum number of ticks the spawner waits until spawning new entities automatically.",
},
GetMaxNearbyEntities =
{
Returns =
{
Type = "number",
},
Notes = "Returns the maximum number of entities of the same type that can be present before the spawner cannot spawn more entities.",
},
GetRequiredPlayerRange =
{
Returns =
{
Type = "number"
},
Notes = "Returns the maximum euclidean distance from a player where the spawner can be activated.",
},
ResetTimer = ResetTimer =
{ {
Notes = "Sets the spawn delay to a new random value.", Notes = "Sets the spawn delay to a new random value.",
@ -1401,13 +1449,79 @@ World:ForEachChestInChunk(Player:GetChunkX(), Player:GetChunkZ(),
}, },
Notes = "Sets the spawn delay.", Notes = "Sets the spawn delay.",
}, },
SetSpawnCount =
{
Params =
{
{
Name = "SpawnCount",
Type = "number",
},
},
Notes = "Sets the number of entities the spawner will try to spawn in each activation. Might not spawn all of them due to spawn limitations of the entity.",
},
SetSpawnRange =
{
Params =
{
{
Name = "SpawnRange",
Type = "number",
},
},
Notes = "Sets half the length of the square the spawner will try to spawn entities in.",
},
SetMinSpawnDelay =
{
Params =
{
{
Name = "MinSpawnDelay",
Type = "number",
},
},
Notes = "Sets the minimum amount of ticks the spawner will wait before spawning new entities.",
},
SetMaxSpawnDelay =
{
Params =
{
{
Name = "MaxSpawnDelay",
Type = "number",
},
},
Notes = "Sets the maximum amount of ticks the spawner will wait before spawning new entities.",
},
SetMaxNearbyEntities =
{
Params =
{
{
Name = "MaxNearbyEntities",
Type = "number",
},
},
Notes = "Sets the maximum amount of nearby entities until the spawner will stop spawning this entity type.",
},
SetRequiredPlayerRange =
{
Params =
{
{
Name = "RequiredPlayerRange",
Type = "number",
},
},
Notes = "Sets the maximum euclidean distance from a player where the spawner can be activated.",
},
SpawnEntity = SpawnEntity =
{ {
Notes = "Spawns the entity. This function automaticly change the spawn delay!", Notes = "Spawns the entity. NOTE: This function resets the delay before the next automatic activation of the spawner.",
}, },
UpdateActiveState = UpdateActiveState =
{ {
Notes = "Upate the active flag from the mob spawner. This function is called every 5 seconds from the Tick() function.", Notes = "Update the active flag from the mob spawner. This function is called every 5 seconds from the Tick() function.",
}, },
}, },
Inherits = "cBlockEntity", Inherits = "cBlockEntity",

View File

@ -33,6 +33,12 @@ void cMobSpawnerEntity::CopyFrom(const cBlockEntity & a_Src)
m_Entity = src.m_Entity; m_Entity = src.m_Entity;
m_IsActive = src.m_IsActive; m_IsActive = src.m_IsActive;
m_SpawnDelay = src.m_SpawnDelay; m_SpawnDelay = src.m_SpawnDelay;
m_SpawnCount = src.m_SpawnCount;
m_SpawnRange = src.m_SpawnRange;
m_MinSpawnDelay = src.m_MinSpawnDelay;
m_MaxSpawnDelay = src.m_MaxSpawnDelay;
m_MaxNearbyEntities = src.m_MaxNearbyEntities;
m_RequiredPlayerRange = src.m_RequiredPlayerRange;
} }
@ -114,7 +120,7 @@ bool cMobSpawnerEntity::Tick(std::chrono::milliseconds a_Dt, cChunk & a_Chunk)
void cMobSpawnerEntity::ResetTimer(void) void cMobSpawnerEntity::ResetTimer(void)
{ {
m_SpawnDelay = GetRandomProvider().RandInt<short>(200, 800); m_SpawnDelay = GetRandomProvider().RandInt<short>(m_MinSpawnDelay, m_MaxSpawnDelay);
m_World->BroadcastBlockEntity(GetPos()); m_World->BroadcastBlockEntity(GetPos());
} }
@ -125,7 +131,7 @@ void cMobSpawnerEntity::ResetTimer(void)
void cMobSpawnerEntity::SpawnEntity(void) void cMobSpawnerEntity::SpawnEntity(void)
{ {
auto NearbyEntities = GetNearbyMonsterNum(m_Entity); auto NearbyEntities = GetNearbyMonsterNum(m_Entity);
if (NearbyEntities >= 6) if (NearbyEntities >= m_MaxNearbyEntities)
{ {
ResetTimer(); ResetTimer();
return; return;
@ -136,18 +142,18 @@ void cMobSpawnerEntity::SpawnEntity(void)
auto & Random = GetRandomProvider(); auto & Random = GetRandomProvider();
bool HaveSpawnedEntity = false; bool HaveSpawnedEntity = false;
for (size_t i = 0; i < 4; i++) for (short I = 0; I < m_SpawnCount; I++)
{ {
if (NearbyEntities >= 6) if (NearbyEntities >= m_MaxNearbyEntities)
{ {
break; break;
} }
auto SpawnRelPos(GetRelPos()); auto SpawnRelPos(GetRelPos());
SpawnRelPos += Vector3i( SpawnRelPos += Vector3i(
static_cast<int>((Random.RandReal<double>() - Random.RandReal<double>()) * 4.0), static_cast<int>((Random.RandReal<double>() - Random.RandReal<double>()) * static_cast<double>(m_SpawnRange)),
Random.RandInt(-1, 1), Random.RandInt(-1, 1),
static_cast<int>((Random.RandReal<double>() - Random.RandReal<double>()) * 4.0) static_cast<int>((Random.RandReal<double>() - Random.RandReal<double>()) * static_cast<double>(m_SpawnRange))
); );
auto Chunk = a_Chunk.GetRelNeighborChunkAdjustCoords(SpawnRelPos); auto Chunk = a_Chunk.GetRelNeighborChunkAdjustCoords(SpawnRelPos);
@ -191,39 +197,24 @@ void cMobSpawnerEntity::SpawnEntity(void)
int cMobSpawnerEntity::GetNearbyPlayersNum(void) int cMobSpawnerEntity::GetNearbyPlayersNum(void)
{ {
auto SpawnerPos = Vector3d(0.5, 0.5, 0.5) + m_Pos;
int NumPlayers = 0; int NumPlayers = 0;
class cCallback : public cChunkDataCallback auto Callback = [&] (cEntity & a_Entity)
{ {
public: if (!a_Entity.IsPlayer())
cCallback(Vector3d a_SpawnerPos, int & a_NumPlayers) :
m_SpawnerPos(a_SpawnerPos),
m_NumPlayers(a_NumPlayers)
{ {
return false;
} }
if ((m_Pos - a_Entity.GetPosition()).Length() <= m_RequiredPlayerRange)
virtual void Entity(cEntity * a_Entity) override
{ {
if (!a_Entity->IsPlayer()) NumPlayers++;
{
return;
} }
return false;
};
if ((m_SpawnerPos - a_Entity->GetPosition()).Length() <= 16) auto PlayerBoundingBox = cBoundingBox(Vector3d(m_Pos.x, m_Pos.y - m_RequiredPlayerRange, m_Pos.z), m_RequiredPlayerRange, m_RequiredPlayerRange * 2);
{
m_NumPlayers++;
}
}
protected: m_World->ForEachEntityInBox(PlayerBoundingBox, Callback);
Vector3d m_SpawnerPos;
int & m_NumPlayers;
} Callback(SpawnerPos, NumPlayers);
int ChunkX = GetChunkX();
int ChunkZ = GetChunkZ();
m_World->ForEachChunkInRect(ChunkX - 1, ChunkX + 1, ChunkZ - 1, ChunkZ + 1, Callback);
return NumPlayers; return NumPlayers;
} }
@ -234,47 +225,26 @@ int cMobSpawnerEntity::GetNearbyPlayersNum(void)
int cMobSpawnerEntity::GetNearbyMonsterNum(eMonsterType a_EntityType) int cMobSpawnerEntity::GetNearbyMonsterNum(eMonsterType a_EntityType)
{ {
auto SpawnerPos = Vector3d(0.5, 0.5, 0.5) + m_Pos;
int NumEntities = 0; int NumEntities = 0;
class cCallback : public cChunkDataCallback auto Callback = [&] (cEntity & a_Entity)
{ {
public: if (!a_Entity.IsMob())
cCallback(Vector3d a_SpawnerPos, eMonsterType a_CallbackEntityType, int & a_NumEntities) :
m_SpawnerPos(a_SpawnerPos),
m_EntityType(a_CallbackEntityType),
m_NumEntities(a_NumEntities)
{ {
return false;
} }
virtual void Entity(cEntity * a_Entity) override auto & Mob = static_cast<cMonster &>(a_Entity);
if (Mob.GetMobType() == m_Entity)
{ {
if (!a_Entity->IsMob()) NumEntities++;
{
return;
} }
return false;
};
cMonster * Mob = static_cast<cMonster *>(a_Entity); auto EntityBoundingBox = cBoundingBox(Vector3d(m_Pos.x, m_Pos.y - 4, m_Pos.z), m_SpawnRange, 8);
if (Mob->GetMobType() != m_EntityType)
{
return;
}
if ((Diff(m_SpawnerPos.x, a_Entity->GetPosX()) <= 8.0) && (Diff(m_SpawnerPos.y, a_Entity->GetPosY()) <= 4.0) && (Diff(m_SpawnerPos.z, a_Entity->GetPosZ()) <= 8.0)) m_World->ForEachEntityInBox(EntityBoundingBox, Callback);
{
m_NumEntities++;
}
}
protected:
Vector3d m_SpawnerPos;
eMonsterType m_EntityType;
int & m_NumEntities;
} Callback(SpawnerPos, a_EntityType, NumEntities);
int ChunkX = GetChunkX();
int ChunkZ = GetChunkZ();
m_World->ForEachChunkInRect(ChunkX - 1, ChunkX + 1, ChunkZ - 1, ChunkZ + 1, Callback);
return NumEntities; return NumEntities;
} }

View File

@ -33,46 +33,68 @@ public: // tolua_export
virtual bool UsedBy(cPlayer * a_Player) override; virtual bool UsedBy(cPlayer * a_Player) override;
virtual bool Tick(std::chrono::milliseconds a_Dt, cChunk & a_Chunk) override; virtual bool Tick(std::chrono::milliseconds a_Dt, cChunk & a_Chunk) override;
static const char * GetClassStatic(void) { return "cMobSpawnerEntity"; }
// tolua_begin // tolua_begin
/** Upate the active flag from the mob spawner. This function will called every 5 seconds from the Tick() function. */ /** Update the active flag from the mob spawner. This function will called every 5 seconds from the Tick() function. */
void UpdateActiveState(void); void UpdateActiveState(void);
/** Sets the spawn delay to a new random value. */ /** Sets the spawn delay to a new random value. */
void ResetTimer(void); void ResetTimer(void);
/** Spawns the entity. This function automaticly change the spawn delay! */ /** Spawns the entity. This function automatically change the spawn delay! */
void SpawnEntity(void); void SpawnEntity(void);
/** Returns the entity type that will be spawn by this mob spawner. */ // Getters
eMonsterType GetEntity(void) const { return m_Entity; } int GetNearbyMonsterNum(eMonsterType a_EntityType);
/** Sets the entity type who will be spawn by this mob spawner. */
void SetEntity(eMonsterType a_EntityType) { m_Entity = a_EntityType; }
/** Returns the spawn delay. This is the tick delay that is needed to spawn new monsters. */
short GetSpawnDelay(void) const { return m_SpawnDelay; }
/** Sets the spawn delay. */
void SetSpawnDelay(short a_Delay) { m_SpawnDelay = a_Delay; }
/** Returns the amount of the nearby players in a 16-block radius. */
int GetNearbyPlayersNum(void); int GetNearbyPlayersNum(void);
/** Returns the amount of this monster type in a 8-block radius (Y: 4-block radius). */ eMonsterType GetEntity(void) const { return m_Entity; }
int GetNearbyMonsterNum(eMonsterType a_EntityType); short GetSpawnCount(void) const { return m_SpawnCount; }
short GetSpawnRange(void) const { return m_SpawnRange; }
short GetSpawnDelay(void) const { return m_SpawnDelay; }
short GetMinSpawnDelay(void) const { return m_MinSpawnDelay; }
short GetMaxSpawnDelay(void) const { return m_MaxSpawnDelay; }
short GetMaxNearbyEntities(void) const { return m_MaxNearbyEntities; }
short GetRequiredPlayerRange(void) const { return m_RequiredPlayerRange; }
// Setters
void SetEntity(eMonsterType a_EntityType) { m_Entity = a_EntityType; }
void SetSpawnDelay(short a_Delay) { m_SpawnDelay = a_Delay; }
void SetSpawnCount(short a_SpawnCount) { m_SpawnCount = a_SpawnCount; }
void SetSpawnRange(short a_SpawnRange) { m_SpawnRange = a_SpawnRange; }
void SetMinSpawnDelay(short a_Min) { m_MinSpawnDelay = a_Min; }
void SetMaxSpawnDelay(short a_Max) { m_MaxSpawnDelay = a_Max; }
void SetMaxNearbyEntities(short a_MaxNearbyEntities) { m_MaxNearbyEntities = a_MaxNearbyEntities; }
void SetRequiredPlayerRange(short a_RequiredPlayerRange) { m_RequiredPlayerRange = a_RequiredPlayerRange; }
// tolua_end // tolua_end
static const char * GetClassStatic(void) { return "cMobSpawnerEntity"; }
private: private:
/** The entity to spawn. */ /** The entity to spawn. */
eMonsterType m_Entity; eMonsterType m_Entity;
/** Time in ticks until the next entity spawns */
short m_SpawnDelay; short m_SpawnDelay;
bool m_IsActive; bool m_IsActive;
/** Number of entities the spawner tries to spawn each activation. */
short m_SpawnCount = 4;
/** Diameter of the square the new monsters are spawned in */
short m_SpawnRange = 8;
short m_MinSpawnDelay = 200;
short m_MaxSpawnDelay = 800;
/** Maximum amount of the same entity type in proximity. */
short m_MaxNearbyEntities = 6;
/** Maximum distance to player for activation */
short m_RequiredPlayerRange = 16;
} ; // tolua_end } ; // tolua_end

View File

@ -547,7 +547,13 @@ public:
mWriter.BeginCompound(""); mWriter.BeginCompound("");
AddBasicTileEntity(a_MobSpawner, "MobSpawner"); AddBasicTileEntity(a_MobSpawner, "MobSpawner");
mWriter.AddString("EntityId", cMonster::MobTypeToVanillaName(a_MobSpawner->GetEntity())); mWriter.AddString("EntityId", cMonster::MobTypeToVanillaName(a_MobSpawner->GetEntity()));
mWriter.AddShort("SpawnCount", a_MobSpawner->GetSpawnCount());
mWriter.AddShort("SpawnRange", a_MobSpawner->GetSpawnRange());
mWriter.AddShort("Delay", a_MobSpawner->GetSpawnDelay()); mWriter.AddShort("Delay", a_MobSpawner->GetSpawnDelay());
mWriter.AddShort("MinSpawnDelay", a_MobSpawner->GetMinSpawnDelay());
mWriter.AddShort("MaxSpawnDelay", a_MobSpawner->GetMaxSpawnDelay());
mWriter.AddShort("MaxNearbyEntities", a_MobSpawner->GetMaxNearbyEntities());
mWriter.AddShort("RequiredPlayerRange", a_MobSpawner->GetRequiredPlayerRange());
mWriter.EndCompound(); mWriter.EndCompound();
} }

View File

@ -1349,13 +1349,55 @@ OwnedBlockEntity cWSSAnvil::LoadMobSpawnerFromNBT(const cParsedNBT & a_NBT, int
} }
} }
// Load delay: // Load spawn count:
int Delay = a_NBT.FindChildByName(a_TagIdx, "Delay"); int CurrentLine = a_NBT.FindChildByName(a_TagIdx, "SpawnCount");
if ((Delay >= 0) && (a_NBT.GetType(Delay) == TAG_Short)) if ((CurrentLine >= 0) && (a_NBT.GetType(CurrentLine) == TAG_Short))
{ {
MobSpawner->SetSpawnDelay(a_NBT.GetShort(Delay)); MobSpawner->SetSpawnCount(a_NBT.GetShort(CurrentLine));
} }
// Load spawn range:
CurrentLine = a_NBT.FindChildByName(a_TagIdx, "SpawnRange");
if ((CurrentLine >= 0) && (a_NBT.GetType(CurrentLine) == TAG_Short))
{
MobSpawner->SetSpawnRange(a_NBT.GetShort(CurrentLine));
}
// Load delay:
CurrentLine = a_NBT.FindChildByName(a_TagIdx, "Delay");
if ((CurrentLine >= 0) && (a_NBT.GetType(CurrentLine) == TAG_Short))
{
MobSpawner->SetSpawnDelay(a_NBT.GetShort(CurrentLine));
}
// Load delay range:
CurrentLine = a_NBT.FindChildByName(a_TagIdx, "MinSpawnDelay");
if ((CurrentLine >= 0) && (a_NBT.GetType(CurrentLine) == TAG_Short))
{
MobSpawner->SetMinSpawnDelay(a_NBT.GetShort(CurrentLine));
}
CurrentLine = a_NBT.FindChildByName(a_TagIdx, "MaxSpawnDelay");
if ((CurrentLine >= 0) && (a_NBT.GetType(CurrentLine) == TAG_Short))
{
MobSpawner->SetMaxSpawnDelay(a_NBT.GetShort(CurrentLine));
}
// Load MaxNearbyEntities:
CurrentLine = a_NBT.FindChildByName(a_TagIdx, "MaxNearbyEntities");
if ((CurrentLine >= 0) && (a_NBT.GetType(CurrentLine) == TAG_Short))
{
MobSpawner->SetMaxNearbyEntities(a_NBT.GetShort(CurrentLine));
}
// Load RequiredPlayerRange:
CurrentLine = a_NBT.FindChildByName(a_TagIdx, "RequiredPlayerRange");
if ((CurrentLine >= 0) && (a_NBT.GetType(CurrentLine) == TAG_Short))
{
MobSpawner->SetRequiredPlayerRange(a_NBT.GetShort(CurrentLine));
}
return MobSpawner; return MobSpawner;
} }