raze/source/games/exhumed/src/bullet.cpp

881 lines
23 KiB
C++
Raw Normal View History

//-------------------------------------------------------------------------
/*
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 "engine.h"
#include "aistuff.h"
#include "sequence.h"
#include "exhumed.h"
#include "sound.h"
#include "player.h"
#include "names.h"
#include <string.h>
#include <assert.h>
#ifndef __WATCOMC__
//#include <cmath>
#else
//#include <math.h>
#include <stdlib.h>
#endif
BEGIN_PS_NS
enum { kMaxBullets = 500 };
// 32 bytes
struct Bullet
{
short nSeq;
short nFrame;
short nSprite;
2019-09-25 21:16:12 +00:00
short field_6;
short field_8;
short nType;
short field_C;
short field_E;
2019-10-01 17:59:20 +00:00
uint16_t field_10;
2019-09-25 21:16:12 +00:00
uint8_t field_12;
uint8_t field_13;
int x;
int y;
int z;
2020-11-30 00:10:52 +00:00
int enemy;
};
2020-11-30 00:10:52 +00:00
FreeListArray<Bullet, kMaxBullets> BulletList;
int lasthitz, lasthitx, lasthity;
short lasthitsect, lasthitsprite, lasthitwall;
short nRadialBullet = 0;
2020-11-30 00:10:52 +00:00
FSerializer& Serialize(FSerializer& arc, const char* keyname, Bullet& w, Bullet* def)
{
static Bullet nul;
if (!def)
{
def = &nul;
if (arc.isReading()) w = {};
}
if (arc.BeginObject(keyname))
{
arc("seq", w.nSeq, def->nSeq)
("frame", w.nFrame, def->nFrame)
("sprite", w.nSprite, def->nSprite)
("type", w.nType, def->nType)
("x", w.x, def->x)
("y", w.y, def->y)
("z", w.z, def->z)
("at6", w.field_6, def->field_6)
("at8", w.field_8, def->field_8)
("atc", w.field_C, def->field_C)
("ate", w.field_E, def->field_E)
("at10", w.field_10, def->field_10)
("at12", w.field_12, def->field_12)
("at13", w.field_13, def->field_13)
("enemy", w.enemy, def->enemy)
.EndObject();
}
return arc;
}
2020-11-30 00:10:52 +00:00
void SerializeBullet(FSerializer& arc)
{
if (arc.BeginObject("bullets"))
{
arc ("list", BulletList)
("lasthitx", lasthitx)
("lasthity", lasthity)
("lasthitz", lasthitz)
("lasthitsect", lasthitsect)
("lasthitspr", lasthitsprite)
("lasthitwall", lasthitwall)
("radialbullet", nRadialBullet)
.EndObject();
}
}
bulletInfo BulletInfo[] = {
{ 25, 1, 20, -1, -1, 13, 0, 0, -1 },
{ 25, -1, 65000, -1, 31, 73, 0, 0, -1 },
{ 15, -1, 60000, -1, 31, 73, 0, 0, -1 },
{ 5, 15, 2000, -1, 14, 38, 4, 5, 3 },
{ 250, 100, 2000, -1, 33, 34, 4, 20, -1 },
{ 200, -1, 2000, -1, 20, 23, 4, 10, -1 },
{ 200, -1, 60000, 68, 68, -1, -1, 0, -1 },
{ 300, 1, 0, -1, -1, -1, 0, 50, -1 },
{ 18, -1, 2000, -1, 18, 29, 4, 0, -1 },
{ 20, -1, 2000, 37, 11, 30, 4, 0, -1 },
{ 25, -1, 3000, -1, 44, 36, 4, 15, 90 },
{ 30, -1, 1000, -1, 52, 53, 4, 20, 48 },
{ 20, -1, 3500, -1, 54, 55, 4, 30, -1 },
{ 10, -1, 5000, -1, 57, 76, 4, 0, -1 },
{ 40, -1, 1500, -1, 63, 38, 4, 10, 40 },
{ 20, -1, 2000, -1, 60, 12, 0, 0, -1 },
{ 5, -1, 60000, -1, 31, 76, 0, 0, -1 }
};
void InitBullets()
{
2020-11-30 00:10:52 +00:00
BulletList.Clear();
}
2020-11-30 00:10:52 +00:00
int GrabBullet()
{
2020-11-30 00:10:52 +00:00
int grabbed = BulletList.Get();
if (grabbed < 0) return -1;
BulletList[grabbed].enemy = -1;
return grabbed;
}
void DestroyBullet(short nBullet)
{
2019-09-25 21:16:12 +00:00
short nSprite = BulletList[nBullet].nSprite;
auto pSprite = &sprite[nSprite];
2019-09-25 21:16:12 +00:00
runlist_DoSubRunRec(BulletList[nBullet].field_6);
runlist_DoSubRunRec(pSprite->lotag - 1);
2019-09-25 21:16:12 +00:00
runlist_SubRunRec(BulletList[nBullet].field_8);
2019-09-25 21:16:12 +00:00
StopSpriteSound(nSprite);
2019-09-25 21:16:12 +00:00
mydeletesprite(nSprite);
2020-11-30 00:10:52 +00:00
BulletList.Release(nBullet);
}
void IgniteSprite(int nSprite)
{
auto pSprite = &sprite[nSprite];
pSprite->hitag += 2;
int nAnim = BuildAnim(-1, 38, 0, pSprite->x, pSprite->y, pSprite->z, pSprite->sectnum, 40, 20);
2019-09-25 21:16:12 +00:00
short nAnimSprite = GetAnimSprite(nAnim);
2019-09-25 21:16:12 +00:00
sprite[nAnimSprite].hitag = nSprite;
changespritestat(nAnimSprite, kStatIgnited);
short yRepeat = (tileHeight(sprite[nAnimSprite].picnum) * 32) / nFlameHeight;
2019-09-25 21:16:12 +00:00
if (yRepeat < 1)
yRepeat = 1;
2019-09-25 21:16:12 +00:00
sprite[nAnimSprite].yrepeat = (uint8_t)yRepeat;
}
void BulletHitsSprite(Bullet *pBullet, short nBulletSprite, short nHitSprite, int x, int y, int z, int nSector)
{
2019-09-25 21:16:12 +00:00
assert(nSector >= 0 && nSector < kMaxSectors);
bulletInfo *pBulletInfo = &BulletInfo[pBullet->nType];
short nStat = sprite[nHitSprite].statnum;
switch (pBullet->nType)
{
case 3:
{
if (nStat > 107 || nStat == kStatAnubisDrum) {
2019-09-25 21:16:12 +00:00
return;
}
sprite[nHitSprite].hitag++;
if (sprite[nHitSprite].hitag == 15) {
IgniteSprite(nHitSprite);
}
if (!RandomSize(2)) {
2021-10-24 17:18:11 +00:00
BuildAnim(nullptr, pBulletInfo->field_C, 0, x, y, z, nSector, 40, pBulletInfo->nFlags);
2019-09-25 21:16:12 +00:00
}
return;
}
case 14:
{
if (nStat > 107 || nStat == kStatAnubisDrum) {
2019-09-25 21:16:12 +00:00
return;
}
// else - fall through to below cases
fallthrough__;
}
case 1:
case 2:
case 8:
case 9:
case 12:
case 13:
case 15:
case 16:
{
// loc_29E59
if (!nStat || nStat > 98) {
break;
}
short nSprite = pBullet->nSprite;
spritetype *pSprite = &sprite[nSprite];
spritetype *pHitSprite = &sprite[nHitSprite];
if (nStat == kStatAnubisDrum)
{
short nAngle = (pSprite->ang + 256) - RandomSize(9);
pHitSprite->xvel = bcos(nAngle, 1);
pHitSprite->yvel = bsin(nAngle, 1);
pHitSprite->zvel = (-(RandomSize(3) + 1)) << 8;
}
else
2019-09-25 21:16:12 +00:00
{
int xVel = pHitSprite->xvel;
int yVel = pHitSprite->yvel;
pHitSprite->xvel = bcos(pSprite->ang, -2);
pHitSprite->yvel = bsin(pSprite->ang, -2);
2019-09-25 21:16:12 +00:00
MoveCreature(nHitSprite);
pHitSprite->xvel = xVel;
pHitSprite->yvel = yVel;
}
break;
}
default:
break;
}
// BHS_switchBreak:
short nDamage = pBulletInfo->nDamage;
if (pBullet->field_13 > 1) {
nDamage *= 2;
}
runlist_DamageEnemy(nHitSprite, nBulletSprite, nDamage);
if (nStat <= 90 || nStat >= 199)
{
BuildAnim(nullptr, pBulletInfo->field_C, 0, x, y, z, nSector, 40, pBulletInfo->nFlags);
2019-09-25 21:16:12 +00:00
return;
}
switch (nStat)
{
case kStatDestructibleSprite:
break;
case kStatAnubisDrum:
case 102:
case kStatExplodeTrigger:
case kStatExplodeTarget:
BuildAnim(nullptr, 12, 0, x, y, z, nSector, 40, 0);
break;
default:
BuildAnim(nullptr, 39, 0, x, y, z, nSector, 40, 0);
if (pBullet->nType > 2)
{
BuildAnim(nullptr, pBulletInfo->field_C, 0, x, y, z, nSector, 40, pBulletInfo->nFlags);
}
break;
2019-09-25 21:16:12 +00:00
}
}
void BackUpBullet(int *x, int *y, short nAngle)
{
*x -= bcos(nAngle, -11);
*y -= bsin(nAngle, -11);
}
int MoveBullet(short nBullet)
{
2019-09-25 21:16:12 +00:00
short hitsect = -1;
short hitwall = -1;
short hitsprite = -1;
Bullet *pBullet = &BulletList[nBullet];
short nType = pBullet->nType;
bulletInfo *pBulletInfo = &BulletInfo[nType];
short nSprite = BulletList[nBullet].nSprite;
spritetype *pSprite = &sprite[nSprite];
int x = pSprite->x;
int y = pSprite->y;
int z = pSprite->z; // ebx
short nSectFlag = SectFlag[pSprite->sectnum];
int x2, y2, z2;
int nVal;
2019-10-01 17:59:20 +00:00
if (pBullet->field_10 < 30000)
2019-09-25 21:16:12 +00:00
{
2020-11-30 00:10:52 +00:00
short nEnemySprite = BulletList[nBullet].enemy;
2019-09-25 21:16:12 +00:00
if (nEnemySprite > -1)
{
if (!(sprite[nEnemySprite].cstat & 0x101))
2020-11-30 00:10:52 +00:00
BulletList[nBullet].enemy = -1;
2019-09-25 21:16:12 +00:00
else
{
2019-10-01 17:59:20 +00:00
nVal = AngleChase(nSprite, nEnemySprite, pBullet->field_10, 0, 16);
2019-09-25 21:16:12 +00:00
goto MOVEEND;
}
}
if (nType == 3)
{
if (pBullet->field_E < 8)
{
pSprite->xrepeat -= 1;
pSprite->yrepeat += 8;
pBullet->z -= 200;
if (pSprite->shade < 90) {
pSprite->shade += 35;
}
if (pBullet->field_E == 3)
{
pBullet->nSeq = 45;
pBullet->nFrame = 0;
2019-09-25 21:16:12 +00:00
pSprite->xrepeat = 40;
pSprite->yrepeat = 40;
pSprite->shade = 0;
pSprite->z += 512;
}
}
else
{
pSprite->xrepeat += 4;
pSprite->yrepeat += 4;
}
}
nVal = movesprite(nSprite, pBullet->x, pBullet->y, pBullet->z, pSprite->clipdist >> 1, pSprite->clipdist >> 1, CLIPMASK1);
MOVEEND:
if (nVal)
{
x2 = pSprite->x;
y2 = pSprite->y;
z2 = pSprite->z;
hitsect = pSprite->sectnum;
if (nVal & 0x30000)
{
hitwall = nVal & 0x3FFF;
goto HITWALL;
}
else
{
switch (nVal & 0xc000)
{
case 0x8000:
hitwall = nVal & 0x3FFF;
goto HITWALL;
case 0xc000:
hitsprite = nVal & 0x3FFF;
goto HITSPRITE;
}
}
}
// pSprite->sectnum may have changed since we set nSectFlag ?
2019-09-25 21:16:12 +00:00
short nFlagVal = nSectFlag ^ SectFlag[pSprite->sectnum];
if (nFlagVal & kSectUnderwater)
{
DestroyBullet(nBullet);
nVal = 1;
}
if (nVal == 0 && nType != 15 && nType != 3)
{
AddFlash(pSprite->sectnum, pSprite->x, pSprite->y, pSprite->z, 0);
2019-09-25 21:16:12 +00:00
if (pSprite->pal != 5) {
pSprite->pal = 1;
2019-09-25 21:16:12 +00:00
}
}
}
else
{
nVal = 1;
2020-11-30 00:10:52 +00:00
if (BulletList[nBullet].enemy > -1)
2019-09-25 21:16:12 +00:00
{
2020-11-30 00:10:52 +00:00
hitsprite = BulletList[nBullet].enemy;
2019-09-25 21:16:12 +00:00
x2 = sprite[hitsprite].x;
y2 = sprite[hitsprite].y;
z2 = sprite[hitsprite].z - (GetSpriteHeight(hitsprite) >> 1);
hitsect = sprite[hitsprite].sectnum;
}
else
{
vec3_t startPos = { x, y, z };
hitdata_t hitData;
2019-10-27 16:36:25 +00:00
int dz;
if (bVanilla)
dz = -bsin(pBullet->field_C, 3);
2019-10-27 16:36:25 +00:00
else
dz = -pBullet->field_C * 512;
hitscan(&startPos, pSprite->sectnum, bcos(pSprite->ang), bsin(pSprite->ang), dz, &hitData, CLIPMASK1);
x2 = hitData.pos.x;
y2 = hitData.pos.y;
z2 = hitData.pos.z;
hitsprite = hitData.sprite;
hitsect = hitData.sect;
hitwall = hitData.wall;
2019-09-25 21:16:12 +00:00
}
lasthitx = x2;
lasthity = y2;
lasthitz = z2;
lasthitsect = hitsect;
lasthitwall = hitwall;
lasthitsprite = hitsprite;
if (hitsprite > -1)
{
HITSPRITE:
if (pSprite->pal == 5 && sprite[hitsprite].statnum == 100)
2019-09-25 21:16:12 +00:00
{
short nPlayer = GetPlayerFromSprite(hitsprite);
if (!PlayerList[nPlayer].bIsMummified)
{
PlayerList[nPlayer].bIsMummified = true;
2019-09-25 21:16:12 +00:00
SetNewWeapon(nPlayer, kWeaponMummified);
}
}
else
{
BulletHitsSprite(pBullet, pSprite->owner, hitsprite, x2, y2, z2, hitsect);
}
}
else if (hitwall > -1)
{
HITWALL:
if (wall[hitwall].picnum == kEnergy1)
{
short nSector = wall[hitwall].nextsector;
if (nSector > -1)
{
short nDamage = BulletInfo[pBullet->nType].nDamage;
if (pBullet->field_13 > 1) {
nDamage *= 2;
}
short nNormal = GetWallNormal(hitwall) & kAngleMask;
runlist_DamageEnemy(sector[nSector].extra, nNormal, nDamage);
}
}
}
if (hitsect > -1) // NOTE: hitsect can be -1. this check wasn't in original code. TODO: demo compatiblity?
2019-09-25 21:16:12 +00:00
{
if (hitsprite < 0 && hitwall < 0)
2019-09-25 21:16:12 +00:00
{
if ((SectBelow[hitsect] >= 0 && (SectFlag[SectBelow[hitsect]] & kSectUnderwater)) || SectDepth[hitsect])
{
pSprite->x = x2;
pSprite->y = y2;
pSprite->z = z2;
2021-10-24 17:18:11 +00:00
BuildSplash(&exhumedActors[nSprite], hitsect);
}
else
{
BuildAnim(nullptr, pBulletInfo->field_C, 0, x2, y2, z2, hitsect, 40, pBulletInfo->nFlags);
}
2019-09-25 21:16:12 +00:00
}
else
{
if (hitwall >= 0)
{
BackUpBullet(&x2, &y2, pSprite->ang);
2019-09-25 21:16:12 +00:00
if (nType != 3 || RandomSize(2) == 0)
{
int zOffset = RandomSize(8) << 3;
2019-09-25 21:16:12 +00:00
if (!RandomBit()) {
zOffset = -zOffset;
}
2019-09-25 21:16:12 +00:00
// draws bullet puff on walls when they're shot
BuildAnim(nullptr, pBulletInfo->field_C, 0, x2, y2, z2 + zOffset + -4096, hitsect, 40, pBulletInfo->nFlags);
}
}
else
{
pSprite->x = x2;
pSprite->y = y2;
pSprite->z = z2;
2019-09-25 21:16:12 +00:00
mychangespritesect(nSprite, hitsect);
}
2019-09-25 21:16:12 +00:00
if (BulletInfo[nType].nRadius)
{
nRadialBullet = nType;
2019-09-25 21:16:12 +00:00
runlist_RadialDamageEnemy(nSprite, pBulletInfo->nDamage, pBulletInfo->nRadius);
2019-09-25 21:16:12 +00:00
nRadialBullet = -1;
2019-09-25 21:16:12 +00:00
AddFlash(pSprite->sectnum, pSprite->x, pSprite->y, pSprite->z, 128);
}
}
2019-09-25 21:16:12 +00:00
}
DestroyBullet(nBullet);
}
return nVal;
}
void SetBulletEnemy(short nBullet, short nEnemy)
{
2019-09-25 21:16:12 +00:00
if (nBullet >= 0) {
2020-11-30 00:10:52 +00:00
BulletList[nBullet].enemy = nEnemy;
2019-09-25 21:16:12 +00:00
}
}
int BuildBullet(short nSprite, int nType, int, int, int val1, int nAngle, int val2, int val3)
{
auto pSprite = &sprite[nSprite];
2019-09-25 21:16:12 +00:00
Bullet sBullet;
bulletInfo *pBulletInfo = &BulletInfo[nType];
2019-09-25 21:16:12 +00:00
if (pBulletInfo->field_4 > 30000)
{
if (val2 >= 10000)
{
val2 -= 10000;
2019-09-25 21:16:12 +00:00
short nTargetSprite = val2;
spritetype *pTargetSprite = &sprite[nTargetSprite];
// assert(pTargetSprite->sectnum <= kMaxSectors);
2019-09-25 21:16:12 +00:00
if (pTargetSprite->cstat & 0x101)
{
sBullet.nType = nType;
sBullet.field_13 = val3;
sBullet.nSprite = insertsprite(pSprite->sectnum, 200);
2019-09-25 21:16:12 +00:00
sprite[sBullet.nSprite].ang = nAngle;
int nHeight = GetSpriteHeight(nTargetSprite);
assert(pTargetSprite->sectnum >= 0 && pTargetSprite->sectnum < kMaxSectors);
2019-09-25 21:16:12 +00:00
BulletHitsSprite(&sBullet, nSprite, nTargetSprite, pTargetSprite->x, pTargetSprite->y, pTargetSprite->z - (nHeight >> 1), pTargetSprite->sectnum);
mydeletesprite(sBullet.nSprite);
return -1;
}
else
{
val2 = 0;
}
}
}
2020-11-30 00:10:52 +00:00
int nBullet = GrabBullet();
if (nBullet < 0) {
2019-09-25 21:16:12 +00:00
return -1;
}
short nSector;
if (pSprite->statnum == 100)
2019-09-25 21:16:12 +00:00
{
nSector = nPlayerViewSect[GetPlayerFromSprite(nSprite)];
}
else
{
nSector = pSprite->sectnum;
2019-09-25 21:16:12 +00:00
}
short nBulletSprite = insertsprite(nSector, 200);
auto pBulletSprite = &sprite[nBulletSprite];
2019-09-25 21:16:12 +00:00
int nHeight = GetSpriteHeight(nSprite);
nHeight = nHeight - (nHeight >> 2);
if (val1 == -1) {
val1 = -nHeight;
}
pBulletSprite->x = pSprite->x;
pBulletSprite->y = pSprite->y;
pBulletSprite->z = pSprite->z;
2019-09-25 21:16:12 +00:00
// why is this done here???
assert(nBulletSprite >= 0 && nBulletSprite < kMaxSprites);
Bullet *pBullet = &BulletList[nBullet];
2020-11-30 00:10:52 +00:00
pBullet->enemy = -1;
2019-09-25 21:16:12 +00:00
pBulletSprite->cstat = 0;
pBulletSprite->shade = -64;
2019-09-25 21:16:12 +00:00
if (pBulletInfo->nFlags & 4) {
pBulletSprite->pal = 4;
2019-09-25 21:16:12 +00:00
}
else {
pBulletSprite->pal = 0;
2019-09-25 21:16:12 +00:00
}
pBulletSprite->clipdist = 25;
2019-09-25 21:16:12 +00:00
short nRepeat = pBulletInfo->xyRepeat;
if (nRepeat < 0) {
nRepeat = 30;
}
pBulletSprite->xrepeat = (uint8_t)nRepeat;
pBulletSprite->yrepeat = (uint8_t)nRepeat;
pBulletSprite->xoffset = 0;
pBulletSprite->yoffset = 0;
pBulletSprite->ang = nAngle;
pBulletSprite->xvel = 0;
pBulletSprite->yvel = 0;
pBulletSprite->zvel = 0;
pBulletSprite->owner = nSprite;
pBulletSprite->lotag = runlist_HeadRun() + 1;
pBulletSprite->extra = -1;
pBulletSprite->hitag = 0;
// GrabTimeSlot(3);
2019-09-25 21:16:12 +00:00
pBullet->field_10 = 0;
pBullet->field_E = pBulletInfo->field_2;
pBullet->nFrame = 0;
2019-09-25 21:16:12 +00:00
short nSeq;
if (pBulletInfo->field_8 != -1)
{
pBullet->field_12 = 0;
nSeq = pBulletInfo->field_8;
}
else
{
pBullet->field_12 = 1;
nSeq = pBulletInfo->nSeq;
}
pBullet->nSeq = nSeq;
pBulletSprite->picnum = seq_GetSeqPicnum(nSeq, 0, 0);
2019-09-25 21:16:12 +00:00
if (nSeq == kSeqBullet) {
pBulletSprite->cstat |= 0x8000;
2019-09-25 21:16:12 +00:00
}
2019-10-27 16:36:25 +00:00
pBullet->field_C = val2;
2019-09-25 21:16:12 +00:00
pBullet->nType = nType;
pBullet->nSprite = nBulletSprite;
pBullet->field_6 = runlist_AddRunRec(pBulletSprite->lotag - 1, nBullet, 0xB0000);
pBullet->field_8 = runlist_AddRunRec(NewRun, nBullet, 0xB0000);
2019-09-25 21:16:12 +00:00
pBullet->field_13 = val3;
pBulletSprite->z += val1;
pBulletSprite->backuppos();
2019-09-25 21:16:12 +00:00
int var_18;
nSector = pBulletSprite->sectnum;
2019-09-25 21:16:12 +00:00
while (pBulletSprite->z < sector[nSector].ceilingz)
2019-09-25 21:16:12 +00:00
{
if (SectAbove[nSector] == -1)
{
pBulletSprite->z = sector[nSector].ceilingz;
2019-09-25 21:16:12 +00:00
break;
}
nSector = SectAbove[nSector];
mychangespritesect(nBulletSprite, nSector);
}
if (val2 < 10000)
{
var_18 = (-bsin(val2) * pBulletInfo->field_4) >> 11;
2019-09-25 21:16:12 +00:00
}
else
{
val2 -= 10000;
short nTargetSprite = val2;
auto pTargetSprite = &sprite[nTargetSprite];
2019-09-25 21:16:12 +00:00
if ((unsigned int)pBulletInfo->field_4 > 30000)
{
2020-11-30 00:10:52 +00:00
BulletList[nBullet].enemy = nTargetSprite;
2019-09-25 21:16:12 +00:00
}
else
{
nHeight = GetSpriteHeight(nTargetSprite);
if (pTargetSprite->statnum == 100)
2019-09-25 21:16:12 +00:00
{
nHeight -= nHeight >> 2;
}
else
{
nHeight -= nHeight >> 1;
}
int var_20 = pTargetSprite->z - nHeight;
2019-09-25 21:16:12 +00:00
int x, y;
if (nSprite != -1 && pSprite->statnum != 100)
2019-09-25 21:16:12 +00:00
{
x = pTargetSprite->x;
y = pTargetSprite->y;
2019-09-25 21:16:12 +00:00
if (pTargetSprite->statnum != 100)
2019-09-25 21:16:12 +00:00
{
x += (pTargetSprite->xvel * 20) >> 6;
y += (pTargetSprite->yvel * 20) >> 6;
2019-09-25 21:16:12 +00:00
}
else
{
int nPlayer = GetPlayerFromSprite(nTargetSprite);
if (nPlayer > -1)
{
x += nPlayerDX[nPlayer] * 15;
y += nPlayerDY[nPlayer] * 15;
}
}
x -= pBulletSprite->x;
y -= pBulletSprite->y;
2019-09-25 21:16:12 +00:00
nAngle = GetMyAngle(x, y);
pSprite->ang = nAngle;
2019-09-25 21:16:12 +00:00
}
else
{
// loc_2ABA3:
x = pTargetSprite->x - pBulletSprite->x;
y = pTargetSprite->y - pBulletSprite->y;
2019-09-25 21:16:12 +00:00
}
int nSqrt = lsqrt(y*y + x*x);
if ((unsigned int)nSqrt > 0)
{
var_18 = ((var_20 - pBulletSprite->z) * pBulletInfo->field_4) / nSqrt;
2019-09-25 21:16:12 +00:00
}
else
{
var_18 = 0;
}
}
}
pBullet->z = 0;
pBullet->x = (pSprite->clipdist << 2) * bcos(nAngle);
pBullet->y = (pSprite->clipdist << 2) * bsin(nAngle);
2020-11-30 00:10:52 +00:00
BulletList[nBullet].enemy = -1;
2019-09-25 21:16:12 +00:00
if (MoveBullet(nBullet))
{
nBulletSprite = -1;
}
else
{
pBullet->field_10 = pBulletInfo->field_4;
pBullet->x = bcos(nAngle, -3) * pBulletInfo->field_4;
pBullet->y = bsin(nAngle, -3) * pBulletInfo->field_4;
2019-09-25 21:16:12 +00:00
pBullet->z = var_18 >> 3;
}
return nBulletSprite | (nBullet << 16);
}
2021-10-15 19:10:16 +00:00
void AIBullet::Tick(RunListEvent* ev)
{
short nBullet = RunData[ev->nRun].nObjIndex;
2019-09-25 21:16:12 +00:00
assert(nBullet >= 0 && nBullet < kMaxBullets);
short nSeq = SeqOffsets[BulletList[nBullet].nSeq];
short nSprite = BulletList[nBullet].nSprite;
2021-10-15 19:10:16 +00:00
auto pSprite = &sprite[nSprite];
2019-09-25 21:16:12 +00:00
2021-10-15 19:10:16 +00:00
short nFlag = FrameFlag[SeqBase[nSeq] + BulletList[nBullet].nFrame];
2019-09-25 21:16:12 +00:00
2021-10-15 19:10:16 +00:00
seq_MoveSequence(nSprite, nSeq, BulletList[nBullet].nFrame);
2019-09-25 21:16:12 +00:00
2021-10-15 19:10:16 +00:00
if (nFlag & 0x80)
{
2021-10-24 17:18:11 +00:00
BuildAnim(nullptr, 45, 0, pSprite->x, pSprite->y, pSprite->z, pSprite->sectnum, pSprite->xrepeat, 0);
2021-10-15 19:10:16 +00:00
}
2019-09-25 21:16:12 +00:00
2021-10-15 19:10:16 +00:00
BulletList[nBullet].nFrame++;
if (BulletList[nBullet].nFrame >= SeqSize[nSeq])
{
if (!BulletList[nBullet].field_12)
{
BulletList[nBullet].nSeq = BulletInfo[BulletList[nBullet].nType].nSeq;
BulletList[nBullet].field_12++;
}
2019-09-25 21:16:12 +00:00
2021-10-15 19:10:16 +00:00
BulletList[nBullet].nFrame = 0;
}
2019-09-25 21:16:12 +00:00
2021-10-15 19:10:16 +00:00
if (BulletList[nBullet].field_E != -1 && --BulletList[nBullet].field_E == 0)
{
DestroyBullet(nBullet);
}
else
{
MoveBullet(nBullet);
}
}
2019-09-25 21:16:12 +00:00
2021-10-15 19:10:16 +00:00
void AIBullet::Draw(RunListEvent* ev)
{
short nBullet = RunData[ev->nRun].nObjIndex;
2021-10-15 19:10:16 +00:00
assert(nBullet >= 0 && nBullet < kMaxBullets);
2019-09-25 21:16:12 +00:00
2021-10-15 19:10:16 +00:00
short nSeq = SeqOffsets[BulletList[nBullet].nSeq];
2019-09-25 21:16:12 +00:00
short nSprite2 = ev->nParam;
2021-10-15 19:10:16 +00:00
mytsprite[nSprite2].statnum = 1000;
2019-09-25 21:16:12 +00:00
2021-10-15 19:10:16 +00:00
if (BulletList[nBullet].nType == 15)
{
seq_PlotArrowSequence(nSprite2, nSeq, BulletList[nBullet].nFrame);
2019-09-25 21:16:12 +00:00
}
2021-10-15 19:10:16 +00:00
else
{
seq_PlotSequence(nSprite2, nSeq, BulletList[nBullet].nFrame, 0);
ev->pTSprite->owner = -1;
}
}
void FuncBullet(int nObject, int nMessage, int nDamage, int nRun)
{
AIBullet ai;
runlist_DispatchEvent(&ai, nObject, nMessage, nDamage, nRun);
}
END_PS_NS