- Player Control type fixes

- Custom Dude type fixes
This commit is contained in:
NoOneBlood 2019-12-23 23:10:28 +03:00 committed by Christoph Oelckers
parent ec782a28de
commit 5bcfcc1b55
5 changed files with 73 additions and 59 deletions

View file

@ -2557,9 +2557,6 @@ short gSightSpritesCount; // current count
short gPhysSpritesList[]; // by NoOne: list of additional sprites which can be affected by physics short gPhysSpritesList[]; // by NoOne: list of additional sprites which can be affected by physics
short gPhysSpritesCount; // current count short gPhysSpritesCount; // current count
short gQavPlayerIndex = -1; // by NoOne: index of sprite which currently activated to play qav
void actInit(bool bSaveLoad) { void actInit(bool bSaveLoad) {
// by NoOne: init code for all my stuff // by NoOne: init code for all my stuff
@ -2590,6 +2587,8 @@ void actInit(bool bSaveLoad) {
// add in list of physics affected sprites // add in list of physics affected sprites
if (pXSprite->physAttr != 0) { if (pXSprite->physAttr != 0) {
//xvel[pSprite->index] = yvel[pSprite->index] = zvel[pSprite->index] = 0;
gPhysSpritesList[gPhysSpritesCount++] = pSprite->index; // add sprite index gPhysSpritesList[gPhysSpritesCount++] = pSprite->index; // add sprite index
getSpriteMassBySize(pSprite); // create mass cache getSpriteMassBySize(pSprite); // create mass cache
} }
@ -2634,7 +2633,7 @@ void actInit(bool bSaveLoad) {
else { else {
gProxySpritesList[gProxySpritesCount++] = pSprite->xvel; gProxySpritesList[gProxySpritesCount++] = pSprite->xvel;
if (gProxySpritesCount == kMaxSuperXSprites) if (gProxySpritesCount == kMaxSuperXSprites)
viewSetSystemMessage("Max (%d) *additional* Proximity sprites reached!", kMaxSuperXSprites); ThrowError("Max (%d) *additional* Proximity sprites reached!", kMaxSuperXSprites);
} }
break; break;
} }
@ -2658,7 +2657,7 @@ void actInit(bool bSaveLoad) {
default: default:
gSightSpritesList[gSightSpritesCount++] = pSprite->xvel; gSightSpritesList[gSightSpritesCount++] = pSprite->xvel;
if (gSightSpritesCount == kMaxSuperXSprites) if (gSightSpritesCount == kMaxSuperXSprites)
viewSetSystemMessage("Max (%d) Sight sprites reached!", kMaxSuperXSprites); ThrowError("Max (%d) Sight sprites reached!", kMaxSuperXSprites);
break; break;
} }
} }
@ -2691,7 +2690,11 @@ void actInit(bool bSaveLoad) {
int nType = pSprite->type - kThingBase; int nType = pSprite->type - kThingBase;
pXSprite->health = thingInfo[nType].startHealth << 4; pXSprite->health = thingInfo[nType].startHealth << 4;
pSprite->clipdist = thingInfo[nType].clipdist; // by NoOne: allow level designer to set custom clipdist.
// this is especially useful for various Gib and Explode objects which have clipdist 1 for some reason predefined,
// but what if it have voxel model...?
if (!gModernMap) pSprite->clipdist = thingInfo[nType].clipdist;
pSprite->flags = thingInfo[nType].flags; pSprite->flags = thingInfo[nType].flags;
if (pSprite->flags & kPhysGravity) pSprite->flags |= kPhysFalling; if (pSprite->flags & kPhysGravity) pSprite->flags |= kPhysFalling;
xvel[nSprite] = yvel[nSprite] = zvel[nSprite] = 0; xvel[nSprite] = yvel[nSprite] = zvel[nSprite] = 0;
@ -4259,16 +4262,17 @@ void ProcessTouchObjects(spritetype *pSprite, int nXSprite)
break; break;
case kDudeModernCustom: case kDudeModernCustom:
case kDudeModernCustomBurning: case kDudeModernCustomBurning:
int dmg = (getSpriteMassBySize(pSprite2) - getSpriteMassBySize(pSprite)) + pSprite2->clipdist; int dmg = 0;
if (dmg > 0) { if (!IsDudeSprite(pSprite) || (dmg = ClipLow((getSpriteMassBySize(pSprite2) - getSpriteMassBySize(pSprite)) >> 1, 0)) == 0)
if (IsPlayerSprite(pSprite) && powerupCheck(&gPlayer[pSprite->type - kDudePlayer1], kPwUpJumpBoots) > 0) break;
actDamageSprite(pSprite2->xvel, pSprite, DAMAGE_TYPE_3, dmg);
else
actDamageSprite(pSprite2->xvel, pSprite, DAMAGE_TYPE_0, dmg);
}
if (!IsPlayerSprite(pSprite) && pSprite2->extra >= 0 && !isActive(pSprite2->xvel)) if (!IsPlayerSprite(pSprite)) {
aiActivateDude(pSprite2, &xsprite[pSprite2->extra]); actDamageSprite(pSprite2->index, pSprite, DAMAGE_TYPE_0, dmg);
if (xspriRangeIsFine(pSprite->extra) && !isActive(pSprite->index))
aiActivateDude(pSprite, &xsprite[pSprite->extra]);
}
else if (powerupCheck(&gPlayer[pSprite->type - kDudePlayer1], kPwUpJumpBoots) > 0) actDamageSprite(pSprite2->index, pSprite, DAMAGE_TYPE_3, dmg);
else actDamageSprite(pSprite2->index, pSprite, DAMAGE_TYPE_0, dmg);
break; break;
} }
@ -4483,7 +4487,7 @@ void ProcessTouchObjects(spritetype *pSprite, int nXSprite)
} }
// by NoOne: add more trigger statements for Touch flag // by NoOne: add more trigger statements for Touch flag
if (gModernMap) { if (gModernMap && IsDudeSprite(pSprite)) {
// Touch sprites // Touch sprites
int nHSprite = -1; int nHSprite = -1;
@ -4494,7 +4498,7 @@ void ProcessTouchObjects(spritetype *pSprite, int nXSprite)
else if ((gSpriteHit[nXSprite].ceilhit & 0xc000) == 0xc000) else if ((gSpriteHit[nXSprite].ceilhit & 0xc000) == 0xc000)
nHSprite = gSpriteHit[nXSprite].ceilhit & 0x3fff; nHSprite = gSpriteHit[nXSprite].ceilhit & 0x3fff;
if (nHSprite >= 0 && sprite[nHSprite].extra >= 0) { if (spriRangeIsFine(nHSprite) && xspriRangeIsFine(sprite[nHSprite].extra)) {
XSPRITE* pXHSprite = &xsprite[sprite[nHSprite].extra]; XSPRITE* pXHSprite = &xsprite[sprite[nHSprite].extra];
if (pXHSprite->Touch && !pXHSprite->isTriggered && (!pXHSprite->DudeLockout || IsPlayerSprite(pSprite))) if (pXHSprite->Touch && !pXHSprite->isTriggered && (!pXHSprite->DudeLockout || IsPlayerSprite(pSprite)))
trTriggerSprite(nHSprite, pXHSprite, kCmdSpriteTouch, nSprite); trTriggerSprite(nHSprite, pXHSprite, kCmdSpriteTouch, nSprite);
@ -4503,12 +4507,14 @@ void ProcessTouchObjects(spritetype *pSprite, int nXSprite)
// Touch walls // Touch walls
int nHWall = -1; int nHWall = -1;
if ((gSpriteHit[nXSprite].hit & 0xc000) == 0x8000) { if ((gSpriteHit[nXSprite].hit & 0xc000) == 0x8000) {
if ((nHWall = gSpriteHit[nXSprite].hit & 0x3fff) >= 0 && wall[nHWall].extra >= 0) { nHWall = gSpriteHit[nXSprite].hit & 0x3fff;
if (wallRangeIsFine(nHWall) && xwallRangeIsFine(wall[nHWall].extra)) {
XWALL* pXHWall = &xwall[wall[nHWall].extra]; XWALL* pXHWall = &xwall[wall[nHWall].extra];
if (pXHWall->triggerTouch && !pXHWall->isTriggered && (!pXHWall->dudeLockout || IsPlayerSprite(pSprite))) if (pXHWall->triggerTouch && !pXHWall->isTriggered && (!pXHWall->dudeLockout || IsPlayerSprite(pSprite)))
trTriggerWall(nHWall, pXHWall, kCmdWallTouch, nSprite); trTriggerWall(nHWall, pXHWall, kCmdWallTouch, nSprite);
} }
} }
} }
} }
@ -4788,14 +4794,8 @@ void MoveDude(spritetype *pSprite)
gHitInfo = hitInfo; gHitInfo = hitInfo;
} }
if (pHitXSprite && pHitXSprite->Touch && !pHitXSprite->isTriggered) { if (!gModernMap && pHitXSprite && pHitXSprite->Touch && !pHitXSprite->state && !pHitXSprite->isTriggered)
// by NoOne: do not check state (so, things can work with touch too) and allow dudelockout
if ((gModernMap) && (!pHitXSprite->DudeLockout || IsPlayerSprite(pSprite)))
trTriggerSprite(nHitSprite, pHitXSprite, kCmdSpriteTouch, nSprite); trTriggerSprite(nHitSprite, pHitXSprite, kCmdSpriteTouch, nSprite);
else if (!pHitXSprite->state) // or check like vanilla do
trTriggerSprite(nHitSprite, pHitXSprite, kCmdSpriteTouch, nSprite);
}
if (pDudeInfo->lockOut && pHitXSprite && pHitXSprite->Push && !pHitXSprite->key && !pHitXSprite->DudeLockout && !pHitXSprite->state && !pHitXSprite->busy && !pPlayer) if (pDudeInfo->lockOut && pHitXSprite && pHitXSprite->Push && !pHitXSprite->key && !pHitXSprite->DudeLockout && !pHitXSprite->state && !pHitXSprite->busy && !pPlayer)
trTriggerSprite(nHitSprite, pHitXSprite, kCmdSpritePush, nSprite); trTriggerSprite(nHitSprite, pHitXSprite, kCmdSpritePush, nSprite);
@ -5663,10 +5663,9 @@ void actProcessSprites(void)
// by NoOne: process Debris sprites for movement // by NoOne: process Debris sprites for movement
if (gPhysSpritesCount > 0) { if (gPhysSpritesCount > 0) {
//System.err.println("PHYS COUNT: "+gPhysSpritesCount); //viewSetSystemMessage("PHYS COUNT: %d", gPhysSpritesCount);
for (int i = 0; i < gPhysSpritesCount; i++) { for (int i = 0; i < gPhysSpritesCount; i++) {
if (gPhysSpritesList[i] == -1) continue; if (gPhysSpritesList[i] == -1) continue;
else if (sprite[gPhysSpritesList[i]].statnum == kStatFree || (sprite[gPhysSpritesList[i]].flags & kHitagFree) != 0) { else if (sprite[gPhysSpritesList[i]].statnum == kStatFree || (sprite[gPhysSpritesList[i]].flags & kHitagFree) != 0) {
gPhysSpritesList[i] = -1; gPhysSpritesList[i] = -1;
continue; continue;
@ -5680,7 +5679,6 @@ void actProcessSprites(void)
spritetype* pDebris = &sprite[gPhysSpritesList[i]]; spritetype* pDebris = &sprite[gPhysSpritesList[i]];
XSECTOR* pXSector = (sector[pDebris->sectnum].extra >= 0) ? &xsector[sector[pDebris->sectnum].extra] : NULL; XSECTOR* pXSector = (sector[pDebris->sectnum].extra >= 0) ? &xsector[sector[pDebris->sectnum].extra] : NULL;
viewBackupSpriteLoc(pDebris->xvel, pDebris); viewBackupSpriteLoc(pDebris->xvel, pDebris);
int airVel = gSpriteMass[pDebris->extra].airVel; int airVel = gSpriteMass[pDebris->extra].airVel;
if (pXSector != NULL) { if (pXSector != NULL) {

View file

@ -188,7 +188,8 @@ void genDudeProcess(spritetype* pSprite, XSPRITE* pXSprite) {
if (pXSprite->stateTimer == 0 && pXSprite->aiState->nextState && (pXSprite->aiState->stateTicks > 0 || seqGetStatus(3, pSprite->extra) < 0)) if (pXSprite->stateTimer == 0 && pXSprite->aiState->nextState && (pXSprite->aiState->stateTicks > 0 || seqGetStatus(3, pSprite->extra) < 0))
aiGenDudeNewState(pSprite, pXSprite->aiState->nextState); aiGenDudeNewState(pSprite, pXSprite->aiState->nextState);
if (pXSprite->health > 0 && ((dudeInfo[pSprite->type - kDudeBase].hinderDamage << 4) <= cumulDamage[pSprite->extra])) { int hinder = ((pExtra->isMelee) ? 25 : 5) << 4;
if (pXSprite->health > 0 && hinder <= cumulDamage[pSprite->extra]) {
pXSprite->data3 = cumulDamage[pSprite->extra]; pXSprite->data3 = cumulDamage[pSprite->extra];
RecoilDude(pSprite, pXSprite); RecoilDude(pSprite, pXSprite);
} }

View file

@ -1560,8 +1560,8 @@ DUDEINFO dudeInfo[kDudeMax-kDudeBase] =
618, // melee distance 618, // melee distance
5, // flee health 5, // flee health
5, // hinder damage 5, // hinder damage
0x0100, // change target chance 0x0000, // change target chance
0x0010, // change target to kin chance 0x0000, // change target to kin chance
0x8000, // alertChance 0x8000, // alertChance
0, // lockout 0, // lockout
46603, // frontSpeed 46603, // frontSpeed

View file

@ -589,16 +589,24 @@ void deactivateSizeShrooms(PLAYER* pPlayer) {
PLAYER* getPlayerById(short id) { PLAYER* getPlayerById(short id) {
if (id > 0) {
// relative to connected players
if (id >= 1 && id <= kMaxPlayers) {
id = id - 1;
for (int i = connecthead; i >= 0; i = connectpoint2[i]) { for (int i = connecthead; i >= 0; i = connectpoint2[i]) {
if (id < kMaxPlayers && id == i + 1) return &gPlayer[i]; // relative to connected players if (id == gPlayer[i].nPlayer)
else if (id >= kDudePlayer1 && id <= kDudePlayer8 && id == gPlayer[i].pSprite->type) // absolute type
return &gPlayer[i]; return &gPlayer[i];
} }
if (id >= kDudePlayer1 && id <= kDudePlayer8) viewSetSystemMessage("There is no player #%d", (kDudePlayer8 - id) + kMaxPlayers); // absolute sprite type
else viewSetSystemMessage("There is no player #%d", id); } else if (id >= kDudePlayer1 && id <= kDudePlayer8) {
for (int i = connecthead; i >= 0; i = connectpoint2[i]) {
if (id == gPlayer[i].pSprite->type)
return &gPlayer[i];
} }
}
viewSetSystemMessage("There is no player id #%d", id);
return NULL; return NULL;
} }

View file

@ -440,6 +440,7 @@ void OperateSprite(int nSprite, XSPRITE *pXSprite, EVENT event)
// Random Event Switch takes random data field and uses it as TX ID // Random Event Switch takes random data field and uses it as TX ID
case kModernRandomTX: { case kModernRandomTX: {
int tx = 0; int maxRetries = 10; int tx = 0; int maxRetries = 10;
// set range of TX ID if data2 and data3 is empty. // set range of TX ID if data2 and data3 is empty.
if (pXSprite->data1 > 0 && pXSprite->data2 <= 0 && pXSprite->data3 <= 0 && pXSprite->data4 > 0) { if (pXSprite->data1 > 0 && pXSprite->data2 <= 0 && pXSprite->data3 <= 0 && pXSprite->data4 > 0) {
@ -472,7 +473,6 @@ void OperateSprite(int nSprite, XSPRITE *pXSprite, EVENT event)
SetSpriteState(nSprite, pXSprite, pXSprite->state ^ 1, event.causedBy); SetSpriteState(nSprite, pXSprite, pXSprite->state ^ 1, event.causedBy);
} }
return; return;
// Sequential Switch takes values from data fields starting from data1 and uses it as TX ID // Sequential Switch takes values from data fields starting from data1 and uses it as TX ID
case kModernSequentialTX: { case kModernSequentialTX: {
bool range = false; int cnt = 3; int tx = 0; bool range = false; int cnt = 3; int tx = 0;
@ -866,10 +866,14 @@ void OperateSprite(int nSprite, XSPRITE *pXSprite, EVENT event)
if (pCtrl->qavScene.index != nSprite || pXSprite->Interrutable) if (pCtrl->qavScene.index != nSprite || pXSprite->Interrutable)
trPlayerCtrlStartScene(pXSprite, pPlayer, event.causedBy); trPlayerCtrlStartScene(pXSprite, pPlayer, event.causedBy);
return; return;
case kCmdNumberic + 4: // stop playing qav scene case kCmdNumberic + 4: { // stop playing qav scene
if (pCtrl->qavScene.index == nSprite || event.type != 3 || sprite[event.index].type != kModernPlayerControl) int scnIndex = pCtrl->qavScene.index;
if (spriRangeIsFine(scnIndex) && (scnIndex == nSprite || event.type != 3 || sprite[event.index].type != kModernPlayerControl)) {
if (scnIndex != nSprite) pXSprite = &xsprite[sprite[scnIndex].extra];
trPlayerCtrlStopScene(pXSprite, pPlayer); trPlayerCtrlStopScene(pXSprite, pPlayer);
}
return; return;
}
default: default:
oldCmd = pXSprite->command; oldCmd = pXSprite->command;
pXSprite->command = event.cmd; // convert event command to current sprite command pXSprite->command = event.cmd; // convert event command to current sprite command
@ -993,8 +997,8 @@ void OperateSprite(int nSprite, XSPRITE *pXSprite, EVENT event)
// angle // angle
// TO-DO: if tx > 0, take a look on TX ID sprite // TO-DO: if tx > 0, take a look on TX ID sprite
if (pXSprite->data3 == 1) pPlayer->pSprite->ang = pXSprite->data3; if (pXSprite->data3 == 1) pPlayer->pSprite->ang = pSprite->ang;
else if (valueIsBetween(pXSprite->data3, 0, kAng180)) else if (valueIsBetween(pXSprite->data3, 1, kAng180))
pPlayer->pSprite->ang = pXSprite->data3; pPlayer->pSprite->ang = pXSprite->data3;
} }
//viewSetSystemMessage("ANGLE: %d, SLOPE: %d", pPlayer->pSprite->ang, pPlayer->q16look); //viewSetSystemMessage("ANGLE: %d, SLOPE: %d", pPlayer->pSprite->ang, pPlayer->q16look);
@ -1070,7 +1074,9 @@ void OperateSprite(int nSprite, XSPRITE *pXSprite, EVENT event)
pPlayer->hasWeapon[pXSprite->data3] = true; pPlayer->hasWeapon[pXSprite->data3] = true;
if (pXSprite->data4 == 0) { // switch on it if (pXSprite->data4 == 0) { // switch on it
if (pPlayer->sceneQav >= 0) { pPlayer->nextWeapon = 0;
if (pPlayer->sceneQav >= 0 && spriRangeIsFine(pCtrl->qavScene.index)) {
XSPRITE* pXScene = &xsprite[sprite[pCtrl->qavScene.index].extra]; XSPRITE* pXScene = &xsprite[sprite[pCtrl->qavScene.index].extra];
pXScene->dropMsg = pXSprite->data3; pXScene->dropMsg = pXSprite->data3;
} else if (pPlayer->curWeapon != pXSprite->data3) { } else if (pPlayer->curWeapon != pXSprite->data3) {
@ -2017,8 +2023,7 @@ void useSectorWindGen(XSPRITE* pXSource, sectortype* pSector) {
void useSpriteDamager(XSPRITE* pXSource, spritetype* pSprite) { void useSpriteDamager(XSPRITE* pXSource, spritetype* pSprite) {
spritetype* pSource = &sprite[pXSource->reference]; spritetype* pSource = &sprite[pXSource->reference];
if (pSprite != NULL && xspriRangeIsFine(pSprite->extra)) { if (pSprite != NULL && xspriRangeIsFine(pSprite->extra) && xsprite[pSprite->extra].health > 0) {
XSPRITE* pXSprite = &xsprite[pSprite->extra];
DAMAGE_TYPE dmgType = (DAMAGE_TYPE)ClipRange(pXSource->data2, kDmgFall, kDmgElectric); DAMAGE_TYPE dmgType = (DAMAGE_TYPE)ClipRange(pXSource->data2, kDmgFall, kDmgElectric);
int dmg = (pXSource->data3 == 0) ? 65535 : ClipRange(pXSource->data3 << 1, 1, 65535); int dmg = (pXSource->data3 == 0) ? 65535 : ClipRange(pXSource->data3 << 1, 1, 65535);
if (pXSource->data2 >= 0) actDamageSprite(pSource->index, pSprite, dmgType, dmg); if (pXSource->data2 >= 0) actDamageSprite(pSource->index, pSprite, dmgType, dmg);
@ -2026,8 +2031,10 @@ void useSpriteDamager(XSPRITE* pXSource, spritetype* pSprite) {
PLAYER* pPlayer = getPlayerById(pSprite->type); PLAYER* pPlayer = getPlayerById(pSprite->type);
if (pPlayer == NULL || !pPlayer->godMode) { if (pPlayer == NULL || !pPlayer->godMode) {
xsprite[pSprite->extra].health = ClipLow(xsprite[pSprite->extra].health - dmg, 0); xsprite[pSprite->extra].health = ClipLow(xsprite[pSprite->extra].health - dmg, 0);
if (xsprite[pSprite->extra].health == 0) if (xsprite[pSprite->extra].health == 0) {
actKillDude(pSource->index, pSprite, DAMAGE_TYPE_0, 65535); if (pPlayer == NULL) actKillDude(pSource->index, pSprite, DAMAGE_TYPE_0, 4);
else playerDamageSprite(pSource->index, pPlayer, DAMAGE_TYPE_0, 4);
}
} }
} }
} }
@ -3450,18 +3457,18 @@ void pastePropertiesInObj(int type, int nDest, EVENT event) {
if (event.type != 3) return; if (event.type != 3) return;
spritetype* pSource = &sprite[event.index]; spritetype* pSource = &sprite[event.index];
if (pSource->extra < 0) return; if (!xspriRangeIsFine(pSource->extra)) return;
XSPRITE* pXSource = &xsprite[pSource->extra]; XSPRITE* pXSource = &xsprite[pSource->extra];
switch (type) { switch (type) {
case 6: case 6:
if (sector[nDest].extra < 0) return; if (!xsectRangeIsFine(sector[nDest].extra)) return;
break; break;
case 0: case 0:
if (wall[nDest].extra < 0) return; if (!xwallRangeIsFine(wall[nDest].extra)) return;
break; break;
case 3: case 3:
if (sprite[nDest].extra < 0) return; if (!xspriRangeIsFine(sprite[nDest].extra)) return;
break; break;
default: default:
return; return;
@ -3475,7 +3482,7 @@ void pastePropertiesInObj(int type, int nDest, EVENT event) {
} else if (pSource->type == kModernSpriteDamager) { } else if (pSource->type == kModernSpriteDamager) {
/* - damages xsprite via TX ID - */ /* - damages xsprite via TX ID - */
if (type != 3) return; if (type != 3) return;
else if (xsprite[sprite[nDest].extra].health > 0) useSpriteDamager(pXSource, &sprite[nDest]); useSpriteDamager(pXSource, &sprite[nDest]);
return; return;
} if (pSource->type == kModernEffectSpawner) { } if (pSource->type == kModernEffectSpawner) {