//------------------------------------------------------------------------- /* Copyright (C) 1996, 2003 - 3D Realms Entertainment Copyright (C) 2000, 2003 - Matt Saettler (EDuke Enhancements) Copyright (C) 2020 - Christoph Oelckers This file is part of Enhanced Duke Nukem 3D version 1.5 - Atomic Edition Duke Nukem 3D is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. Original Source: 1996 - Todd Replogle Prepared for public release: 03/21/2003 - Charlie Wiederhold, 3D Realms EDuke enhancements integrated: 04/13/2003 - Matt Saettler Note: EDuke source was in transition. Changes are in-progress in the source as it is released. */ //------------------------------------------------------------------------- #include #include "ns.h" #include "global.h" #include "sounds.h" #include "dukeactor.h" #include "interpolate.h" // PRIMITIVE BEGIN_DUKE_NS static int interptype[] = { Interp_Sect_Floorz, Interp_Sect_Ceilingz, Interp_Wall_X, Interp_Wall_Y }; //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- bool ceilingspace(sectortype* sectp) { return (sectp && (sectp->ceilingstat & CSTAT_SECTOR_SKY) && sectp->ceilingpal == 0 && (tileflags(sectp->ceilingpicnum) & TFLAG_OUTERSPACE)); } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- bool floorspace(sectortype* sectp) { return (sectp && (sectp->floorstat & CSTAT_SECTOR_SKY) && sectp->floorpal == 0 && (tileflags(sectp->floorpicnum) & TFLAG_OUTERSPACE)); } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- static bool haltsoundhack; int callsound(sectortype* sn, DDukeActor* whatsprite, bool endstate) { if (!isRRRA() && haltsoundhack) { haltsoundhack = 0; return -1; } DukeSectIterator it(sn); while (auto act = it.Next()) { if (issoundcontroller(act) && act->spr.lotag < 1000) { if (whatsprite == nullptr) whatsprite = act; int snum = act->spr.lotag; auto flags = S_GetUserFlags(snum); // Reset if the desired actor isn't playing anything. bool hival = S_IsSoundValid(act->spr.hitag); if (act->temp_data[0] == 1 && !hival && !endstate) { if (!S_CheckActorSoundPlaying(act->temp_actor, snum)) act->temp_data[0] = 0; } if (act->temp_data[0] == 0) { if ((flags & (SF_GLOBAL | SF_DTAG)) != SF_GLOBAL) { if (snum) { if (act->spr.hitag && snum != act->spr.hitag) S_StopSound(act->spr.hitag, act->temp_actor); S_PlayActorSound(snum, whatsprite); act->temp_actor = whatsprite; } if ((act->sector()->lotag & 0xff) != ST_22_SPLITTING_DOOR) act->temp_data[0] = 1; } } else if (act->spr.hitag < 1000) { // The original code performed these two actions in reverse order which in case of a looped sound being stopped // being the same as the sound about to be started, the newly started sound would fall through some cracks in the sound system and be rejected. // Here this case needs to be simulated. bool stopped = false; if ((flags & SF_LOOP) || (act->spr.hitag && act->spr.hitag != act->spr.lotag)) { S_StopSound(act->spr.lotag, act->temp_actor); if (act->spr.hitag == act->spr.lotag) stopped = true; } if (act->spr.hitag && !stopped) S_PlayActorSound(act->spr.hitag, whatsprite); act->temp_data[0] = 0; act->temp_actor = whatsprite; } return act->spr.lotag; } } return -1; } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- int check_activator_motion(int lotag) { DukeStatIterator it(STAT_ACTIVATOR); while (auto act = it.Next()) { if (act->spr.lotag == lotag) { for (int j = animates.Size() - 1; j >= 0; j--) if (act->sector() == animates[j].sect) return(1); DukeStatIterator it1(STAT_EFFECTOR); while (auto act2 = it1.Next()) { if (act->sector() == act2->sector()) switch (act2->spr.lotag) { case SE_11_SWINGING_DOOR: case SE_30_TWO_WAY_TRAIN: if (act2->temp_data[4]) return(1); break; case SE_18_INCREMENTAL_SECTOR_RISE_FALL: if (isRRRA()) break; [[fallthrough]]; case SE_20_STRETCH_BRIDGE: case SE_31_FLOOR_RISE_FALL: case SE_32_CEILING_RISE_FALL: if (act2->temp_data[0]) return(1); break; } } } } return(0); } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- bool isanunderoperator(int lotag) { switch (lotag & 0xff) { case ST_15_WARP_ELEVATOR: case ST_16_PLATFORM_DOWN: case ST_17_PLATFORM_UP: case ST_18_ELEVATOR_DOWN: case ST_19_ELEVATOR_UP: case ST_26_SPLITTING_ST_DOOR: return true; case ST_22_SPLITTING_DOOR: return !isRR(); } return false; } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- bool isanearoperator(int lotag) { switch (lotag & 0xff) { case ST_9_SLIDING_ST_DOOR: case ST_15_WARP_ELEVATOR: case ST_16_PLATFORM_DOWN: case ST_17_PLATFORM_UP: case ST_18_ELEVATOR_DOWN: case ST_19_ELEVATOR_UP: case ST_20_CEILING_DOOR: case ST_21_FLOOR_DOOR: case ST_22_SPLITTING_DOOR: case ST_23_SWINGING_DOOR: case ST_25_SLIDING_DOOR: case ST_26_SPLITTING_ST_DOOR: case ST_29_TEETH_DOOR: return true; case 41: return isRR(); } return false; } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- int findplayer(const DDukeActor* actor, double* d) { int j, closest_player; const auto s = actor->spr.pos; if (ud.multimode < 2) { if (d) *d = (ps[myconnectindex].GetActor()->getPrevPosWithOffsetZ() - s).plusZ(28).Sum(); return myconnectindex; } double closest = 0x7fffffff; closest_player = 0; for (j = connecthead; j >= 0; j = connectpoint2[j]) { double x = (ps[j].GetActor()->getPrevPosWithOffsetZ() - s).plusZ(28).Sum(); if (x < closest && ps[j].GetActor()->spr.extra > 0) { closest_player = j; closest = x; } } if (d) *d = closest; return closest_player; } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- int findotherplayer(int p, double* d) { int j, closest_player; double closest = 0x7fffffff; closest_player = p; for (j = connecthead; j >= 0; j = connectpoint2[j]) if (p != j && ps[j].GetActor()->spr.extra > 0) { double x = (ps[j].GetActor()->getPrevPosWithOffsetZ() - ps[p].GetActor()->getPosWithOffsetZ()).Sum(); if (x < closest) { closest_player = j; closest = x; } } *d = closest; return closest_player; } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- double getanimatevalue(int type, int index) { switch (type) { case anim_floorz: return sector[index].floorz; case anim_ceilingz: return sector[index].ceilingz; case anim_vertexx: return wall[index].pos.X; case anim_vertexy: return wall[index].pos.Y; default: assert(false); return 0; } } double getanimatevalue(int i) { return getanimatevalue(animates[i].type, animates[i].target); } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- void setanimatevalue(int type, int index, double value) { static int scratch; switch (type) { case anim_floorz: sector[index].setfloorz(value); break; case anim_ceilingz: sector[index].setceilingz(value); break; case anim_vertexx: wall[index].pos.X = value; wall[index].moved(); break; case anim_vertexy: wall[index].pos.Y = value; wall[index].moved(); break; default: assert(false); } } void setanimatevalue(int i, double value) { return setanimatevalue(animates[i].type, animates[i].target, value); } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- void doanimations(void) { for (int i = animates.Size() - 1; i >= 0; i--) { double a = getanimatevalue(i); double const v = animates[i].vel * TICSPERFRAME; auto dasectp = animates[i].sect; int type = animates[i].type; if (a == animates[i].goal) { StopInterpolation(animates[i].target, interptype[animates[i].type]); animates[i] = animates.Last(); animates.Pop(); if (dasectp->lotag == ST_18_ELEVATOR_DOWN || dasectp->lotag == ST_19_ELEVATOR_UP) if (type == anim_ceilingz) continue; if ((dasectp->lotag & 0xff) != ST_22_SPLITTING_DOOR) callsound(dasectp, nullptr, true); continue; } if (v > 0) { a = min(a + v, animates[i].goal); } else { a = max(a + v, animates[i].goal); } if (type == anim_floorz) { for (auto p = connecthead; p >= 0; p = connectpoint2[p]) if (ps[p].cursector == dasectp) if ((dasectp->floorz - ps[p].GetActor()->getOffsetZ()) < 64) if (ps[p].GetActor()->GetOwner() != nullptr) { ps[p].GetActor()->spr.pos.Z += v; ps[p].vel.Z = 0; } DukeSectIterator it(dasectp); while (auto act = it.Next()) { if (act->spr.statnum != STAT_EFFECTOR) { if (act->spr.statnum != STAT_PLAYER) { act->backupz(); act->spr.pos.Z += v; } act->floorz = dasectp->floorz + v; } } } setanimatevalue(i, a); } } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- int getanimationindex(int animtype, sectortype* animtargetp) { int i, j; j = -1; int animtarget = sectindex(animtargetp); for (i = animates.Size() - 1; i >= 0; i--) if (animtype == animates[i].type && animtarget == animates[i].target) { j = i; break; } return(j); } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- static int dosetanimation(sectortype* animsect, int animtype, int animtarget, double thegoal, double thevel) { int j = -1; for (unsigned i = 0; i < animates.Size(); i++) if (animtype == animates[i].type && animtarget == animates[i].target) { j = i; break; } if (j == -1) j = animates.Reserve(1); auto animval = getanimatevalue(animtype, animtarget); animates[j].sect = animsect; animates[j].type = animtype; animates[j].target = animtarget; animates[j].goal = thegoal; if (thegoal >= animval) animates[j].vel = thevel; else animates[j].vel = -thevel; StartInterpolation(animates[j].target, interptype[animates[j].type]); return(j); } int setanimation(sectortype* animsect, int animtype, walltype* animtarget, double thegoal, double thevel) { assert(animtype == anim_vertexx || animtype == anim_vertexy); return dosetanimation(animsect, animtype, wallindex(animtarget), thegoal, thevel); } int setanimation(sectortype* animsect, int animtype, sectortype* animtarget, double thegoal, double thevel) { assert(animtype == anim_ceilingz || animtype == anim_floorz); return dosetanimation(animsect, animtype, sectindex(animtarget), thegoal, thevel); } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- bool activatewarpelevators(DDukeActor* actor, int d) //Parm = sectoreffectornum { auto sect = actor->sector(); // See if the sector exists DukeStatIterator it(STAT_EFFECTOR); DDukeActor *act2; while ((act2 = it.Next())) { if (act2->spr.lotag == SE_17_WARP_ELEVATOR || (isRRRA() && act2->spr.lotag == SE_18_INCREMENTAL_SECTOR_RISE_FALL)) if (act2->spr.hitag == actor->spr.hitag) if ((abs(sect->floorz - actor->temp_pos.X) > act2->spr.yint * maptoworld) || (act2->sector()->hitag == (sect->hitag - d))) break; } if (act2 == nullptr) { d = 0; return 1; // No find } else { if (d == 0) S_PlayActorSound(ELEVATOR_OFF, actor); else S_PlayActorSound(ELEVATOR_ON, actor); } it.Reset(STAT_EFFECTOR); while ((act2 = it.Next())) { if (act2->spr.lotag == SE_17_WARP_ELEVATOR || (isRRRA() && act2->spr.lotag == SE_18_INCREMENTAL_SECTOR_RISE_FALL)) if (act2->spr.hitag == actor->spr.hitag) { act2->temp_data[0] = d; if (act2->spr.lotag == SE_17_WARP_ELEVATOR) act2->temp_data[1] = d; //Make all check warp (only SE17, in SE18 this is a coordinate) } } return 0; } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- static void handle_st09(sectortype* sptr, DDukeActor* actor) { double dax, day, dax2, day2, sp; walltype* wallfind[2]; sp = (sptr->extra >> 4) / 16.; //first find center point by averaging all points dax = 0, day = 0; for (auto& wal : sptr->walls) { dax += wal.pos.X; day += wal.pos.Y; } dax /= sptr->walls.Size(); day /= sptr->walls.Size(); //find any points with either same x or same y coordinate // as center (dax, day) - should be 2 points found. wallfind[0] = nullptr; wallfind[1] = nullptr; for (auto& wal : sptr->walls) // more precise checks won't work here. if (abs(wal.pos.X - dax) <= (1 / 32.) || abs(wal.pos.Y - day) <= (1 / 32.)) { if (wallfind[0] == nullptr) wallfind[0] = &wal; else wallfind[1] = &wal; } for (int j = 0; j < 2; j++) { auto wal = wallfind[j]; //find what direction door should open by averaging the // 2 neighboring points of wallfind[0] & wallfind[1]. auto prevwall = wal - 1; if (prevwall < sptr->walls.Data()) prevwall += sptr->walls.Size(); if ((wal->pos.X == dax) && (wal->pos.Y == day)) { dax2 = ((prevwall->pos.X + wal->point2Wall()->pos.X) * 0.5) - wal->pos.X; day2 = ((prevwall->pos.Y + wal->point2Wall()->pos.Y) * 0.5) - wal->pos.Y; if (dax2 != 0) { dax2 = wal->point2Wall()->point2Wall()->pos.X; dax2 -= wal->point2Wall()->pos.X; setanimation(sptr, anim_vertexx, wal, wal->pos.X + dax2, sp); setanimation(sptr, anim_vertexx, prevwall, prevwall->pos.X + dax2, sp); setanimation(sptr, anim_vertexx, wal->point2Wall(), wal->point2Wall()->pos.X + dax2, sp); callsound(sptr, actor); } else if (day2 != 0) { day2 = wal->point2Wall()->point2Wall()->pos.Y; day2 -= wal->point2Wall()->pos.Y; setanimation(sptr, anim_vertexy, wal, wal->pos.Y + day2, sp); setanimation(sptr, anim_vertexy, prevwall, prevwall->pos.Y + day2, sp); setanimation(sptr, anim_vertexy, wal->point2Wall(), wal->point2Wall()->pos.Y + day2, sp); callsound(sptr, actor); } } else { dax2 = ((prevwall->pos.X + wal->point2Wall()->pos.X) * 0.5) - wal->pos.X; day2 = ((prevwall->pos.Y + wal->point2Wall()->pos.Y) * 0.5) - wal->pos.Y; if (dax2 != 0) { setanimation(sptr, anim_vertexx, wal, dax, sp); setanimation(sptr, anim_vertexx, prevwall, dax + dax2, sp); setanimation(sptr, anim_vertexx, wal->point2Wall(), dax + dax2, sp); callsound(sptr, actor); } else if (day2 != 0) { setanimation(sptr, anim_vertexy, wal, day, sp); setanimation(sptr, anim_vertexy, prevwall, day + day2, sp); setanimation(sptr, anim_vertexy, wal->point2Wall(), day + day2, sp); callsound(sptr, actor); } } } } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- static void handle_st15(sectortype* sptr, DDukeActor* actor) { if (!actor->isPlayer()) return; DukeSectIterator it(sptr); DDukeActor* a2; while ((a2 = it.Next())) { if (iseffector(a2) && a2->spr.lotag == ST_17_PLATFORM_UP) break; } if (!a2) return; if (actor->sector() == sptr) { if (activatewarpelevators(a2, -1)) activatewarpelevators(a2, 1); else if (activatewarpelevators(a2, 1)) activatewarpelevators(a2, -1); return; } else { if (sptr->floorz > a2->spr.pos.Z) activatewarpelevators(a2, -1); else activatewarpelevators(a2, 1); } } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- static void handle_st16(sectortype* sptr, DDukeActor* actor) { int i = getanimationindex(anim_floorz, sptr); sectortype* sectp; if (i == -1) { sectp = nextsectorneighborzptr(sptr, sptr->floorz, Find_FloorDown); if (sectp == nullptr) { sectp = nextsectorneighborzptr(sptr, sptr->floorz, Find_FloorUp); if (sectp == nullptr) return; } setanimation(sptr, anim_floorz, sptr, sectp->floorz, sptr->extra / 256.); callsound(sptr, actor); } } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- static void handle_st18(sectortype* sptr, DDukeActor* actor) { int i = getanimationindex(anim_floorz, sptr); if (i == -1) { auto sectp = nextsectorneighborzptr(sptr, sptr->floorz, Find_FloorUp); if (sectp == nullptr) sectp = nextsectorneighborzptr(sptr, sptr->floorz, Find_FloorDown); if (sectp == nullptr) return; double speed = sptr->extra / 256.; setanimation(sptr, anim_floorz, sptr, sectp->floorz, speed); setanimation(sptr, anim_ceilingz, sptr, sectp->floorz + sptr->ceilingz - sptr->floorz, speed); callsound(sptr, actor); } } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- static void handle_st29(sectortype* sptr, DDukeActor* actor) { double j; if (sptr->lotag & 0x8000) j = nextsectorneighborzptr(sptr, sptr->ceilingz, Find_FloorDown | Find_Safe)->floorz; else j = nextsectorneighborzptr(sptr, sptr->ceilingz, Find_CeilingUp | Find_Safe)->ceilingz; DukeStatIterator it(STAT_EFFECTOR); while (auto act2 = it.Next()) { if ((act2->spr.lotag == 22) && (act2->spr.hitag == sptr->hitag)) { act2->sector()->extra = -act2->sector()->extra; act2->temp_data[0] = sectindex(sptr); act2->temp_data[1] = 1; } } sptr->lotag ^= 0x8000; setanimation(sptr, anim_ceilingz, sptr, j, sptr->extra / 256.); callsound(sptr, actor); } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- static void handle_st20(sectortype* sptr, DDukeActor* actor) { double j = 0; REDODOOR: if (sptr->lotag & 0x8000) { DDukeActor* a2; DukeSectIterator it(sptr); while ((a2 = it.Next())) { if (a2->spr.statnum == STAT_EFFECTOR && a2->spr.lotag == SE_9_DOWN_OPEN_DOOR_LIGHTS) { j = a2->spr.pos.Z; break; } } if (a2 == nullptr) j = sptr->floorz; } else { auto sectp = nextsectorneighborzptr(sptr, sptr->ceilingz, Find_CeilingUp); if (sectp) j = sectp->ceilingz; else { sptr->lotag |= 32768; goto REDODOOR; } } sptr->lotag ^= 0x8000; setanimation(sptr, anim_ceilingz, sptr, j, sptr->extra / 256.); callsound(sptr, actor); } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- static void handle_st21(sectortype* sptr, DDukeActor* actor) { int i = getanimationindex(anim_floorz, sptr); double j; if (i >= 0) { if (animates[i].goal == sptr->ceilingz) animates[i].goal = nextsectorneighborzptr(sptr, sptr->ceilingz, Find_FloorDown | Find_Safe)->floorz; else animates[i].goal = sptr->ceilingz; j = animates[i].goal; } else { if (sptr->ceilingz == sptr->floorz) j = nextsectorneighborzptr(sptr, sptr->ceilingz, Find_FloorDown | Find_Safe)->floorz; else j = sptr->ceilingz; sptr->lotag ^= 0x8000; if (setanimation(sptr, anim_floorz, sptr, j, sptr->extra / 256.) >= 0) callsound(sptr, actor); } } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- static void handle_st22(sectortype* sptr, DDukeActor* actor) { int j; double z; double speed = sptr->extra / 256.; if ((sptr->lotag & 0x8000)) { z = (sptr->ceilingz + sptr->floorz) * 0.5; j = setanimation(sptr, anim_floorz, sptr, z, speed); j = setanimation(sptr, anim_ceilingz, sptr, z, speed); } else { z = nextsectorneighborzptr(sptr, sptr->floorz, Find_FloorDown | Find_Safe)->floorz; j = setanimation(sptr, anim_floorz, sptr, z, speed); z = nextsectorneighborzptr(sptr, sptr->ceilingz, Find_CeilingUp | Find_Safe)->ceilingz; j = setanimation(sptr, anim_ceilingz, sptr, z, speed); } sptr->lotag ^= 0x8000; callsound(sptr, actor); } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- static void handle_st23(sectortype* sptr, DDukeActor* actor) { int q = 0; DukeStatIterator it(STAT_EFFECTOR); DDukeActor* act2; while ((act2 = it.Next())) { if (act2->spr.lotag == SE_11_SWINGING_DOOR && act2->sector() == sptr && !act2->temp_data[4]) { break; } } if (!act2) return; int l = act2->sector()->lotag & 0x8000; if (act2) { DukeStatIterator itr(STAT_EFFECTOR); while (auto act3 = itr.Next()) { if (l == (act3->sector()->lotag & 0x8000) && act3->spr.lotag == SE_11_SWINGING_DOOR && act2->spr.hitag == act3->spr.hitag && act3->temp_data[4]) { return; } } itr.Reset(STAT_EFFECTOR); while (auto act3 = itr.Next()) { if (l == (act3->sector()->lotag & 0x8000) && act3->spr.lotag == SE_11_SWINGING_DOOR && act2->spr.hitag == act3->spr.hitag) { if (act3->sector()->lotag & 0x8000) act3->sector()->lotag &= 0x7fff; else act3->sector()->lotag |= 0x8000; act3->temp_data[4] = 1; act3->temp_data[3] = -act3->temp_data[3]; if (q == 0) { callsound(sptr, act3); q = 1; } } } } } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- static void handle_st25(sectortype* sptr, DDukeActor* actor) { DukeStatIterator it(STAT_EFFECTOR); DDukeActor* act2; while ((act2 = it.Next())) { if (act2->spr.lotag == 15 && act2->sector() == sptr) { break; } } if (act2 == nullptr) return; it.Reset(STAT_EFFECTOR); while (auto act3 = it.Next()) { if (act3->spr.hitag == act2->spr.hitag) { if (act3->spr.lotag == 15) { act3->sector()->lotag ^= 0x8000; // Toggle the open or close act3->spr.Angles.Yaw += DAngle180; if (act3->temp_data[4]) callsound(act3->sector(), act3); callsound(act3->sector(), act3); if (act3->sector()->lotag & 0x8000) act3->temp_data[4] = 1; else act3->temp_data[4] = 2; } } } } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- static void handle_st27(sectortype* sptr, DDukeActor* actor) { DukeStatIterator it(STAT_EFFECTOR); while (auto act2 = it.Next()) { if ((act2->spr.lotag & 0xff) == 20 && act2->sector() == sptr) //Bridge { sptr->lotag ^= 0x8000; if (sptr->lotag & 0x8000) //OPENING act2->temp_data[0] = 1; else act2->temp_data[0] = 2; callsound(sptr, actor); break; } } } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- static void handle_st28(sectortype* sptr, DDukeActor* actor) { //activate the rest of them int j = -1; DukeSectIterator it(sptr); while (auto a2 = it.Next()) { if (a2->spr.statnum == 3 && (a2->spr.lotag & 0xff) == 21) { j = a2->spr.hitag; break; //Found it } } if (j == -1) return; DukeStatIterator it1(STAT_EFFECTOR); while (auto act3 = it.Next()) { if ((act3->spr.lotag & 0xff) == 21 && !act3->temp_data[0] && (act3->spr.hitag) == j) act3->temp_data[0] = 1; } callsound(sptr, actor); } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- void operatesectors(sectortype* sptr, DDukeActor *actor) { int j=0; int i; switch (sptr->lotag & (0x3fff)) { case 41: if (isRR()) operatejaildoors(sptr->hitag); break; case 7: if (!isRR()) break; for (auto& wal : sptr->walls) { setanimation(sptr, anim_vertexx, &wal, wal.pos.X + 64, 1 / 4.); if (wal.twoSided()) setanimation(sptr, anim_vertexx, wal.nextWall(), wal.nextWall()->pos.X + 64, 1 / 4.); } break; case ST_30_ROTATE_RISE_BRIDGE: { auto act = barrier_cast(sptr->hitagactor); if (!act) break; if (act->tempval == 0 || act->tempval == 256) callsound(sptr, actor); if (act->spr.extra == 1) act->spr.extra = 3; else act->spr.extra = 1; break; } case ST_31_TWO_WAY_TRAIN: { auto act = barrier_cast(sptr->hitagactor); if (!act) break; if (act->temp_data[4] == 0) act->temp_data[4] = 1; callsound(sptr, actor); break; } case ST_26_SPLITTING_ST_DOOR: //The split doors i = getanimationindex(anim_ceilingz, sptr); if (i == -1) //if the door has stopped { haltsoundhack = 1; sptr->lotag &= 0xff00; sptr->lotag |= ST_22_SPLITTING_DOOR; operatesectors(sptr, actor); sptr->lotag &= 0xff00; sptr->lotag |= ST_9_SLIDING_ST_DOOR; operatesectors(sptr, actor); sptr->lotag &= 0xff00; sptr->lotag |= ST_26_SPLITTING_ST_DOOR; } return; case ST_9_SLIDING_ST_DOOR: handle_st09(sptr, actor); return; case ST_15_WARP_ELEVATOR://Warping elevators handle_st15(sptr, actor); return; case ST_16_PLATFORM_DOWN: case ST_17_PLATFORM_UP: handle_st16(sptr, actor); return; case ST_18_ELEVATOR_DOWN: case ST_19_ELEVATOR_UP: handle_st18(sptr, actor); return; case ST_29_TEETH_DOOR: handle_st29(sptr, actor); return; case ST_20_CEILING_DOOR: handle_st20(sptr, actor); return; case ST_21_FLOOR_DOOR: handle_st21(sptr, actor); return; case ST_22_SPLITTING_DOOR: handle_st22(sptr, actor); return; case ST_23_SWINGING_DOOR: //Swingdoor handle_st23(sptr, actor); return; case ST_25_SLIDING_DOOR: //Subway type sliding doors { handle_st25(sptr, actor); return; } case ST_27_STRETCH_BRIDGE: //Extended bridge handle_st27(sptr, actor); return; case ST_28_DROP_FLOOR: handle_st28(sptr, actor); return; } } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- void operateactivators(int low, player_struct* plr) { int i, j, k; Cycler * p; walltype* wal; for (i = cyclers.Size() - 1; i >= 0; i--) { p = &cyclers[i]; if (p->hitag == low) { auto sect = p->sector; p->state = !p->state; sect->floorshade = sect->ceilingshade = (int8_t)p->shade2; wal = sect->walls.Data(); for (j = sect->walls.Size(); j > 0; j--, wal++) wal->shade = (int8_t)p->shade2; } } k = -1; DukeStatIterator it(STAT_ACTIVATOR); while (auto act = it.Next()) { if (act->spr.lotag == low) { if (islockedactivator(act)) { act->sector()->lotag ^= 16384; if (plr) { if (act->sector()->lotag & 16384) FTA(4, plr); else FTA(8, plr); } } else { switch (act->spr.hitag) { case 0: break; case 1: if (act->sector()->floorz != act->sector()->ceilingz) { continue; } break; case 2: if (act->sector()->floorz == act->sector()->ceilingz) { continue; } break; } if (act->sector()->lotag < 3) { DukeSectIterator itr(act->sector()); while (auto a2 = itr.Next()) { if (a2->spr.statnum == 3) switch (a2->spr.lotag) { case SE_18_INCREMENTAL_SECTOR_RISE_FALL: if (isRRRA()) break; [[fallthrough]]; case SE_36_PROJ_SHOOTER: case SE_31_FLOOR_RISE_FALL: case SE_32_CEILING_RISE_FALL: a2->temp_data[0] = 1 - a2->temp_data[0]; callsound(act->sector(), a2); break; } } } if (k == -1 && (act->sector()->lotag & 0xff) == SE_22_TEETH_DOOR) k = callsound(act->sector(), act); operatesectors(act->sector(), act); } } } operaterespawns(low); } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- void operatemasterswitches(int low) { DukeStatIterator it(STAT_STANDABLE); while (auto act2 = it.Next()) { if (ismasterswitch(act2) && act2->spr.lotag == low && act2->spr.yint == 0) { act2->spr.yint = 1; } } } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- void operateforcefields_common(DDukeActor *effector, int low, const std::initializer_list &tiles) { for (int p = numanimwalls-1; p >= 0; p--) { auto wal = animwall[p].wall; if (low == wal->lotag || low == -1) if (isIn(wal->overpicnum, tiles)) { animwall[p].tag = 0; if (wal->cstat) { wal->cstat = 0; if (effector && iseffector(effector) && effector->spr.lotag == SE_30_TWO_WAY_TRAIN) wal->lotag = 0; } else wal->cstat = (CSTAT_WALL_BLOCK | CSTAT_WALL_ALIGN_BOTTOM | CSTAT_WALL_MASKED | CSTAT_WALL_BLOCK_HITSCAN); } } } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- void checkhitwall(DDukeActor* spr, walltype* wal, const DVector3& pos) { if (wal->overpicnum == TILE_MIRROR && actorflag(spr, SFLAG2_BREAKMIRRORS)) { lotsofglass(spr, wal, 70); wal->cstat &= ~CSTAT_WALL_MASKED; wal->overpicnum = TILE_MIRRORBROKE; wal->portalflags = 0; S_PlayActorSound(GLASS_HEAVYBREAK, spr); return; } auto handler = [=](const BreakWallRec* data, int16_t* pic) { if (!data->handler) { *pic = data->brokentex; S_PlayActorSound(data->breaksound, spr); } else { VMValue args[7] = { wal, data->brokentex, data->breaksound.index(), spr, pos.X, pos.Y, pos.Z }; VMCall(data->handler, args, 7, nullptr, 0); } }; if (wal->twoSided() && wal->nextSector()->floorz > pos.Z && wal->nextSector()->floorz - wal->nextSector()->ceilingz) { auto data = breakWallMap.CheckKey(wal->overpicnum); if (data && (data->flags & 1) && (!(data->flags & 2) || wal->cstat & CSTAT_WALL_MASKED)) { handler(data, &wal->overpicnum); } } auto data = breakWallMap.CheckKey(wal->picnum); if (data && !(data->flags & 1)) { handler(data, &wal->picnum); } } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- bool checkhitceiling(sectortype* sectp) { auto data = breakCeilingMap.CheckKey(sectp->ceilingpicnum); if (data && !(data->flags & 1)) { if (!data->handler) { sectp->ceilingpicnum = data->brokentex; S_PlayActorSound(data->breaksound, ps[screenpeek].GetActor()); // this is nonsense but what the original code did. } else { VMValue args[7] = { sectp, data->brokentex, data->breaksound.index() }; VMCall(data->handler, args, 3, nullptr, 0); } if (data->flags & 1) { if (!sectp->hitag) { DukeSectIterator it(sectp); while (auto act = it.Next()) { if (iseffector(act) && act->spr.lotag == SE_12_LIGHT_SWITCH) { DukeStatIterator it1(STAT_EFFECTOR); while (auto act2 = it1.Next()) { if (act2->spr.hitag == act->spr.hitag) act2->temp_data[3] = 1; } break; } } } int j = krand() & 1; DukeStatIterator it(STAT_EFFECTOR); while (auto act = it.Next()) { if (act->spr.hitag == (sectp->hitag) && act->spr.lotag == SE_3_RANDOM_LIGHTS_AFTER_SHOT_OUT) { act->temp_data[2] = j; act->temp_data[4] = 1; } } } if (data->flags & 2) { ceilingglass(ps[myconnectindex].GetActor(), sectp, 10); } return true; } return false; } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- void allignwarpelevators(void) { DukeStatIterator it(STAT_EFFECTOR); while (auto act = it.Next()) { if (act->spr.lotag == SE_17_WARP_ELEVATOR && act->spr.shade > 16) { DukeStatIterator it1(STAT_EFFECTOR); while (auto act2 = it1.Next()) { if ((act2->spr.lotag) == SE_17_WARP_ELEVATOR && act != act2 && act->spr.hitag == act2->spr.hitag) { act2->sector()->setfloorz(act->sector()->floorz); act2->sector()->setceilingz(act->sector()->ceilingz); } } } } } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- void moveclouds(double interpfrac) { // The math here is very messy.. :( int myclock = interpfrac < 0.5 ? PlayClock-2 : PlayClock; if (myclock > cloudclock || myclock < (cloudclock - 7)) { cloudclock = myclock + 6; // cloudx/y were an array, but all entries were always having the same value so a single pair is enough. cloudx += (float)ps[screenpeek].GetActor()->spr.Angles.Yaw.Cos() * 0.5f; cloudy += (float)ps[screenpeek].GetActor()->spr.Angles.Yaw.Sin() * 0.5f; for (int i = 0; i < numclouds; i++) { // no clamping here! clouds[i]->ceilingxpan_ = cloudx; clouds[i]->ceilingypan_ = cloudy; clouds[i]->exflags |= SECTOREX_CLOUDSCROLL; } } } END_DUKE_NS