//------------------------------------------------------------------------- /* Copyright (C) 2010-2019 EDuke32 developers and contributors Copyright (C) 2019 sirlemonhead, Nuke.YKT This file is part of PCExhumed. PCExhumed is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License version 2 as published by the Free Software Foundation. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ //------------------------------------------------------------------------- #include "ns.h" #include "aistuff.h" #include "exhumed.h" #include "engine.h" #include "sequence.h" #include "sound.h" #include BEGIN_PS_NS static actionSeq SpiderSeq[] = { {16, 0}, {8, 0}, {32, 0}, {24, 0}, {0, 0}, {40, 1}, {41, 1} }; DExhumedActor* BuildSpider(DExhumedActor* spp, int x, int y, int z, sectortype* pSector, int nAngle) { if (spp == nullptr) { spp = insertActor(pSector, 99); } else { ChangeActorStat(spp, 99); x = spp->spr.pos.X; y = spp->spr.pos.Y; z = spp->sector()->floorz; nAngle = spp->spr.ang; } spp->spr.pos.X = x; spp->spr.pos.Y = y; spp->spr.pos.Z = z; spp->spr.cstat = CSTAT_SPRITE_BLOCK_ALL; spp->spr.shade = -12; spp->spr.clipdist = 15; spp->spr.xvel = 0; spp->spr.yvel = 0; spp->spr.zvel = 0; spp->spr.xrepeat = 40; spp->spr.yrepeat = 40; spp->spr.pal = spp->sector()->ceilingpal; spp->spr.xoffset = 0; spp->spr.yoffset = 0; spp->spr.ang = nAngle; spp->spr.picnum = 1; spp->spr.hitag = 0; spp->spr.lotag = runlist_HeadRun() + 1; spp->spr.extra = -1; // GrabTimeSlot(3); spp->nAction = 0; spp->nFrame = 0; spp->pTarget = nullptr; spp->nHealth = 160; spp->nPhase = Counters[kCountSpider]++; spp->spr.owner = runlist_AddRunRec(spp->spr.lotag - 1, spp, 0xC0000); spp->nRun = runlist_AddRunRec(NewRun, spp, 0xC0000); nCreaturesTotal++; return spp; } void AISpider::Tick(RunListEvent* ev) { auto spp = ev->pObjActor; if (!spp) return; int nAction = spp->nAction; int nVel = 6; if (spp->nHealth) { if (spp->spr.cstat & CSTAT_SPRITE_YFLIP) { spp->spr.pos.Z = spp->sector()->ceilingz + GetActorHeight(spp); } else { Gravity(spp); } } int nSeq = SeqOffsets[kSeqSpider] + SpiderSeq[nAction].a; spp->spr.picnum = seq_GetSeqPicnum2(nSeq, spp->nFrame); seq_MoveSequence(spp, nSeq, spp->nFrame); int nFrameFlag = FrameFlag[SeqBase[nSeq] + spp->nFrame]; spp->nFrame++; if (spp->nFrame >= SeqSize[nSeq]) { spp->nFrame = 0; } DExhumedActor* pTarget = spp->pTarget; if (pTarget == nullptr || pTarget->spr.cstat & CSTAT_SPRITE_BLOCK_ALL) { switch (nAction) { default: return; case 0: { if ((spp->nPhase & 0x1F) == (totalmoves & 0x1F)) { if (pTarget == nullptr) { pTarget = FindPlayer(spp, 100); } if (pTarget) { spp->nAction = 1; spp->nFrame = 0; spp->pTarget = pTarget; spp->spr.xvel = bcos(spp->spr.ang); spp->spr.yvel = bsin(spp->spr.ang); return; } } break; } case 1: { if (pTarget) { nVel++; } goto case_3; break; } case 4: { if (!spp->nFrame) { spp->nFrame = 0; spp->nAction = 1; } [[fallthrough]]; } case 3: { case_3: auto pSector =spp->sector(); if (spp->spr.cstat & CSTAT_SPRITE_YFLIP) { spp->spr.zvel = 0; spp->spr.pos.Z = pSector->ceilingz + (tileHeight(spp->spr.picnum) << 5); if (pSector->ceilingstat & CSTAT_SECTOR_SKY) { spp->spr.cstat ^= CSTAT_SPRITE_YFLIP; spp->spr.zvel = 1; spp->nAction = 3; spp->nFrame = 0; } } if ((totalmoves & 0x1F) == (spp->nPhase & 0x1F)) { PlotCourseToSprite(spp, pTarget); if (RandomSize(3)) { spp->spr.xvel = bcos(spp->spr.ang); spp->spr.yvel = bsin(spp->spr.ang); } else { spp->spr.xvel = 0; spp->spr.yvel = 0; } if (spp->nAction == 1 && RandomBit()) { if (spp->spr.cstat & CSTAT_SPRITE_YFLIP) { spp->spr.cstat ^= CSTAT_SPRITE_YFLIP; spp->spr.zvel = 1; spp->spr.pos.Z = pSector->ceilingz + GetActorHeight(spp); } else { spp->spr.zvel = -5120; } spp->nAction = 3; spp->nFrame = 0; if (!RandomSize(3)) { D3PlayFX(StaticSound[kSound29], spp); } } } break; } case 5: { if (!spp->nFrame) { runlist_DoSubRunRec(spp->spr.owner); runlist_FreeRun(spp->spr.lotag - 1); runlist_SubRunRec(spp->nRun); spp->spr.cstat = CSTAT_SPRITE_INVISIBLE; DeleteActor(spp); } return; } case 2: { if (pTarget) { if (nFrameFlag & 0x80) { runlist_DamageEnemy(pTarget, spp, 3); D3PlayFX(StaticSound[kSound38], spp); } if (PlotCourseToSprite(spp, pTarget) < 1024) { return; } spp->nAction = 1; } else { spp->nAction = 0; spp->spr.xvel = 0; spp->spr.yvel = 0; } spp->nFrame = 0; break; } } } else { spp->pTarget = nullptr; spp->nAction = 0; spp->nFrame = 0; spp->spr.xvel = 0; spp->spr.yvel = 0; } auto nMov = movesprite(spp, spp->spr.xvel << nVel, spp->spr.yvel << nVel, spp->spr.zvel, 1280, -1280, CLIPMASK0); if (nMov.type == kHitNone && nMov.exbits == 0) return; if (nMov.exbits & kHitAux1 && spp->spr.zvel < 0 && hiHit.type != kHitSprite && !((spp->sector()->ceilingstat) & CSTAT_SECTOR_SKY)) { spp->spr.cstat |= CSTAT_SPRITE_YFLIP; spp->spr.pos.Z = GetActorHeight(spp) + spp->sector()->ceilingz; spp->spr.zvel = 0; spp->nAction = 1; spp->nFrame = 0; return; } else { switch (nMov.type) { case kHitWall: { spp->spr.ang = (spp->spr.ang + 256) & 0x7EF; spp->spr.xvel = bcos(spp->spr.ang); spp->spr.yvel = bsin(spp->spr.ang); return; } case kHitSprite: { if (nMov.actor() == pTarget) { int nAng = getangle(pTarget->spr.pos.X - spp->spr.pos.X, pTarget->spr.pos.Y - spp->spr.pos.Y); if (AngleDiff(spp->spr.ang, nAng) < 64) { spp->nAction = 2; spp->nFrame = 0; } } return; } default: break; } if (spp->nAction == 3) { spp->nAction = 1; spp->nFrame = 0; } return; } return; } void AISpider::Draw(RunListEvent* ev) { auto spp = ev->pObjActor; if (!spp) return; int nAction = spp->nAction; seq_PlotSequence(ev->nParam, SeqOffsets[kSeqSpider] + SpiderSeq[nAction].a, spp->nFrame, SpiderSeq[nAction].b); } void AISpider::RadialDamage(RunListEvent* ev) { auto spp = ev->pObjActor; if (!spp) return; if (spp->nHealth <= 0) return; ev->nDamage = runlist_CheckRadialDamage(spp); Damage(ev); } void AISpider::Damage(RunListEvent* ev) { auto spp = ev->pObjActor; if (!spp) return; if (!ev->nDamage) return; DExhumedActor* pTarget = ev->pOtherActor; spp->nHealth -= dmgAdjust(ev->nDamage); if (spp->nHealth > 0) { /* NOTE: nTarget check was added, but should we return if it's invalid instead or should code below (action set, b set) happen? Other AI doesn't show consistency in this regard (see Scorpion code) */ if (pTarget && pTarget->spr.statnum == 100) { spp->pTarget = pTarget; } spp->nAction = 4; spp->nFrame = 0; } else { // creature is dead, make some chunks spp->nHealth = 0; spp->nAction = 5; spp->nFrame = 0; spp->spr.cstat &= ~CSTAT_SPRITE_BLOCK_ALL; nCreaturesKilled++; for (int i = 0; i < 7; i++) { BuildCreatureChunk(spp, seq_GetSeqPicnum(kSeqSpider, i + 41, 0)); } } } END_PS_NS