mirror of
https://github.com/ZDoom/raze-gles.git
synced 2025-01-13 11:30:44 +00:00
6db18e61d9
This also does not need to be serialized because it gets cleared at the end of each game tick and will always be empty when saving is done.
294 lines
6.2 KiB
C++
294 lines
6.2 KiB
C++
#pragma once
|
|
|
|
BEGIN_BLD_NS
|
|
|
|
extern int cumulDamage[kMaxXSprites];
|
|
|
|
|
|
|
|
// Due to the messed up array storage of all the game data we cannot do any direct references here yet. We have to access everything via wrapper functions for now.
|
|
// Note that the indexing is very inconsistent - partially by sprite index, partially by xsprite index.
|
|
class DBloodActor
|
|
{
|
|
int index;
|
|
DBloodActor* base();
|
|
|
|
public:
|
|
int dudeSlope;
|
|
DUDEEXTRA dudeExtra;
|
|
|
|
DBloodActor() :index(int(this - base())) { /*assert(index >= 0 && index < kMaxSprites);*/ }
|
|
DBloodActor& operator=(const DBloodActor& other) = default;
|
|
|
|
void Clear()
|
|
{
|
|
dudeSlope = 0;
|
|
dudeExtra = {};
|
|
}
|
|
bool hasX() { return sprite[index].extra > 0; }
|
|
void addX()
|
|
{
|
|
if (s().extra == -1) dbInsertXSprite(s().index);
|
|
}
|
|
spritetype& s() { return sprite[index]; }
|
|
XSPRITE& x() { return xsprite[sprite[index].extra]; } // calling this does not validate the xsprite!
|
|
SPRITEHIT& hit() { return gSpriteHit[sprite[index].extra]; }
|
|
int& xvel() { return Blood::xvel[index]; }
|
|
int& yvel() { return Blood::yvel[index]; }
|
|
int& zvel() { return Blood::zvel[index]; }
|
|
|
|
int& cumulDamage() { return Blood::cumulDamage[sprite[index].extra]; }
|
|
SPRITEMASS& spriteMass() { return gSpriteMass[sprite[index].extra]; }
|
|
GENDUDEEXTRA& genDudeExtra() { return Blood::gGenDudeExtra[index]; }
|
|
POINT3D& basePoint() { return Blood::baseSprite[index]; }
|
|
|
|
void SetOwner(DBloodActor* own)
|
|
{
|
|
s().owner = own? own->s().index : -1;
|
|
}
|
|
|
|
DBloodActor* GetOwner()
|
|
{
|
|
if (s().owner == -1 || s().owner == kMaxSprites-1) return nullptr;
|
|
return base() + s().owner;
|
|
}
|
|
|
|
void SetTarget(DBloodActor* own)
|
|
{
|
|
x().target = own ? own->s().index : -1;
|
|
}
|
|
|
|
DBloodActor* GetTarget()
|
|
{
|
|
if (x().target == -1 || x().target == kMaxSprites - 1) return nullptr;
|
|
return base() + x().target;
|
|
}
|
|
|
|
void SetBurnSource(DBloodActor* own)
|
|
{
|
|
x().burnSource = own ? own->s().index : -1;
|
|
}
|
|
|
|
DBloodActor* GetBurnSource()
|
|
{
|
|
if (x().burnSource == -1 || x().burnSource == kMaxSprites - 1) return nullptr;
|
|
return base() + x().burnSource;
|
|
}
|
|
|
|
void SetSpecialOwner() // nnext hackery
|
|
{
|
|
s().owner = kMaxSprites - 1;
|
|
}
|
|
|
|
bool GetSpecialOwner()
|
|
{
|
|
return (s().owner == kMaxSprites - 1);
|
|
}
|
|
|
|
bool IsPlayerActor()
|
|
{
|
|
return s().type >= kDudePlayer1 && s().type <= kDudePlayer8;
|
|
}
|
|
|
|
bool IsDudeActor()
|
|
{
|
|
return s().type >= kDudeBase && s().type < kDudeMax;
|
|
}
|
|
|
|
bool IsItemActor()
|
|
{
|
|
return s().type >= kItemBase && s().type < kItemMax;
|
|
}
|
|
|
|
bool IsWeaponActor()
|
|
{
|
|
return s().type >= kItemWeaponBase && s().type < kItemWeaponMax;
|
|
}
|
|
|
|
bool IsAmmoActor()
|
|
{
|
|
return s().type >= kItemAmmoBase && s().type < kItemAmmoMax;
|
|
}
|
|
|
|
bool isActive()
|
|
{
|
|
if (!hasX())
|
|
return false;
|
|
|
|
switch (x().aiState->stateType)
|
|
{
|
|
case kAiStateIdle:
|
|
case kAiStateGenIdle:
|
|
case kAiStateSearch:
|
|
case kAiStateMove:
|
|
case kAiStateOther:
|
|
return false;
|
|
default:
|
|
return true;
|
|
}
|
|
}
|
|
|
|
void addExtra()
|
|
{
|
|
if (s().extra <= 0) s().extra = dbInsertXSprite(index);
|
|
}
|
|
|
|
};
|
|
|
|
extern DBloodActor bloodActors[kMaxSprites];
|
|
|
|
inline DBloodActor* DBloodActor::base() { return bloodActors; }
|
|
|
|
// Iterator wrappers that return an actor pointer, not an index.
|
|
class BloodStatIterator : public StatIterator
|
|
{
|
|
public:
|
|
BloodStatIterator(int stat) : StatIterator(stat)
|
|
{
|
|
}
|
|
|
|
DBloodActor* Next()
|
|
{
|
|
int n = NextIndex();
|
|
return n >= 0 ? &bloodActors[n] : nullptr;
|
|
}
|
|
|
|
DBloodActor* Peek()
|
|
{
|
|
int n = PeekIndex();
|
|
return n >= 0 ? &bloodActors[n] : nullptr;
|
|
}
|
|
};
|
|
|
|
class BloodSectIterator : public SectIterator
|
|
{
|
|
public:
|
|
BloodSectIterator(int stat) : SectIterator(stat)
|
|
{
|
|
}
|
|
|
|
DBloodActor* Next()
|
|
{
|
|
int n = NextIndex();
|
|
return n >= 0 ? &bloodActors[n] : nullptr;
|
|
}
|
|
|
|
DBloodActor* Peek()
|
|
{
|
|
int n = PeekIndex();
|
|
return n >= 0 ? &bloodActors[n] : nullptr;
|
|
}
|
|
};
|
|
|
|
inline int DeleteSprite(DBloodActor* nSprite)
|
|
{
|
|
if (nSprite) return DeleteSprite(nSprite->s().index);
|
|
return 0;
|
|
}
|
|
|
|
inline void actBurnSprite(DBloodActor* pSource, DBloodActor* pTarget, int nTime)
|
|
{
|
|
auto pXSprite = &pTarget->x();
|
|
pXSprite->burnTime = ClipHigh(pXSprite->burnTime + nTime, sprite[pXSprite->reference].statnum == kStatDude ? 2400 : 1200);
|
|
pXSprite->burnSource = pSource? pSource->s().index : -1;
|
|
}
|
|
|
|
inline void GetActorExtents(DBloodActor* actor, int* top, int* bottom)
|
|
{
|
|
GetSpriteExtents(&actor->s(), top, bottom);
|
|
}
|
|
|
|
inline DBloodActor *PLAYER::fragger()
|
|
{
|
|
return fraggerId == -1? nullptr : &bloodActors[fraggerId];
|
|
}
|
|
|
|
inline void PLAYER::setFragger(DBloodActor* actor)
|
|
{
|
|
fraggerId = actor == nullptr ? -1 : actor->s().index;
|
|
}
|
|
|
|
|
|
// Wrapper around the insane collision info mess from Build.
|
|
struct Collision
|
|
{
|
|
int type;
|
|
int index;
|
|
int legacyVal; // should be removed later, but needed for converting back for unadjusted code.
|
|
DBloodActor* actor;
|
|
|
|
Collision() = default;
|
|
Collision(int legacyval) { setFromEngine(legacyval); }
|
|
|
|
int setNone()
|
|
{
|
|
type = kHitNone;
|
|
index = -1;
|
|
legacyVal = 0;
|
|
actor = nullptr;
|
|
return kHitNone;
|
|
}
|
|
|
|
int setSector(int num)
|
|
{
|
|
type = kHitSector;
|
|
index = num;
|
|
legacyVal = type | index;
|
|
actor = nullptr;
|
|
return kHitSector;
|
|
}
|
|
int setWall(int num)
|
|
{
|
|
type = kHitWall;
|
|
index = num;
|
|
legacyVal = type | index;
|
|
actor = nullptr;
|
|
return kHitWall;
|
|
}
|
|
int setSprite(DBloodActor* num)
|
|
{
|
|
type = kHitSprite;
|
|
index = -1;
|
|
legacyVal = type | int(num - bloodActors);
|
|
actor = num;
|
|
return kHitSprite;
|
|
}
|
|
|
|
int setFromEngine(int value)
|
|
{
|
|
legacyVal = value;
|
|
type = value & kHitTypeMask;
|
|
if (type == 0) { index = -1; actor = nullptr; }
|
|
else if (type != kHitSprite) { index = value & kHitIndexMask; actor = nullptr; }
|
|
else { index = -1; actor = &bloodActors[value & kHitIndexMask]; }
|
|
return type;
|
|
}
|
|
};
|
|
|
|
|
|
inline DBloodActor* getUpperLink(int sect)
|
|
{
|
|
auto l = gUpperLink[sect];
|
|
return l == -1 ? nullptr : &bloodActors[l];
|
|
}
|
|
|
|
inline DBloodActor* getLowerLink(int sect)
|
|
{
|
|
auto l = gLowerLink[sect];
|
|
return l == -1 ? nullptr : &bloodActors[l];
|
|
}
|
|
|
|
inline void viewBackupSpriteLoc(DBloodActor* actor)
|
|
{
|
|
viewBackupSpriteLoc(actor->s().index, &actor->s());
|
|
}
|
|
|
|
inline FSerializer& Serialize(FSerializer& arc, const char* keyname, DBloodActor*& w, DBloodActor** def)
|
|
{
|
|
int index = w? int(w - bloodActors) : -1;
|
|
Serialize(arc, keyname, index, nullptr);
|
|
if (arc.isReading()) w = index == -1? nullptr : &bloodActors[index];
|
|
return arc;
|
|
}
|
|
|
|
END_BLD_NS
|