#pragma once #include "coreactor.h" BEGIN_BLD_NS class DBloodActor; struct SPRITEHIT { Collision hit, ceilhit, florhit; }; class DBloodActor : public DCoreActor { DBloodActor* base(); public: int dudeSlope; int xvel, yvel, zvel; bool hasx; XSPRITE xsprite; SPRITEHIT hit; DUDEEXTRA dudeExtra; SPRITEMASS spriteMass; GENDUDEEXTRA genDudeExtra; DBloodActor* prevmarker; // needed by the nnext marker code. This originally hijacked targetX in XSPRITE DBloodActor* ownerActor; // was previously stored in the sprite's owner field. POINT3D basePoint; EventObject condition[2]; bool explosionhackflag; // this originally hijacked the target field which is not safe when working with pointers. // transient data (not written to savegame) int cumulDamage; bool interpolated; DBloodActor() { index = (int(this - base())); } DBloodActor& operator=(const DBloodActor& other) = default; void Clear() { dudeSlope = 0; hit = {}; dudeExtra = {}; spriteMass = {}; genDudeExtra = {}; prevmarker = nullptr; basePoint = {}; xsprite = {}; hasx = false; interpolated = false; xvel = yvel = zvel = 0; explosionhackflag = false; interpolated = false; } bool hasX() { return hasx; } void addX() { hasx = true; } XSPRITE& x() { return xsprite; } // calling this does not validate the xsprite! void SetOwner(DBloodActor* own) { ownerActor = own; } DBloodActor* GetOwner() { return ownerActor; } void SetTarget(DBloodActor* own) { x().target = own; } DBloodActor* GetTarget() { return x().target; } bool ValidateTarget(const char* func) { if (GetTarget() == nullptr) { Printf(PRINT_HIGH | PRINT_NOTIFY, "%s: invalid target in calling actor\n", func); return false; } return true; } void SetBurnSource(DBloodActor* own) { x().burnSource = own; } DBloodActor* GetBurnSource() { return x().burnSource; } void SetSpecialOwner() // nnext hackery { ownerActor = nullptr; s().owner = kMaxSprites - 1; } bool GetSpecialOwner() { return ownerActor == nullptr && (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; } } }; extern DBloodActor bloodActors[kMaxSprites]; inline DBloodActor* DBloodActor::base() { return bloodActors; } // subclassed to add a game specific actor() method extern HitInfo gHitInfo; // Iterator wrappers that return an actor pointer, not an index. using BloodStatIterator = TStatIterator; using BloodSectIterator = TSectIterator; using BloodSpriteIterator = TSpriteIterator; inline void GetActorExtents(DBloodActor* actor, int* top, int* bottom) { GetSpriteExtents(&actor->s(), top, bottom); } END_BLD_NS