/* Emacs style mode select -*- C++ -*- *----------------------------------------------------------------------------- * * * PrBoom a Doom port merged with LxDoom and LSDLDoom * based on BOOM, a modified and improved DOOM engine * Copyright (C) 1999 by * id Software, Chi Hoang, Lee Killough, Jim Flynn, Rand Phares, Ty Halderman * Copyright (C) 1999-2000 by * Jess Haas, Nicolas Kalkhof, Colin Phipps, Florian Schulze * * This program 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. * * DESCRIPTION: * General plane mover and floor mover action routines * Floor motion, pure changer types, raising stairs. donuts, elevators * *-----------------------------------------------------------------------------*/ #include "doomstat.h" #include "r_main.h" #include "p_map.h" #include "p_spec.h" #include "p_tick.h" #include "s_sound.h" #include "sounds.h" #include "rockmacros.h" /////////////////////////////////////////////////////////////////////// // // Plane (floor or ceiling), Floor motion and Elevator action routines // /////////////////////////////////////////////////////////////////////// // // T_MovePlane() // // Move a plane (floor or ceiling) and check for crushing. Called // every tick by all actions that move floors or ceilings. // // Passed the sector to move a plane in, the speed to move it at, // the dest height it is to achieve, whether it crushes obstacles, // whether it moves a floor or ceiling, and the direction up or down // to move. // // Returns a result_e: // ok - plane moved normally, has not achieved destination yet // pastdest - plane moved normally and is now at destination height // crushed - plane encountered an obstacle, is holding until removed // result_e T_MovePlane ( sector_t* sector, fixed_t speed, fixed_t dest, boolean crush, int floorOrCeiling, int direction ) { boolean flag; fixed_t lastpos; fixed_t destheight; //jff 02/04/98 used to keep floors/ceilings // from moving thru each other switch(floorOrCeiling) { case 0: // Moving a floor switch(direction) { case -1: // Moving a floor down if (sector->floorheight - speed < dest) { lastpos = sector->floorheight; sector->floorheight = dest; flag = P_CheckSector(sector,crush); //jff 3/19/98 use faster chk if (flag == true) { sector->floorheight =lastpos; P_CheckSector(sector,crush); //jff 3/19/98 use faster chk } return pastdest; } else { lastpos = sector->floorheight; sector->floorheight -= speed; flag = P_CheckSector(sector,crush); //jff 3/19/98 use faster chk /* cph - make more compatible with original Doom, by * reintroducing this code. This means floors can't lower * if objects are stuck in the ceiling */ if ((flag == true) && comp[comp_floors]) { sector->floorheight = lastpos; P_ChangeSector(sector,crush); return crushed; } } break; case 1: // Moving a floor up // jff 02/04/98 keep floor from moving thru ceilings // jff 2/22/98 weaken check to demo_compatibility destheight = (comp[comp_floors] || destceilingheight)? dest : sector->ceilingheight; if (sector->floorheight + speed > destheight) { lastpos = sector->floorheight; sector->floorheight = destheight; flag = P_CheckSector(sector,crush); //jff 3/19/98 use faster chk if (flag == true) { sector->floorheight = lastpos; P_CheckSector(sector,crush); //jff 3/19/98 use faster chk } return pastdest; } else { // crushing is possible lastpos = sector->floorheight; sector->floorheight += speed; flag = P_CheckSector(sector,crush); //jff 3/19/98 use faster chk if (flag == true) { /* jff 1/25/98 fix floor crusher */ if (comp[comp_floors]) { if (crush == true) return crushed; } sector->floorheight = lastpos; P_CheckSector(sector,crush); //jff 3/19/98 use faster chk return crushed; } } break; } break; case 1: // moving a ceiling switch(direction) { case -1: // moving a ceiling down // jff 02/04/98 keep ceiling from moving thru floors // jff 2/22/98 weaken check to demo_compatibility destheight = (comp[comp_floors] || dest>sector->floorheight)? dest : sector->floorheight; if (sector->ceilingheight - speed < destheight) { lastpos = sector->ceilingheight; sector->ceilingheight = destheight; flag = P_CheckSector(sector,crush); //jff 3/19/98 use faster chk if (flag == true) { sector->ceilingheight = lastpos; P_CheckSector(sector,crush); //jff 3/19/98 use faster chk } return pastdest; } else { // crushing is possible lastpos = sector->ceilingheight; sector->ceilingheight -= speed; flag = P_CheckSector(sector,crush); //jff 3/19/98 use faster chk if (flag == true) { if (crush == true) return crushed; sector->ceilingheight = lastpos; P_CheckSector(sector,crush); //jff 3/19/98 use faster chk return crushed; } } break; case 1: // moving a ceiling up if (sector->ceilingheight + speed > dest) { lastpos = sector->ceilingheight; sector->ceilingheight = dest; flag = P_CheckSector(sector,crush); //jff 3/19/98 use faster chk if (flag == true) { sector->ceilingheight = lastpos; P_CheckSector(sector,crush); //jff 3/19/98 use faster chk } return pastdest; } else { lastpos = sector->ceilingheight; sector->ceilingheight += speed; flag = P_CheckSector(sector,crush); //jff 3/19/98 use faster chk } break; } break; } return ok; } // // T_MoveFloor() // // Move a floor to it's destination (up or down). // Called once per tick for each moving floor. // // Passed a floormove_t structure that contains all pertinent info about the // move. See P_SPEC.H for fields. // No return. // // jff 02/08/98 all cases with labels beginning with gen added to support // generalized line type behaviors. void T_MoveFloor(floormove_t* floor) { result_e res; res = T_MovePlane // move the floor ( floor->sector, floor->speed, floor->floordestheight, floor->crush, 0, floor->direction ); if (!(leveltime&7)) // make the floormove sound S_StartSound((mobj_t *)&floor->sector->soundorg, sfx_stnmov); if (res == pastdest) // if destination height is reached { if (floor->direction == 1) // going up { switch(floor->type) // handle texture/type changes { case donutRaise: floor->sector->special = floor->newspecial; floor->sector->floorpic = floor->texture; break; case genFloorChgT: case genFloorChg0: floor->sector->special = floor->newspecial; //jff add to fix bug in special transfers from changes floor->sector->oldspecial = floor->oldspecial; //fall thru case genFloorChg: floor->sector->floorpic = floor->texture; break; default: break; } } else if (floor->direction == -1) // going down { switch(floor->type) // handle texture/type changes { case lowerAndChange: floor->sector->special = floor->newspecial; //jff add to fix bug in special transfers from changes floor->sector->oldspecial = floor->oldspecial; floor->sector->floorpic = floor->texture; break; case genFloorChgT: case genFloorChg0: floor->sector->special = floor->newspecial; //jff add to fix bug in special transfers from changes floor->sector->oldspecial = floor->oldspecial; //fall thru case genFloorChg: floor->sector->floorpic = floor->texture; break; default: break; } } floor->sector->floordata = NULL; //jff 2/22/98 P_RemoveThinker(&floor->thinker);//remove this floor from list of movers //jff 2/26/98 implement stair retrigger lockout while still building // note this only applies to the retriggerable generalized stairs if (floor->sector->stairlock==-2) // if this sector is stairlocked { sector_t *sec = floor->sector; sec->stairlock=-1; // thinker done, promote lock to -1 while (sec->prevsec!=-1 && sectors[sec->prevsec].stairlock!=-2) sec = §ors[sec->prevsec]; // search for a non-done thinker if (sec->prevsec==-1) // if all thinkers previous are done { sec = floor->sector; // search forward while (sec->nextsec!=-1 && sectors[sec->nextsec].stairlock!=-2) sec = §ors[sec->nextsec]; if (sec->nextsec==-1) // if all thinkers ahead are done too { while (sec->prevsec!=-1) // clear all locks { sec->stairlock = 0; sec = §ors[sec->prevsec]; } sec->stairlock = 0; } } } // make floor stop sound S_StartSound((mobj_t *)&floor->sector->soundorg, sfx_pstop); } } // // T_MoveElevator() // // Move an elevator to it's destination (up or down) // Called once per tick for each moving floor. // // Passed an elevator_t structure that contains all pertinent info about the // move. See P_SPEC.H for fields. // No return. // // jff 02/22/98 added to support parallel floor/ceiling motion // void T_MoveElevator(elevator_t* elevator) { result_e res; if (elevator->direction<0) // moving down { res = T_MovePlane //jff 4/7/98 reverse order of ceiling/floor ( elevator->sector, elevator->speed, elevator->ceilingdestheight, 0, 1, // move floor elevator->direction ); if (res==ok || res==pastdest) // jff 4/7/98 don't move ceil if blocked T_MovePlane ( elevator->sector, elevator->speed, elevator->floordestheight, 0, 0, // move ceiling elevator->direction ); } else // up { res = T_MovePlane //jff 4/7/98 reverse order of ceiling/floor ( elevator->sector, elevator->speed, elevator->floordestheight, 0, 0, // move ceiling elevator->direction ); if (res==ok || res==pastdest) // jff 4/7/98 don't move floor if blocked T_MovePlane ( elevator->sector, elevator->speed, elevator->ceilingdestheight, 0, 1, // move floor elevator->direction ); } // make floor move sound if (!(leveltime&7)) S_StartSound((mobj_t *)&elevator->sector->soundorg, sfx_stnmov); if (res == pastdest) // if destination height acheived { elevator->sector->floordata = NULL; //jff 2/22/98 elevator->sector->ceilingdata = NULL; //jff 2/22/98 P_RemoveThinker(&elevator->thinker); // remove elevator from actives // make floor stop sound S_StartSound((mobj_t *)&elevator->sector->soundorg, sfx_pstop); } } /////////////////////////////////////////////////////////////////////// // // Floor motion linedef handlers // /////////////////////////////////////////////////////////////////////// // // EV_DoFloor() // // Handle regular and extended floor types // // Passed the line that activated the floor and the type of floor motion // Returns true if a thinker was created. // int EV_DoFloor ( line_t* line, floor_e floortype ) { int secnum; int rtn; int i; sector_t* sec; floormove_t* floor; secnum = -1; rtn = 0; // move all floors with the same tag as the linedef while ((secnum = P_FindSectorFromLineTag(line,secnum)) >= 0) { sec = §ors[secnum]; // Don't start a second thinker on the same floor if (P_SectorActive(floor_special,sec)) //jff 2/23/98 continue; // new floor thinker rtn = 1; floor = Z_Malloc (sizeof(*floor), PU_LEVSPEC, 0); P_AddThinker (&floor->thinker); sec->floordata = floor; //jff 2/22/98 floor->thinker.function = T_MoveFloor; floor->type = floortype; floor->crush = false; // setup the thinker according to the linedef type switch(floortype) { case lowerFloor: floor->direction = -1; floor->sector = sec; floor->speed = FLOORSPEED; floor->floordestheight = P_FindHighestFloorSurrounding(sec); break; //jff 02/03/30 support lowering floor by 24 absolute case lowerFloor24: floor->direction = -1; floor->sector = sec; floor->speed = FLOORSPEED; floor->floordestheight = floor->sector->floorheight + 24 * FRACUNIT; break; //jff 02/03/30 support lowering floor by 32 absolute (fast) case lowerFloor32Turbo: floor->direction = -1; floor->sector = sec; floor->speed = FLOORSPEED*4; floor->floordestheight = floor->sector->floorheight + 32 * FRACUNIT; break; case lowerFloorToLowest: floor->direction = -1; floor->sector = sec; floor->speed = FLOORSPEED; floor->floordestheight = P_FindLowestFloorSurrounding(sec); break; //jff 02/03/30 support lowering floor to next lowest floor case lowerFloorToNearest: floor->direction = -1; floor->sector = sec; floor->speed = FLOORSPEED; floor->floordestheight = P_FindNextLowestFloor(sec,floor->sector->floorheight); break; case turboLower: floor->direction = -1; floor->sector = sec; floor->speed = FLOORSPEED * 4; floor->floordestheight = P_FindHighestFloorSurrounding(sec); if (floor->floordestheight != sec->floorheight) floor->floordestheight += 8*FRACUNIT; break; case raiseFloorCrush: floor->crush = true; case raiseFloor: floor->direction = 1; floor->sector = sec; floor->speed = FLOORSPEED; floor->floordestheight = P_FindLowestCeilingSurrounding(sec); if (floor->floordestheight > sec->ceilingheight) floor->floordestheight = sec->ceilingheight; floor->floordestheight -= (8*FRACUNIT)*(floortype == raiseFloorCrush); break; case raiseFloorTurbo: floor->direction = 1; floor->sector = sec; floor->speed = FLOORSPEED*4; floor->floordestheight = P_FindNextHighestFloor(sec,sec->floorheight); break; case raiseFloorToNearest: floor->direction = 1; floor->sector = sec; floor->speed = FLOORSPEED; floor->floordestheight = P_FindNextHighestFloor(sec,sec->floorheight); break; case raiseFloor24: floor->direction = 1; floor->sector = sec; floor->speed = FLOORSPEED; floor->floordestheight = floor->sector->floorheight + 24 * FRACUNIT; break; // jff 2/03/30 support straight raise by 32 (fast) case raiseFloor32Turbo: floor->direction = 1; floor->sector = sec; floor->speed = FLOORSPEED*4; floor->floordestheight = floor->sector->floorheight + 32 * FRACUNIT; break; case raiseFloor512: floor->direction = 1; floor->sector = sec; floor->speed = FLOORSPEED; floor->floordestheight = floor->sector->floorheight + 512 * FRACUNIT; break; case raiseFloor24AndChange: floor->direction = 1; floor->sector = sec; floor->speed = FLOORSPEED; floor->floordestheight = floor->sector->floorheight + 24 * FRACUNIT; sec->floorpic = line->frontsector->floorpic; sec->special = line->frontsector->special; //jff 3/14/98 transfer both old and new special sec->oldspecial = line->frontsector->oldspecial; break; case raiseToTexture: { int minsize = INT_MAX; side_t* side; /* jff 3/13/98 no ovf */ if (!comp[comp_model]) minsize = 32000<direction = 1; floor->sector = sec; floor->speed = FLOORSPEED; for (i = 0; i < sec->linecount; i++) { if (twoSided (secnum, i) ) { side = getSide(secnum,i,0); // jff 8/14/98 don't scan texture 0, its not real if (side->bottomtexture > 0 || (comp[comp_model] && !side->bottomtexture)) if (textureheight[side->bottomtexture] < minsize) minsize = textureheight[side->bottomtexture]; side = getSide(secnum,i,1); // jff 8/14/98 don't scan texture 0, its not real if (side->bottomtexture > 0 || (comp[comp_model] && !side->bottomtexture)) if (textureheight[side->bottomtexture] < minsize) minsize = textureheight[side->bottomtexture]; } } if (comp[comp_model]) floor->floordestheight = floor->sector->floorheight + minsize; else { floor->floordestheight = (floor->sector->floorheight>>FRACBITS) + (minsize>>FRACBITS); if (floor->floordestheight>32000) floor->floordestheight = 32000; //jff 3/13/98 do not floor->floordestheight<<=FRACBITS; // allow height overflow } } break; case lowerAndChange: floor->direction = -1; floor->sector = sec; floor->speed = FLOORSPEED; floor->floordestheight = P_FindLowestFloorSurrounding(sec); floor->texture = sec->floorpic; // jff 1/24/98 make sure floor->newspecial gets initialized // in case no surrounding sector is at floordestheight // --> should not affect compatibility <-- floor->newspecial = sec->special; //jff 3/14/98 transfer both old and new special floor->oldspecial = sec->oldspecial; //jff 5/23/98 use model subroutine to unify fixes and handling sec = P_FindModelFloorSector(floor->floordestheight,sec-sectors); if (sec) { floor->texture = sec->floorpic; floor->newspecial = sec->special; //jff 3/14/98 transfer both old and new special floor->oldspecial = sec->oldspecial; } break; default: break; } } return rtn; } // // EV_DoChange() // // Handle pure change types. These change floor texture and sector type // by trigger or numeric model without moving the floor. // // The linedef causing the change and the type of change is passed // Returns true if any sector changes // // jff 3/15/98 added to better support generalized sector types // int EV_DoChange ( line_t* line, change_e changetype ) { int secnum; int rtn; sector_t* sec; sector_t* secm; secnum = -1; rtn = 0; // change all sectors with the same tag as the linedef while ((secnum = P_FindSectorFromLineTag(line,secnum)) >= 0) { sec = §ors[secnum]; rtn = 1; // handle trigger or numeric change type switch(changetype) { case trigChangeOnly: sec->floorpic = line->frontsector->floorpic; sec->special = line->frontsector->special; sec->oldspecial = line->frontsector->oldspecial; break; case numChangeOnly: secm = P_FindModelFloorSector(sec->floorheight,secnum); if (secm) // if no model, no change { sec->floorpic = secm->floorpic; sec->special = secm->special; sec->oldspecial = secm->oldspecial; } break; default: break; } } return rtn; } /* * EV_BuildStairs() * * Handles staircase building. A sequence of sectors chosen by algorithm * rise at a speed indicated to a height that increases by the stepsize * each step. * * Passed the linedef triggering the stairs and the type of stair rise * Returns true if any thinkers are created * * cph 2001/09/21 - compatibility nightmares again * There are three different ways this function has, during its history, stepped * through all the stairs to be triggered by the single switch * - original Doom used a linear P_FindSectorFromLineTag, but failed to preserve * the index of the previous sector found, so instead it would restart its * linear search from the last sector of the previous staircase * - MBF/PrBoom with comp_stairs fail to emulate this, because their * P_FindSectorFromLineTag is a chained hash table implementation. Instead they * start following the hash chain from the last sector of the previous * staircase, which will (probably) have the wrong tag, so they miss any further * stairs * - Boom fixed the bug, and MBF/PrBoom without comp_stairs work right */ static inline int P_FindSectorFromLineTagWithLowerBound (line_t* l, int start, int min) { /* Emulate original Doom's linear lower-bounded P_FindSectorFromLineTag * as needed */ do { start = P_FindSectorFromLineTag(l,start); } while (start >= 0 && start <= min); return start; } int EV_BuildStairs ( line_t* line, stair_e type ) { /* cph 2001/09/22 - cleaned up this function to save my sanity. A separate * outer loop index makes the logic much cleared, and local variables moved * into the inner blocks helps too */ int ssec = -1; int minssec = -1; int rtn = 0; // start a stair at each sector tagged the same as the linedef while ((ssec = P_FindSectorFromLineTagWithLowerBound(line,ssec,minssec)) >= 0) { int secnum = ssec; sector_t* sec = §ors[secnum]; // don't start a stair if the first step's floor is already moving if (!P_SectorActive(floor_special,sec)) { //jff 2/22/98 floormove_t* floor; int texture, height; fixed_t stairsize; fixed_t speed; int ok; // create new floor thinker for first step rtn = 1; floor = Z_Malloc (sizeof(*floor), PU_LEVSPEC, 0); P_AddThinker (&floor->thinker); sec->floordata = floor; floor->thinker.function = T_MoveFloor; floor->direction = 1; floor->sector = sec; floor->type = buildStair; //jff 3/31/98 do not leave uninited // set up the speed and stepsize according to the stairs type switch(type) { default: // killough -- prevent compiler warning case build8: speed = FLOORSPEED/4; stairsize = 8*FRACUNIT; if (!demo_compatibility) floor->crush = false; //jff 2/27/98 fix uninitialized crush field break; case turbo16: speed = FLOORSPEED*4; stairsize = 16*FRACUNIT; if (!demo_compatibility) floor->crush = true; //jff 2/27/98 fix uninitialized crush field break; } floor->speed = speed; height = sec->floorheight + stairsize; floor->floordestheight = height; texture = sec->floorpic; // Find next sector to raise // 1. Find 2-sided line with same sector side[0] (lowest numbered) // 2. Other side is the next sector to raise // 3. Unless already moving, or different texture, then stop building do { int i; ok = 0; for (i = 0;i < sec->linecount;i++) { sector_t* tsec = (sec->lines[i])->frontsector; int newsecnum; if ( !((sec->lines[i])->flags & ML_TWOSIDED) ) continue; newsecnum = tsec-sectors; if (secnum != newsecnum) continue; tsec = (sec->lines[i])->backsector; if (!tsec) continue; //jff 5/7/98 if no backside, continue newsecnum = tsec - sectors; // if sector's floor is different texture, look for another if (tsec->floorpic != texture) continue; /* jff 6/19/98 prevent double stepsize * killough 10/98: intentionally left this way [MBF comment] * cph 2001/02/06: stair bug fix should be controlled by comp_stairs, * except if we're emulating MBF which perversly reverted the fix */ if (comp[comp_stairs] || (compatibility_level == mbf_compatibility)) height += stairsize; // jff 6/28/98 change demo compatibility // if sector's floor already moving, look for another if (P_SectorActive(floor_special,tsec)) //jff 2/22/98 continue; /* cph - see comment above - do this iff we didn't do so above */ if (!comp[comp_stairs] && (compatibility_level != mbf_compatibility)) height += stairsize; sec = tsec; secnum = newsecnum; // create and initialize a thinker for the next step floor = Z_Malloc (sizeof(*floor), PU_LEVSPEC, 0); P_AddThinker (&floor->thinker); sec->floordata = floor; //jff 2/22/98 floor->thinker.function = T_MoveFloor; floor->direction = 1; floor->sector = sec; floor->speed = speed; floor->floordestheight = height; floor->type = buildStair; //jff 3/31/98 do not leave uninited //jff 2/27/98 fix uninitialized crush field if (!demo_compatibility) floor->crush = type==build8? false : true; ok = 1; break; } } while(ok); // continue until no next step is found } /* killough 10/98: compatibility option */ if (comp[comp_stairs]) { /* cph 2001/09/22 - emulate buggy MBF comp_stairs for demos, with logic * reversed since we now have a separate outer loop index. * DEMOSYNC - what about boom_compatibility_compatibility? */ if ((compatibility_level >= mbf_compatibility) && (compatibility_level < prboom_3_compatibility)) ssec = secnum; /* Trash outer loop index */ else { /* cph 2001/09/22 - now the correct comp_stairs - Doom used a linear * search from the last secnum, so we set that as a minimum value and do * a fresh tag search */ ssec = -1; minssec = secnum; } } } return rtn; } // // EV_DoDonut() // // Handle donut function: lower pillar, raise surrounding pool, both to height, // texture and type of the sector surrounding the pool. // // Passed the linedef that triggered the donut // Returns whether a thinker was created // int EV_DoDonut(line_t* line) { sector_t* s1; sector_t* s2; sector_t* s3; int secnum; int rtn; int i; floormove_t* floor; secnum = -1; rtn = 0; // do function on all sectors with same tag as linedef while ((secnum = P_FindSectorFromLineTag(line,secnum)) >= 0) { s1 = §ors[secnum]; // s1 is pillar's sector // do not start the donut if the pillar is already moving if (P_SectorActive(floor_special,s1)) //jff 2/22/98 continue; s2 = getNextSector(s1->lines[0],s1); // s2 is pool's sector if (!s2) continue; // note lowest numbered line around // pillar must be two-sided /* do not start the donut if the pool is already moving * cph - DEMOSYNC - was !compatibility */ if (!comp[comp_floors] && P_SectorActive(floor_special,s2)) continue; //jff 5/7/98 // find a two sided line around the pool whose other side isn't the pillar for (i = 0;i < s2->linecount;i++) { //jff 3/29/98 use true two-sidedness, not the flag // killough 4/5/98: changed demo_compatibility to compatibility if (comp[comp_model]) { if (!(s2->lines[i]->flags & ML_TWOSIDED) || (s2->lines[i]->backsector == s1)) continue; } else if (!s2->lines[i]->backsector || s2->lines[i]->backsector == s1) continue; rtn = 1; //jff 1/26/98 no donut action - no switch change on return s3 = s2->lines[i]->backsector; // s3 is model sector for changes // Spawn rising slime floor = Z_Malloc (sizeof(*floor), PU_LEVSPEC, 0); P_AddThinker (&floor->thinker); s2->floordata = floor; //jff 2/22/98 floor->thinker.function = T_MoveFloor; floor->type = donutRaise; floor->crush = false; floor->direction = 1; floor->sector = s2; floor->speed = FLOORSPEED / 2; floor->texture = s3->floorpic; floor->newspecial = 0; floor->floordestheight = s3->floorheight; // Spawn lowering donut-hole pillar floor = Z_Malloc (sizeof(*floor), PU_LEVSPEC, 0); P_AddThinker (&floor->thinker); s1->floordata = floor; //jff 2/22/98 floor->thinker.function = T_MoveFloor; floor->type = lowerFloor; floor->crush = false; floor->direction = -1; floor->sector = s1; floor->speed = FLOORSPEED / 2; floor->floordestheight = s3->floorheight; break; } } return rtn; } // // EV_DoElevator // // Handle elevator linedef types // // Passed the linedef that triggered the elevator and the elevator action // // jff 2/22/98 new type to move floor and ceiling in parallel // int EV_DoElevator ( line_t* line, elevator_e elevtype ) { int secnum; int rtn; sector_t* sec; elevator_t* elevator; secnum = -1; rtn = 0; // act on all sectors with the same tag as the triggering linedef while ((secnum = P_FindSectorFromLineTag(line,secnum)) >= 0) { sec = §ors[secnum]; // If either floor or ceiling is already activated, skip it if (sec->floordata || sec->ceilingdata) //jff 2/22/98 continue; // create and initialize new elevator thinker rtn = 1; elevator = Z_Malloc (sizeof(*elevator), PU_LEVSPEC, 0); P_AddThinker (&elevator->thinker); sec->floordata = elevator; //jff 2/22/98 sec->ceilingdata = elevator; //jff 2/22/98 elevator->thinker.function = T_MoveElevator; elevator->type = elevtype; // set up the fields according to the type of elevator action switch(elevtype) { // elevator down to next floor case elevateDown: elevator->direction = -1; elevator->sector = sec; elevator->speed = ELEVATORSPEED; elevator->floordestheight = P_FindNextLowestFloor(sec,sec->floorheight); elevator->ceilingdestheight = elevator->floordestheight + sec->ceilingheight - sec->floorheight; break; // elevator up to next floor case elevateUp: elevator->direction = 1; elevator->sector = sec; elevator->speed = ELEVATORSPEED; elevator->floordestheight = P_FindNextHighestFloor(sec,sec->floorheight); elevator->ceilingdestheight = elevator->floordestheight + sec->ceilingheight - sec->floorheight; break; // elevator to floor height of activating switch's front sector case elevateCurrent: elevator->sector = sec; elevator->speed = ELEVATORSPEED; elevator->floordestheight = line->frontsector->floorheight; elevator->ceilingdestheight = elevator->floordestheight + sec->ceilingheight - sec->floorheight; elevator->direction = elevator->floordestheight>sec->floorheight? 1 : -1; break; default: break; } } return rtn; }