2013-10-28 16:41:43 -04:00
|
|
|
|
2013-08-22 02:55:58 -04:00
|
|
|
// ProjectileEntity.cpp
|
|
|
|
|
|
|
|
// Implements the cProjectileEntity class representing the common base class for projectiles, as well as individual projectile types
|
|
|
|
|
|
|
|
#include "Globals.h"
|
2014-04-26 19:45:39 -04:00
|
|
|
|
2014-03-29 10:26:41 -04:00
|
|
|
#include "../Bindings/PluginManager.h"
|
2013-08-22 02:55:58 -04:00
|
|
|
#include "ProjectileEntity.h"
|
2013-08-22 03:07:12 -04:00
|
|
|
#include "../ClientHandle.h"
|
2013-08-27 13:57:37 -04:00
|
|
|
#include "../LineBlockTracer.h"
|
2013-09-01 16:40:35 -04:00
|
|
|
#include "../BoundingBox.h"
|
|
|
|
#include "../ChunkMap.h"
|
|
|
|
#include "../Chunk.h"
|
2013-08-27 13:57:37 -04:00
|
|
|
|
2014-04-27 20:03:06 -04:00
|
|
|
#include "ArrowEntity.h"
|
|
|
|
#include "ThrownEggEntity.h"
|
|
|
|
#include "ThrownEnderPearlEntity.h"
|
|
|
|
#include "ExpBottleEntity.h"
|
|
|
|
#include "ThrownSnowballEntity.h"
|
|
|
|
#include "FireChargeEntity.h"
|
|
|
|
#include "FireworkEntity.h"
|
|
|
|
#include "GhastFireballEntity.h"
|
2014-06-06 03:16:33 -04:00
|
|
|
#include "WitherSkullEntity.h"
|
2014-07-20 05:56:59 -04:00
|
|
|
#include "SplashPotionEntity.h"
|
2014-07-04 11:49:24 -04:00
|
|
|
#include "Player.h"
|
2014-04-26 19:45:39 -04:00
|
|
|
|
2013-08-27 13:57:37 -04:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2015-07-31 10:49:10 -04:00
|
|
|
/** Converts an angle in radians into a byte representation used by the network protocol */
|
2015-05-24 07:56:56 -04:00
|
|
|
#define ANGLE_TO_PROTO(X) static_cast<Byte>(X * 255 / 360)
|
2013-09-07 11:14:37 -04:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2014-07-17 16:15:34 -04:00
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
2013-08-27 13:57:37 -04:00
|
|
|
// cProjectileTracerCallback:
|
|
|
|
|
|
|
|
class cProjectileTracerCallback :
|
|
|
|
public cBlockTracer::cCallbacks
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
cProjectileTracerCallback(cProjectileEntity * a_Projectile) :
|
2013-09-02 15:56:55 -04:00
|
|
|
m_Projectile(a_Projectile),
|
|
|
|
m_SlowdownCoeff(0.99) // Default slowdown when not in water
|
2013-08-27 13:57:37 -04:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2013-09-02 15:56:55 -04:00
|
|
|
double GetSlowdownCoeff(void) const { return m_SlowdownCoeff; }
|
|
|
|
|
2013-08-27 13:57:37 -04:00
|
|
|
protected:
|
|
|
|
cProjectileEntity * m_Projectile;
|
2013-09-02 15:56:55 -04:00
|
|
|
double m_SlowdownCoeff;
|
2013-08-27 13:57:37 -04:00
|
|
|
|
2013-09-02 15:56:55 -04:00
|
|
|
// cCallbacks overrides:
|
2013-08-27 13:57:37 -04:00
|
|
|
virtual bool OnNextBlock(int a_BlockX, int a_BlockY, int a_BlockZ, BLOCKTYPE a_BlockType, NIBBLETYPE a_BlockMeta, char a_EntryFace) override
|
|
|
|
{
|
2013-09-07 15:56:22 -04:00
|
|
|
/*
|
|
|
|
// DEBUG:
|
|
|
|
LOGD("Hit block %d:%d at {%d, %d, %d} face %d, %s (%s)",
|
|
|
|
a_BlockType, a_BlockMeta,
|
|
|
|
a_BlockX, a_BlockY, a_BlockZ, a_EntryFace,
|
2014-03-01 14:34:19 -05:00
|
|
|
cBlockInfo::IsSolid(a_BlockType) ? "solid" : "non-solid",
|
2013-09-07 15:56:22 -04:00
|
|
|
ItemToString(cItem(a_BlockType, 1, a_BlockMeta)).c_str()
|
|
|
|
);
|
|
|
|
*/
|
|
|
|
|
2014-03-01 14:34:19 -05:00
|
|
|
if (cBlockInfo::IsSolid(a_BlockType))
|
2013-08-27 13:57:37 -04:00
|
|
|
{
|
2014-06-22 15:44:01 -04:00
|
|
|
// The projectile hit a solid block, calculate the exact hit coords:
|
2014-07-07 16:14:15 -04:00
|
|
|
cBoundingBox bb(a_BlockX, a_BlockX + 1, a_BlockY, a_BlockY + 1, a_BlockZ, a_BlockZ + 1); // Bounding box of the block hit
|
|
|
|
const Vector3d LineStart = m_Projectile->GetPosition(); // Start point for the imaginary line that goes through the block hit
|
|
|
|
const Vector3d LineEnd = LineStart + m_Projectile->GetSpeed(); // End point for the imaginary line that goes through the block hit
|
|
|
|
double LineCoeff = 0; // Used to calculate where along the line an intersection with the bounding box occurs
|
|
|
|
eBlockFace Face; // Face hit
|
2014-06-22 15:44:01 -04:00
|
|
|
|
|
|
|
if (bb.CalcLineIntersection(LineStart, LineEnd, LineCoeff, Face))
|
2013-09-03 03:41:31 -04:00
|
|
|
{
|
2014-07-07 16:14:15 -04:00
|
|
|
Vector3d Intersection = LineStart + m_Projectile->GetSpeed() * LineCoeff; // Point where projectile goes into the hit block
|
2014-06-22 15:44:01 -04:00
|
|
|
|
2014-05-31 08:14:55 -04:00
|
|
|
if (cPluginManager::Get()->CallHookProjectileHitBlock(*m_Projectile, a_BlockX, a_BlockY, a_BlockZ, Face, &Intersection))
|
2014-03-29 11:00:45 -04:00
|
|
|
{
|
2014-03-29 12:05:24 -04:00
|
|
|
return false;
|
2014-03-29 11:00:45 -04:00
|
|
|
}
|
|
|
|
|
2013-09-03 03:41:31 -04:00
|
|
|
m_Projectile->OnHitSolidBlock(Intersection, Face);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
LOGD("WEIRD! block tracer reports a hit, but BBox tracer doesn't. Ignoring the hit.");
|
|
|
|
}
|
2013-08-27 13:57:37 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// Convey some special effects from special blocks:
|
|
|
|
switch (a_BlockType)
|
|
|
|
{
|
|
|
|
case E_BLOCK_LAVA:
|
|
|
|
case E_BLOCK_STATIONARY_LAVA:
|
|
|
|
{
|
|
|
|
m_Projectile->StartBurning(30);
|
2013-09-02 15:56:55 -04:00
|
|
|
m_SlowdownCoeff = std::min(m_SlowdownCoeff, 0.9); // Slow down to 0.9* the speed each tick when moving through lava
|
2013-08-27 13:57:37 -04:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case E_BLOCK_WATER:
|
|
|
|
case E_BLOCK_STATIONARY_WATER:
|
|
|
|
{
|
|
|
|
m_Projectile->StopBurning();
|
2013-09-02 15:56:55 -04:00
|
|
|
m_SlowdownCoeff = std::min(m_SlowdownCoeff, 0.8); // Slow down to 0.8* the speed each tick when moving through water
|
2013-08-27 13:57:37 -04:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
} // switch (a_BlockType)
|
|
|
|
|
|
|
|
// Continue tracing
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
} ;
|
2013-08-22 02:55:58 -04:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2014-07-17 16:15:34 -04:00
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
2013-09-01 16:40:35 -04:00
|
|
|
// cProjectileEntityCollisionCallback:
|
|
|
|
|
|
|
|
class cProjectileEntityCollisionCallback :
|
|
|
|
public cEntityCallback
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
cProjectileEntityCollisionCallback(cProjectileEntity * a_Projectile, const Vector3d & a_Pos, const Vector3d & a_NextPos) :
|
|
|
|
m_Projectile(a_Projectile),
|
|
|
|
m_Pos(a_Pos),
|
|
|
|
m_NextPos(a_NextPos),
|
|
|
|
m_MinCoeff(1),
|
2014-10-20 16:55:07 -04:00
|
|
|
m_HitEntity(nullptr)
|
2013-09-01 16:40:35 -04:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
virtual bool Item(cEntity * a_Entity) override
|
|
|
|
{
|
2013-09-02 15:40:56 -04:00
|
|
|
if (
|
|
|
|
(a_Entity == m_Projectile) || // Do not check collisions with self
|
2014-07-05 17:59:22 -04:00
|
|
|
(a_Entity->GetUniqueID() == m_Projectile->GetCreatorUniqueID()) // Do not check whoever shot the projectile
|
2013-09-02 15:40:56 -04:00
|
|
|
)
|
2013-09-01 16:40:35 -04:00
|
|
|
{
|
2014-07-20 04:43:07 -04:00
|
|
|
// Don't check creator only for the first 5 ticks so that projectiles can collide with the creator
|
|
|
|
if (m_Projectile->GetTicksAlive() <= 5)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2013-09-01 16:40:35 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
cBoundingBox EntBox(a_Entity->GetPosition(), a_Entity->GetWidth() / 2, a_Entity->GetHeight());
|
|
|
|
|
|
|
|
// Instead of colliding the bounding box with another bounding box in motion, we collide an enlarged bounding box with a hairline.
|
|
|
|
// The results should be good enough for our purposes
|
|
|
|
double LineCoeff;
|
2014-02-04 13:59:05 -05:00
|
|
|
eBlockFace Face;
|
2013-09-01 16:40:35 -04:00
|
|
|
EntBox.Expand(m_Projectile->GetWidth() / 2, m_Projectile->GetHeight() / 2, m_Projectile->GetWidth() / 2);
|
|
|
|
if (!EntBox.CalcLineIntersection(m_Pos, m_NextPos, LineCoeff, Face))
|
|
|
|
{
|
|
|
|
// No intersection whatsoever
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2014-06-22 15:44:01 -04:00
|
|
|
if (!a_Entity->IsMob() && !a_Entity->IsMinecart() && !a_Entity->IsPlayer() && !a_Entity->IsBoat())
|
|
|
|
{
|
|
|
|
// Not an entity that interacts with a projectile
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2014-03-29 10:26:41 -04:00
|
|
|
if (cPluginManager::Get()->CallHookProjectileHitEntity(*m_Projectile, *a_Entity))
|
|
|
|
{
|
|
|
|
// A plugin disagreed.
|
|
|
|
return false;
|
|
|
|
}
|
2013-09-01 16:40:35 -04:00
|
|
|
|
|
|
|
if (LineCoeff < m_MinCoeff)
|
|
|
|
{
|
|
|
|
// The entity is closer than anything we've stored so far, replace it as the potential victim
|
|
|
|
m_MinCoeff = LineCoeff;
|
|
|
|
m_HitEntity = a_Entity;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Don't break the enumeration, we want all the entities
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-07-31 10:49:10 -04:00
|
|
|
/** Returns the nearest entity that was hit, after the enumeration has been completed */
|
2013-09-01 16:40:35 -04:00
|
|
|
cEntity * GetHitEntity(void) const { return m_HitEntity; }
|
|
|
|
|
2015-07-31 10:49:10 -04:00
|
|
|
/** Returns the line coeff where the hit was encountered, after the enumeration has been completed */
|
2013-09-01 16:40:35 -04:00
|
|
|
double GetMinCoeff(void) const { return m_MinCoeff; }
|
|
|
|
|
2015-07-31 10:49:10 -04:00
|
|
|
/** Returns true if the callback has encountered a true hit */
|
2013-09-01 16:40:35 -04:00
|
|
|
bool HasHit(void) const { return (m_MinCoeff < 1); }
|
|
|
|
|
|
|
|
protected:
|
|
|
|
cProjectileEntity * m_Projectile;
|
|
|
|
const Vector3d & m_Pos;
|
|
|
|
const Vector3d & m_NextPos;
|
|
|
|
double m_MinCoeff; // The coefficient of the nearest hit on the Pos line
|
|
|
|
|
|
|
|
// Although it's bad(tm) to store entity ptrs from a callback, we can afford it here, because the entire callback
|
|
|
|
// is processed inside the tick thread, so the entities won't be removed in between the calls and the final processing
|
|
|
|
cEntity * m_HitEntity; // The nearest hit entity
|
|
|
|
} ;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2014-07-17 16:15:34 -04:00
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
2013-08-22 02:55:58 -04:00
|
|
|
// cProjectileEntity:
|
|
|
|
|
|
|
|
cProjectileEntity::cProjectileEntity(eKind a_Kind, cEntity * a_Creator, double a_X, double a_Y, double a_Z, double a_Width, double a_Height) :
|
|
|
|
super(etProjectile, a_X, a_Y, a_Z, a_Width, a_Height),
|
2013-08-22 16:31:15 -04:00
|
|
|
m_ProjectileKind(a_Kind),
|
2014-07-04 17:07:26 -04:00
|
|
|
m_CreatorData(
|
2015-03-21 10:18:17 -04:00
|
|
|
((a_Creator != nullptr) ? a_Creator->GetUniqueID() : cEntity::INVALID_ID),
|
2015-05-24 07:56:56 -04:00
|
|
|
((a_Creator != nullptr) ? (a_Creator->IsPlayer() ? static_cast<cPlayer *>(a_Creator)->GetName() : "") : ""),
|
2014-10-20 16:55:07 -04:00
|
|
|
((a_Creator != nullptr) ? a_Creator->GetEquippedWeapon().m_Enchantments : cEnchantments())
|
2014-07-04 17:07:26 -04:00
|
|
|
),
|
2013-08-27 13:57:37 -04:00
|
|
|
m_IsInGround(false)
|
2013-08-22 02:55:58 -04:00
|
|
|
{
|
2015-03-30 19:42:32 -04:00
|
|
|
SetGravity(-12.0f);
|
2015-03-31 11:03:35 -04:00
|
|
|
SetAirDrag(0.01f);
|
2013-08-22 02:55:58 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
cProjectileEntity::cProjectileEntity(eKind a_Kind, cEntity * a_Creator, const Vector3d & a_Pos, const Vector3d & a_Speed, double a_Width, double a_Height) :
|
|
|
|
super(etProjectile, a_Pos.x, a_Pos.y, a_Pos.z, a_Width, a_Height),
|
2013-08-22 16:31:15 -04:00
|
|
|
m_ProjectileKind(a_Kind),
|
2015-05-24 07:56:56 -04:00
|
|
|
m_CreatorData(a_Creator->GetUniqueID(), a_Creator->IsPlayer() ? static_cast<cPlayer *>(a_Creator)->GetName() : "", a_Creator->GetEquippedWeapon().m_Enchantments),
|
2013-08-27 13:57:37 -04:00
|
|
|
m_IsInGround(false)
|
2013-08-22 02:55:58 -04:00
|
|
|
{
|
|
|
|
SetSpeed(a_Speed);
|
2014-01-16 14:00:49 -05:00
|
|
|
SetYawFromSpeed();
|
2013-09-07 11:14:37 -04:00
|
|
|
SetPitchFromSpeed();
|
2015-03-30 19:42:32 -04:00
|
|
|
SetGravity(-12.0f);
|
2015-03-31 11:03:35 -04:00
|
|
|
SetAirDrag(0.01f);
|
2013-08-22 02:55:58 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2014-07-13 09:09:08 -04:00
|
|
|
cProjectileEntity * cProjectileEntity::Create(eKind a_Kind, cEntity * a_Creator, double a_X, double a_Y, double a_Z, const cItem * a_Item, const Vector3d * a_Speed)
|
2013-08-22 02:55:58 -04:00
|
|
|
{
|
|
|
|
Vector3d Speed;
|
2014-10-20 16:55:07 -04:00
|
|
|
if (a_Speed != nullptr)
|
2013-08-22 02:55:58 -04:00
|
|
|
{
|
|
|
|
Speed = *a_Speed;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (a_Kind)
|
|
|
|
{
|
2013-09-07 12:53:14 -04:00
|
|
|
case pkArrow: return new cArrowEntity (a_Creator, a_X, a_Y, a_Z, Speed);
|
|
|
|
case pkEgg: return new cThrownEggEntity (a_Creator, a_X, a_Y, a_Z, Speed);
|
|
|
|
case pkEnderPearl: return new cThrownEnderPearlEntity(a_Creator, a_X, a_Y, a_Z, Speed);
|
|
|
|
case pkSnowball: return new cThrownSnowballEntity (a_Creator, a_X, a_Y, a_Z, Speed);
|
|
|
|
case pkGhastFireball: return new cGhastFireballEntity (a_Creator, a_X, a_Y, a_Z, Speed);
|
|
|
|
case pkFireCharge: return new cFireChargeEntity (a_Creator, a_X, a_Y, a_Z, Speed);
|
2013-11-16 15:58:17 -05:00
|
|
|
case pkExpBottle: return new cExpBottleEntity (a_Creator, a_X, a_Y, a_Z, Speed);
|
2014-07-20 05:56:59 -04:00
|
|
|
case pkSplashPotion: return new cSplashPotionEntity (a_Creator, a_X, a_Y, a_Z, Speed, *a_Item);
|
2014-06-06 03:16:33 -04:00
|
|
|
case pkWitherSkull: return new cWitherSkullEntity (a_Creator, a_X, a_Y, a_Z, Speed);
|
2014-02-26 18:29:14 -05:00
|
|
|
case pkFirework:
|
|
|
|
{
|
2014-10-20 16:55:07 -04:00
|
|
|
ASSERT(a_Item != nullptr);
|
2014-07-13 09:09:08 -04:00
|
|
|
if (a_Item->m_FireworkItem.m_Colours.empty())
|
2014-02-26 18:29:14 -05:00
|
|
|
{
|
2014-10-20 16:55:07 -04:00
|
|
|
return nullptr;
|
2014-02-26 18:29:14 -05:00
|
|
|
}
|
|
|
|
|
2014-07-13 09:09:08 -04:00
|
|
|
return new cFireworkEntity(a_Creator, a_X, a_Y, a_Z, *a_Item);
|
2014-02-26 18:29:14 -05:00
|
|
|
}
|
2015-05-19 14:32:10 -04:00
|
|
|
case pkFishingFloat: break;
|
2013-08-22 02:55:58 -04:00
|
|
|
}
|
|
|
|
|
2013-08-30 12:10:58 -04:00
|
|
|
LOGWARNING("%s: Unknown projectile kind: %d", __FUNCTION__, a_Kind);
|
2014-10-20 16:55:07 -04:00
|
|
|
return nullptr;
|
2013-08-22 02:55:58 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2014-02-04 13:59:05 -05:00
|
|
|
void cProjectileEntity::OnHitSolidBlock(const Vector3d & a_HitPos, eBlockFace a_HitFace)
|
2013-08-27 13:57:37 -04:00
|
|
|
{
|
2013-09-03 03:41:31 -04:00
|
|
|
// Set the position based on what face was hit:
|
|
|
|
SetPosition(a_HitPos);
|
2013-08-27 13:57:37 -04:00
|
|
|
SetSpeed(0, 0, 0);
|
|
|
|
|
|
|
|
// DEBUG:
|
|
|
|
LOGD("Projectile %d: pos {%.02f, %.02f, %.02f}, hit solid block at face %d",
|
|
|
|
m_UniqueID,
|
2013-09-03 03:41:31 -04:00
|
|
|
a_HitPos.x, a_HitPos.y, a_HitPos.z,
|
|
|
|
a_HitFace
|
2013-08-27 13:57:37 -04:00
|
|
|
);
|
|
|
|
|
|
|
|
m_IsInGround = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2013-08-22 16:31:15 -04:00
|
|
|
AString cProjectileEntity::GetMCAClassName(void) const
|
|
|
|
{
|
|
|
|
switch (m_ProjectileKind)
|
|
|
|
{
|
|
|
|
case pkArrow: return "Arrow";
|
|
|
|
case pkSnowball: return "Snowball";
|
|
|
|
case pkEgg: return "Egg";
|
|
|
|
case pkGhastFireball: return "Fireball";
|
|
|
|
case pkFireCharge: return "SmallFireball";
|
2014-07-04 11:49:24 -04:00
|
|
|
case pkEnderPearl: return "ThrownEnderpearl";
|
2013-08-22 16:31:15 -04:00
|
|
|
case pkExpBottle: return "ThrownExpBottle";
|
2014-07-11 21:58:11 -04:00
|
|
|
case pkSplashPotion: return "SplashPotion";
|
2013-08-22 16:31:15 -04:00
|
|
|
case pkWitherSkull: return "WitherSkull";
|
2014-02-26 18:29:14 -05:00
|
|
|
case pkFirework: return "Firework";
|
2013-08-22 16:31:15 -04:00
|
|
|
case pkFishingFloat: return ""; // Unknown, perhaps MC doesn't save this?
|
|
|
|
}
|
|
|
|
ASSERT(!"Unhandled projectile entity kind!");
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2015-01-11 16:12:26 -05:00
|
|
|
void cProjectileEntity::Tick(std::chrono::milliseconds a_Dt, cChunk & a_Chunk)
|
2013-08-27 13:57:37 -04:00
|
|
|
{
|
|
|
|
super::Tick(a_Dt, a_Chunk);
|
2015-03-20 18:32:32 -04:00
|
|
|
BroadcastMovementUpdate();
|
2013-08-27 13:57:37 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2015-01-11 16:12:26 -05:00
|
|
|
void cProjectileEntity::HandlePhysics(std::chrono::milliseconds a_Dt, cChunk & a_Chunk)
|
2013-08-27 13:57:37 -04:00
|
|
|
{
|
|
|
|
if (m_IsInGround)
|
|
|
|
{
|
|
|
|
// Already-grounded projectiles don't move at all
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-03-30 20:07:19 -04:00
|
|
|
auto DtSec = std::chrono::duration_cast<std::chrono::duration<double>>(a_Dt);
|
|
|
|
|
|
|
|
const Vector3d DeltaSpeed = GetSpeed() * DtSec.count();
|
2014-06-22 15:44:01 -04:00
|
|
|
const Vector3d Pos = GetPosition();
|
2015-03-30 20:07:19 -04:00
|
|
|
const Vector3d NextPos = Pos + DeltaSpeed;
|
2014-06-22 15:44:01 -04:00
|
|
|
|
2013-09-01 16:40:35 -04:00
|
|
|
// Test for entity collisions:
|
|
|
|
cProjectileEntityCollisionCallback EntityCollisionCallback(this, Pos, NextPos);
|
|
|
|
a_Chunk.ForEachEntity(EntityCollisionCallback);
|
|
|
|
if (EntityCollisionCallback.HasHit())
|
2013-08-27 13:57:37 -04:00
|
|
|
{
|
2013-09-01 16:40:35 -04:00
|
|
|
// An entity was hit:
|
|
|
|
Vector3d HitPos = Pos + (NextPos - Pos) * EntityCollisionCallback.GetMinCoeff();
|
2013-09-07 12:53:14 -04:00
|
|
|
|
|
|
|
// DEBUG:
|
2013-09-01 16:40:35 -04:00
|
|
|
LOGD("Projectile %d has hit an entity %d (%s) at {%.02f, %.02f, %.02f} (coeff %.03f)",
|
|
|
|
m_UniqueID,
|
|
|
|
EntityCollisionCallback.GetHitEntity()->GetUniqueID(),
|
|
|
|
EntityCollisionCallback.GetHitEntity()->GetClass(),
|
|
|
|
HitPos.x, HitPos.y, HitPos.z,
|
|
|
|
EntityCollisionCallback.GetMinCoeff()
|
|
|
|
);
|
2014-06-22 15:44:01 -04:00
|
|
|
|
2013-09-07 12:53:14 -04:00
|
|
|
OnHitEntity(*(EntityCollisionCallback.GetHitEntity()), HitPos);
|
2013-08-27 13:57:37 -04:00
|
|
|
}
|
2013-09-01 16:40:35 -04:00
|
|
|
// TODO: Test the entities in the neighboring chunks, too
|
2014-06-22 15:44:01 -04:00
|
|
|
|
|
|
|
// Trace the tick's worth of movement as a line:
|
|
|
|
cProjectileTracerCallback TracerCallback(this);
|
|
|
|
if (!cLineBlockTracer::Trace(*m_World, TracerCallback, Pos, NextPos))
|
|
|
|
{
|
|
|
|
// Something has been hit, abort all other processing
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// The tracer also checks the blocks for slowdown blocks - water and lava - and stores it for later in its SlowdownCoeff
|
2013-09-01 16:40:35 -04:00
|
|
|
|
|
|
|
// Update the position:
|
|
|
|
SetPosition(NextPos);
|
2013-08-27 13:57:37 -04:00
|
|
|
|
2013-09-02 15:56:55 -04:00
|
|
|
// Add slowdown and gravity effect to the speed:
|
|
|
|
Vector3d NewSpeed(GetSpeed());
|
2015-03-30 20:07:19 -04:00
|
|
|
NewSpeed.y += m_Gravity * DtSec.count();
|
2015-03-31 11:03:35 -04:00
|
|
|
NewSpeed -= NewSpeed * (m_AirDrag * 20.0f) * DtSec.count();
|
2013-09-02 15:56:55 -04:00
|
|
|
SetSpeed(NewSpeed);
|
2014-01-16 14:00:49 -05:00
|
|
|
SetYawFromSpeed();
|
2013-09-07 11:14:37 -04:00
|
|
|
SetPitchFromSpeed();
|
2013-08-27 13:57:37 -04:00
|
|
|
|
2014-04-05 16:25:40 -04:00
|
|
|
/*
|
2013-09-07 11:14:37 -04:00
|
|
|
LOGD("Projectile %d: pos {%.02f, %.02f, %.02f}, speed {%.02f, %.02f, %.02f}, rot {%.02f, %.02f}",
|
2013-08-27 13:57:37 -04:00
|
|
|
m_UniqueID,
|
|
|
|
GetPosX(), GetPosY(), GetPosZ(),
|
2013-09-07 11:14:37 -04:00
|
|
|
GetSpeedX(), GetSpeedY(), GetSpeedZ(),
|
2014-01-17 05:11:17 -05:00
|
|
|
GetYaw(), GetPitch()
|
2013-08-27 13:57:37 -04:00
|
|
|
);
|
2014-04-05 16:25:40 -04:00
|
|
|
*/
|
2013-08-27 13:57:37 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2013-08-30 12:10:58 -04:00
|
|
|
void cProjectileEntity::SpawnOn(cClientHandle & a_Client)
|
|
|
|
{
|
|
|
|
// Default spawning - use the projectile kind to spawn an object:
|
2014-01-17 05:11:17 -05:00
|
|
|
a_Client.SendSpawnObject(*this, m_ProjectileKind, 12, ANGLE_TO_PROTO(GetYaw()), ANGLE_TO_PROTO(GetPitch()));
|
2013-09-07 11:14:37 -04:00
|
|
|
a_Client.SendEntityMetadata(*this);
|
2013-08-30 12:10:58 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2014-10-15 13:01:55 -04:00
|
|
|
void cProjectileEntity::CollectedBy(cPlayer & a_Dest)
|
2013-11-12 16:43:20 -05:00
|
|
|
{
|
|
|
|
// Overriden in arrow
|
|
|
|
UNUSED(a_Dest);
|
2014-04-27 12:42:31 -04:00
|
|
|
}
|
2014-10-15 13:01:55 -04:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|