#include "Globals.h" // NOTE: MSVC stupidness requires this to be the same across all modules #include "Chunk.h" #include "Enderman.h" #include "../Entities/Player.h" #include "../LineBlockTracer.h" //////////////////////////////////////////////////////////////////////////////// // cPlayerLookCheck class cPlayerLookCheck { public: cPlayerLookCheck(Vector3d a_EndermanHeadPosition, int a_SightDistance) : m_Player(nullptr), m_EndermanHeadPosition(a_EndermanHeadPosition), m_SightDistance(a_SightDistance) { } bool operator()(cPlayer & a_Player) { // Don't check players who cannot be targeted if (!a_Player.CanMobsTarget()) { return false; } const auto PlayerHeadPosition = a_Player.GetPosition().addedY(a_Player.GetHeight()); const auto Direction = m_EndermanHeadPosition - PlayerHeadPosition; // Don't check players who are more than SightDistance (64) blocks away: if (Direction.Length() > m_SightDistance) { return false; } // Don't check if the player has a pumpkin on his head: if (a_Player.GetEquippedHelmet().m_ItemType == E_BLOCK_PUMPKIN) { return false; } const auto LookVector = a_Player.GetLookVector(); // Note: ||LookVector|| is always 1. const auto Cosine = Direction.Dot(LookVector) / Direction.Length(); // a.b / (||a|| * ||b||) // If the player's crosshair is within 5 degrees of the enderman, it counts as looking: if ((Cosine < std::cos(0.09)) || (Cosine > std::cos(0))) // 0.09 rad ~ 5 degrees { return false; } // TODO: Check if endermen are angered through water in Vanilla if (!cLineBlockTracer::LineOfSightTrace( *a_Player.GetWorld(), m_EndermanHeadPosition, PlayerHeadPosition, cLineBlockTracer::losAirWater )) { // No direct line of sight return false; } m_Player = &a_Player; return true; } cPlayer * GetPlayer(void) const { return m_Player; } protected: cPlayer * m_Player; Vector3d m_EndermanHeadPosition; int m_SightDistance; }; cEnderman::cEnderman(void) : Super( "Enderman", mtEnderman, "entity.endermen.hurt", "entity.endermen.death", "entity.endermen.ambient", 0.6f, 2.9f ), m_bIsScreaming(false), m_CarriedBlock(E_BLOCK_AIR), m_CarriedMeta(0) { } void cEnderman::GetDrops(cItems & a_Drops, cEntity * a_Killer) { unsigned int LootingLevel = 0; if (a_Killer != nullptr) { LootingLevel = a_Killer->GetEquippedWeapon().m_Enchantments.GetLevel(cEnchantments::enchLooting); } AddRandomDropItem(a_Drops, 0, 1 + LootingLevel, E_ITEM_ENDER_PEARL); } void cEnderman::CheckEventSeePlayer(cChunk & a_Chunk) { if (GetTarget() != nullptr) { return; } cPlayerLookCheck Callback(GetPosition().addedY(GetHeight()), m_SightDistance); if (m_World->ForEachPlayer(Callback)) { return; } ASSERT(Callback.GetPlayer() != nullptr); // Target the player: cAggressiveMonster::EventSeePlayer(Callback.GetPlayer(), a_Chunk); m_bIsScreaming = true; GetWorld()->BroadcastEntityMetadata(*this); } void cEnderman::EventLosePlayer() { Super::EventLosePlayer(); m_bIsScreaming = false; GetWorld()->BroadcastEntityMetadata(*this); } void cEnderman::Tick(std::chrono::milliseconds a_Dt, cChunk & a_Chunk) { Super::Tick(a_Dt, a_Chunk); if (!IsTicking()) { // The base class tick destroyed us return; } if (m_EMState != CHASING) { cMonster * EndermiteFound = GetMonsterOfTypeInSight(mtEndermite, 64); if (EndermiteFound != nullptr) { SetTarget(EndermiteFound); m_EMState = CHASING; m_bIsScreaming = true; } } else { const auto Target = GetTarget(); if (Target != nullptr) { if (!Target->IsTicking()) { m_EMState = IDLE; m_bIsScreaming = false; } } } PREPARE_REL_AND_CHUNK(GetPosition().Floor(), a_Chunk); if (!RelSuccess) { return; } // Take damage when wet: if (IsInWater() || Chunk->IsWeatherWetAt(Rel)) { EventLosePlayer(); TakeDamage(dtEnvironment, nullptr, 1, 0); // TODO teleport to a safe location } }