//------------------------------------------------------------------------- /* Copyright (C) 1996, 2003 - 3D Realms Entertainment Copyright (C) 2017-2019 Nuke.YKT Copyright (C) 2020 - Christoph Oelckers This file is part of 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 */ //------------------------------------------------------------------------- #include "ns.h" #include "global.h" #include "mapinfo.h" #include "secrets.h" #include "statistics.h" #include "gamestate.h" #include "sbar.h" #include "automap.h" #include "dukeactor.h" #include "interpolate.h" BEGIN_DUKE_NS int which_palookup = 9; //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- void pickrandomspot(int snum) { struct player_struct *p; short i; p = &ps[snum]; if( ud.multimode > 1 && ud.coop != 1) i = krand()%numplayersprites; else i = snum; p->bobposx = p->oposx = p->posx = po[i].ox; p->bobposy = p->oposy = p->posy = po[i].oy; p->oposz = p->posz = po[i].oz; p->angle.oang = p->angle.ang = buildang(po[i].oa); p->cursectnum = po[i].os; } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- void resetplayerstats(int snum) { struct player_struct *p; p = &ps[snum]; gFullMap = 0; p->dead_flag = 0; p->resurrected = false; p->wackedbyactor = nullptr; p->falling_counter = 0; p->quick_kick = 0; p->subweapon = 0; p->last_full_weapon = 0; p->ftq = 0; p->tipincs = 0; p->buttonpalette = 0; p->actorsqu =nullptr; p->invdisptime = 0; p->refresh_inventory= 0; p->last_pissed_time = 0; p->holster_weapon = 0; p->pycount = 0; p->pyoff = 0; p->opyoff = 0; p->loogcnt = 0; p->psectlotag = 0; p->weapon_sway = 0; // p->select_dir = 0; p->extra_extra8 = 0; p->show_empty_weapon= 0; p->dummyplayersprite=nullptr; p->crack_time = 0; p->hbomb_hold_delay = 0; p->transporter_hold = 0; p->wantweaponfire = -1; p->hurt_delay = 0; p->hurt_delay2 = 0; p->footprintcount = 0; p->footprintpal = 0; p->footprintshade = 0; p->jumping_toggle = 0; p->horizon.ohoriz = p->horizon.horiz = q16horiz(40); p->horizon.ohorizoff = p->horizon.horizoff = q16horiz(0); p->bobcounter = 0; p->on_ground = 0; p->player_par = 0; p->sync.actions |= SB_CENTERVIEW; p->airleft = 15*26; p->rapid_fire_hold = 0; p->toggle_key_flag = 0; p->access_spritenum = nullptr; if(ud.multimode > 1 && ud.coop != 1 ) p->got_access = 7; else p->got_access = 0; p->random_club_frame= 0; p->on_warping_sector = 0; p->spritebridge = 0; if(p->steroids_amount < 400 ) { p->steroids_amount = 0; p->inven_icon = 0; } p->heat_on = 0; p->jetpack_on = 0; p->holoduke_on = nullptr; p->angle.olook_ang = p->angle.look_ang = buildlook(512 - ((currentLevel->levelNumber & 1) << 10)); p->angle.orotscrnang = p->angle.rotscrnang = buildlook(0); p->newOwner =nullptr; p->jumping_counter = 0; p->hard_landing = 0; p->posxv = 0; //!! p->posyv = 0; p->poszv = 0; p->fric.x = 0; p->fric.y = 0; p->somethingonplayer =nullptr; p->angle.spin = bamlook(0); p->on_crane = nullptr; if(p->curr_weapon == PISTOL_WEAPON) p->okickback_pic = p->kickback_pic = isRR()? 22 : 5; else p->okickback_pic = p->kickback_pic = 0; p->oweapon_pos = p->weapon_pos = 6; p->walking_snd_toggle= 0; p->weapon_ang = 0; p->knuckle_incs = 1; p->fist_incs = 0; p->knee_incs = 0; p->stairs = 0; p->noise_x = 0; p->noise_y = 0; p->donoise = 0; p->noise_radius = 0; if (isRR() && ud.multimode > 1 && ud.coop != 1) { p->keys[0] = 1; p->keys[1] = 1; p->keys[2] = 1; p->keys[3] = 1; p->keys[4] = 1; } else { p->keys[0] = 0; p->keys[1] = 0; p->keys[2] = 0; p->keys[3] = 0; p->keys[4] = 0; } wupass = 0; //p->at582 = 0; p->drunkang = 1647; p->eatang = 1647; p->drink_amt = 0; p->eat = 0; p->drink_timer = 4096; p->eat_timer = 4096; p->shotgun_state[0] = 0; p->shotgun_state[1] = 0; p->detonate_time = 0; p->detonate_count = 0; p->recoil = 0; p->yehaa_timer = 0; chickenphase = 0; if (p->OnMotorcycle) { p->OnMotorcycle = 0; p->gotweapon.Clear(MOTORCYCLE_WEAPON); p->curr_weapon = isRRRA()? SLINGBLADE_WEAPON : KNEE_WEAPON; // just in case this is made available for the other games } p->lotag800kill = 0; p->moto_do_bump = 0; p->MotoOnGround = 1; p->moto_underwater = 0; p->MotoSpeed = 0; p->TiltStatus = 0; p->moto_drink = 0; p->VBumpTarget = 0; p->VBumpNow =0; p->moto_bump_fast = 0; p->TurbCount = 0; p->moto_on_mud = 0; p->moto_on_oil = 0; if (p->OnBoat) { p->OnBoat = 0; p->gotweapon.Clear(BOAT_WEAPON); p->curr_weapon = isRRRA()? SLINGBLADE_WEAPON : KNEE_WEAPON; // just in case this is made available for the other games } p->NotOnWater = 0; p->SeaSick = 0; p->nocheat = 0; p->DrugMode = 0; p->drug_stat[0] = 0; p->drug_stat[1] = 0; p->drug_stat[2] = 0; p->drug_aspect = 0; resetlanepics(); if (numplayers < 2) { ufospawn = isRRRA()? 3 : std::min(ud.player_skill*4+1, 32); ufocnt = 0; hulkspawn = ud.player_skill + 1; } else { ufospawn = isRRRA()? 0 :32; ufocnt = 0; hulkspawn = isRRRA()? 0 :2; } } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- void resetweapons(int snum) { short weapon; struct player_struct *p; p = &ps[snum]; for (weapon = PISTOL_WEAPON; weapon < MAX_WEAPONS; weapon++) { p->ammo_amount[weapon] = 0; } p->gotweapon.Zero(); p->oweapon_pos = p->weapon_pos = 6; p->okickback_pic = p->kickback_pic = 5; p->curr_weapon = PISTOL_WEAPON; p->gotweapon.Set(PISTOL_WEAPON); p->gotweapon.Set(KNEE_WEAPON); p->ammo_amount[PISTOL_WEAPON] = std::min(gs.max_ammo_amount[PISTOL_WEAPON], 48); p->gotweapon.Set(HANDREMOTE_WEAPON); p->last_weapon = -1; p->show_empty_weapon= 0; p->last_pissed_time = 0; p->holster_weapon = 0; // Always clear these, even for non-RRRA p->OnMotorcycle = 0; p->moto_underwater = 0; p->OnBoat = 0; p->lotag800kill = 0; if (isRRRA()) { chickenphase = 0; p->ammo_amount[KNEE_WEAPON] = 1; p->gotweapon.Set(SLINGBLADE_WEAPON); p->ammo_amount[SLINGBLADE_WEAPON] = 1; } OnEvent(EVENT_RESETWEAPONS, snum, nullptr, -1); } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- void resetinventory(int snum) { struct player_struct* p; p = &ps[snum]; p->inven_icon = 0; p->boot_amount = 0; p->scuba_on = 0; p->scuba_amount = 0; p->heat_amount = 0; p->heat_on = 0; p->jetpack_on = 0; p->jetpack_amount = 0; p->shield_amount = gs.max_armour_amount; p->holoduke_on = nullptr; p->holoduke_amount = 0; p->firstaid_amount = 0; p->steroids_amount = 0; p->inven_icon = 0; if (isRR() && ud.multimode > 1 && ud.coop != 1) { p->keys[0] = 1; p->keys[1] = 1; p->keys[2] = 1; p->keys[3] = 1; p->keys[4] = 1; } else { p->keys[0] = 0; p->keys[1] = 0; p->keys[2] = 0; p->keys[3] = 0; p->keys[4] = 0; } p->drunkang = 1647; p->eatang = 1647; p->drink_amt = 0; p->eat = 0; p->drink_timer = 0; p->eat_timer = 0; p->shotgun_state[0] = 0; p->shotgun_state[1] = 0; p->detonate_time = 0; p->detonate_count = 0; p->recoil = 0; p->yehaa_timer = 0; resetlanepics(); if (numplayers < 2) { ufospawn = std::min(ud.player_skill*4+1, 32); ufocnt = 0; hulkspawn = ud.player_skill + 1; } else { ufospawn = 32; ufocnt = 0; hulkspawn = 2; } OnEvent(EVENT_RESETINVENTORY, snum, p->GetActor()); } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- void resetprestat(int snum,int g) { struct player_struct *p; short i; p = &ps[snum]; spriteqloc = 0; for(i=0;ihbomb_on = 0; p->pals.a = 0; p->toggle_key_flag = 0; p->secret_rooms = 0; p->max_secret_rooms = 0; p->actors_killed = 0; p->max_actors_killed = 0; p->lastrandomspot = 0; p->oweapon_pos = p->weapon_pos = 6; p->okickback_pic = p->kickback_pic = 5; p->last_weapon = -1; p->weapreccnt = 0; p->show_empty_weapon= 0; p->holster_weapon = 0; p->last_pissed_time = 0; p->one_parallax_sectnum = -1; p->visibility = ud.const_visibility; screenpeek = myconnectindex; numanimwalls = 0; numcyclers = 0; animatecnt = 0; parallaxtype = 0; randomseed = 17L; paused = 0; ud.cameraactor =nullptr; tempwallptr = 0; camsprite =nullptr; earthquaketime = 0; WindTime = 0; WindDir = 0; fakebubba_spawn = 0; RRRA_ExitedLevel = 0; BellTime = 0; BellSprite = nullptr; if(p->curr_weapon == HANDREMOTE_WEAPON) { p->ammo_amount[HANDBOMB_WEAPON]++; p->curr_weapon = HANDBOMB_WEAPON; } p->timebeforeexit = 0; p->customexitsound = 0; p->stairs = 0; //if (!isRRRA()) p->fogtype = 0; p->noise_x = 131072; p->noise_y = 131072; p->donoise = 0; p->noise_radius = 0; if (isRR() && ud.multimode > 1 && ud.coop != 1) { p->keys[0] = 1; p->keys[1] = 1; p->keys[2] = 1; p->keys[3] = 1; p->keys[4] = 1; } else { p->keys[0] = 0; p->keys[1] = 0; p->keys[2] = 0; p->keys[3] = 0; p->keys[4] = 0; } p->drunkang = 1647; p->eatang = 1647; p->drink_amt = 0; p->eat = 0; p->drink_timer = 0; p->eat_timer = 0; p->shotgun_state[0] = 0; p->shotgun_state[1] = 0; p->detonate_time = 0; p->detonate_count = 0; p->recoil = 0; p->yehaa_timer = 0; resetlanepics(); if (numplayers < 2) { ufospawn = std::min(ud.player_skill*4+1, 32); ufocnt = 0; hulkspawn = ud.player_skill + 1; } else { ufospawn = 32; ufocnt = 0; hulkspawn = 2; } } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- void resetpspritevars(int g) { int i, j; short circ; int firstx, firsty; spritetype* s; int aimmode[MAXPLAYERS]; STATUSBARTYPE tsbar[MAXPLAYERS]; EGS(ps[0].cursectnum, ps[0].posx, ps[0].posy, ps[0].posz, TILE_APLAYER, 0, 0, 0, ps[0].angle.ang.asbuild(), 0, 0, nullptr, 10); if (ud.recstat != 2) for (i = 0; i < MAXPLAYERS; i++) { aimmode[i] = ps[i].aim_mode; if (ud.multimode > 1 && ud.coop == 1 && ud.last_level >= 0) { for (j = 0; j < MAX_WEAPONS; j++) { tsbar[i].ammo_amount[j] = ps[i].ammo_amount[j]; tsbar[i].gotweapon.Set(j, ps[i].gotweapon[j]); } tsbar[i].shield_amount = ps[i].shield_amount; tsbar[i].curr_weapon = ps[i].curr_weapon; tsbar[i].inven_icon = ps[i].inven_icon; tsbar[i].firstaid_amount = ps[i].firstaid_amount; tsbar[i].steroids_amount = ps[i].steroids_amount; tsbar[i].holoduke_amount = ps[i].holoduke_amount; tsbar[i].jetpack_amount = ps[i].jetpack_amount; tsbar[i].heat_amount = ps[i].heat_amount; tsbar[i].scuba_amount = ps[i].scuba_amount; tsbar[i].boot_amount = ps[i].boot_amount; } } resetplayerstats(0); for (i = 1; i < MAXPLAYERS; i++) memcpy(&ps[i], &ps[0], sizeof(ps[0])); if (ud.recstat != 2) for (i = 0; i < MAXPLAYERS; i++) { ps[i].aim_mode = aimmode[i]; if (ud.multimode > 1 && ud.coop == 1 && ud.last_level >= 0) { for (j = 0; j < MAX_WEAPONS; j++) { ps[i].ammo_amount[j] = tsbar[i].ammo_amount[j]; ps[i].gotweapon.Set(j, tsbar[i].gotweapon[j]); } ps[i].shield_amount = tsbar[i].shield_amount; ps[i].curr_weapon = tsbar[i].curr_weapon; ps[i].inven_icon = tsbar[i].inven_icon; ps[i].firstaid_amount = tsbar[i].firstaid_amount; ps[i].steroids_amount = tsbar[i].steroids_amount; ps[i].holoduke_amount = tsbar[i].holoduke_amount; ps[i].jetpack_amount = tsbar[i].jetpack_amount; ps[i].heat_amount = tsbar[i].heat_amount; ps[i].scuba_amount = tsbar[i].scuba_amount; ps[i].boot_amount = tsbar[i].boot_amount; } } numplayersprites = 0; circ = 2048 / ud.multimode; which_palookup = 9; j = connecthead; DukeStatIterator it(STAT_PLAYER); while (auto act = it.Next()) { s = act->s; if (numplayersprites == MAXPLAYERS) I_Error("Too many player sprites (max 16.)"); if (numplayersprites == 0) { firstx = ps[0].posx; firsty = ps[0].posy; } po[numplayersprites].ox = s->x; po[numplayersprites].oy = s->y; po[numplayersprites].oz = s->z; po[numplayersprites].oa = s->ang; po[numplayersprites].os = s->sectnum; numplayersprites++; if (j >= 0) { act->SetOwner(act); s->shade = 0; s->xrepeat = isRR() ? 24 : 42; s->yrepeat = isRR() ? 17 : 36; s->cstat = 1 + 256; s->xoffset = 0; s->clipdist = 64; if (ps[j].last_extra == 0) { ps[j].last_extra = gs.max_player_health; s->extra = gs.max_player_health; } else s->extra = ps[j].last_extra; s->yvel = j; if (ud.last_level == -1) { if (s->pal == 0) { s->pal = ps[j].palookup = which_palookup; ud.user_pals[j] = which_palookup; which_palookup++; if (which_palookup == 17) which_palookup = 9; } else ud.user_pals[j] = ps[j].palookup = s->pal; } else s->pal = ps[j].palookup = ud.user_pals[j]; ps[j].i = act->GetIndex(); ps[j].frag_ps = j; act->SetOwner(act); ps[j].bobposx = ps[j].oposx = ps[j].posx = s->x; ps[j].bobposy = ps[j].oposy = ps[j].posy = s->y; ps[j].oposz = ps[j].posz = s->z; s->backuppos(); ps[j].angle.oang = ps[j].angle.ang = buildang(s->ang); updatesector(s->x, s->y, &ps[j].cursectnum); j = connectpoint2[j]; } else deletesprite(act); } } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- void lava_cleararrays(); void prelevel_common(int g) { int i; auto p = &ps[screenpeek]; p->sea_sick_stat = 0; ufospawnsminion = 0; pistonsound = 0; p->SlotWin = 0; enemysizecheat = 0; p->MamaEnd = 0; mamaspawn_count = 15; banjosound = 0; RRRA_ExitedLevel = 0; lava_cleararrays(); geocnt = 0; ambientfx = 0; thunderon = 0; chickenplant = 0; WindTime = 0; WindDir = 0; fakebubba_spawn = 0; RRRA_ExitedLevel = 0; mamaspawn_count = 15; BellTime = 0; BellSprite = nullptr; // RRRA E2L1 fog handling. fogactive = 0; resetprestat(0, g); numclouds = 0; for (auto& h : hittype) h.clear(); memset(sectorextra, 0, sizeof(sectorextra)); memset(shadedsector, 0, sizeof(shadedsector)); memset(geosectorwarp, -1, sizeof(geosectorwarp)); memset(geosectorwarp2, -1, sizeof(geosectorwarp2)); memset(ambienthitag, -1, sizeof(ambienthitag)); memset(ambientlotag, -1, sizeof(ambientlotag)); for (i = 0; i < numsectors; i++) { sector[i].extra = 256; switch (sector[i].lotag) { case 20: case 22: if (sector[i].floorz > sector[i].ceilingz) sector[i].lotag |= 32768; continue; } if (sector[i].ceilingstat & 1) { //setupbackdrop(sector[i].ceilingpicnum); if (sector[i].ceilingpicnum == TILE_CLOUDYSKIES && numclouds < 127) clouds[numclouds++] = i; if (ps[0].one_parallax_sectnum == -1) ps[0].one_parallax_sectnum = i; } if (sector[i].lotag == 32767) //Found a secret room { ps[0].max_secret_rooms++; continue; } if (sector[i].lotag == -1) { ps[0].exitx = wall[sector[i].wallptr].x; ps[0].exity = wall[sector[i].wallptr].y; continue; } } } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- void resettimevars(void) { cloudclock = 0; PlayClock = 0; if (camsprite != nullptr) camsprite->temp_data[0] = 0; } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- void donewgame(MapRecord* map, int sk) { auto p = &ps[0]; show_shareware = 26 * 34; //ud.nextLevel = map; ud.player_skill = sk; ud.secretlevel = 0; ud.from_bonus = 0; ud.last_level = -1; M_ClearMenus(); ResetGameVars(); if (m_coop != 1) { if (isWW2GI()) { for (int i = 0; i < 12/*MAX_WEAPONS*/; i++) // aboive 12 have no data defined and would crash. { if (aplWeaponWorksLike[i][0] == PISTOL_WEAPON) { p->curr_weapon = i; p->gotweapon.Set(i); p->ammo_amount[i] = 48; } else if (aplWeaponWorksLike[i][0] == KNEE_WEAPON || aplWeaponWorksLike[i][0] == HANDREMOTE_WEAPON) { p->gotweapon.Set(i); } } } else { p->curr_weapon = PISTOL_WEAPON; p->gotweapon.Set(PISTOL_WEAPON); p->gotweapon.Set(KNEE_WEAPON); p->ammo_amount[PISTOL_WEAPON] = 48; p->gotweapon.Set(HANDREMOTE_WEAPON); p->last_weapon = -1; } p->last_weapon = -1; } display_mirror = 0; if (ud.multimode > 1) { if (numplayers < 2) { connecthead = 0; for (int i = 0; i < MAXPLAYERS; i++) connectpoint2[i] = i + 1; connectpoint2[ud.multimode - 1] = -1; } } else { connecthead = 0; connectpoint2[0] = -1; } } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- static int LoadTheMap(MapRecord *mi, struct player_struct *p, int gamemode) { int16_t lbang; if (isShareware() && (mi->flags & MI_USERMAP)) { I_Error("Cannot load user maps with shareware version!\n"); } engineLoadBoard(mi->fileName, isShareware(), &p->pos, &lbang, &p->cursectnum); currentLevel = mi; SECRET_SetMapName(mi->DisplayName(), mi->name); STAT_NewLevel(mi->fileName); if (isRR() && !isRRRA() && mi->levelNumber == levelnum(1, 1)) { for (int i = PISTOL_WEAPON; i < MAX_WEAPONS; i++) ps[0].ammo_amount[i] = 0; ps[0].gotweapon.Clear(KNEE_WEAPON); } p->angle.ang = buildang(lbang); memset(gotpic, 0, sizeof(gotpic)); if (isRR()) prelevel_r(gamemode); else prelevel_d(gamemode); if (isRRRA() && mi->levelNumber == levelnum(0, 2)) { for (int i = PISTOL_WEAPON; i < MAX_WEAPONS; i++) ps[0].ammo_amount[i] = 0; ps[0].gotweapon.Clear(KNEE_WEAPON); ps[0].gotweapon.Set(SLINGBLADE_WEAPON); ps[0].ammo_amount[SLINGBLADE_WEAPON] = 1; ps[0].curr_weapon = SLINGBLADE_WEAPON; } allignwarpelevators(); resetpspritevars(gamemode); if (isRR()) cacheit_r(); else cacheit_d(); return 0; } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- static void clearfrags(void) { for (int i = 0; i < ud.multimode; i++) { ps[i].frag = ps[i].fraggedself = 0; memset(ps[i].frags, 0, sizeof(ps[i].frags)); } } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- void enterlevel(MapRecord *mi, int gamemode) { // flushpackets(); // waitforeverybody(); ud.respawn_monsters = ud.m_respawn_monsters; ud.respawn_items = ud.m_respawn_items; ud.respawn_inventory = ud.m_respawn_inventory; ud.monsters_off = ud.m_monsters_off; ud.coop = ud.m_coop; ud.ffire = ud.m_ffire; lastlevel = 0; OnEvent(EVENT_ENTERLEVEL); // Stop all sounds FX_StopAllSounds(); FX_SetReverb(0); auto p = &ps[0]; LoadTheMap(mi, p, gamemode); // Try this first so that it can disable the CD player if no tracks are found. if (isRR()) S_PlayRRMusic(); if (ud.recstat != 2) { S_PlayLevelMusic(mi); } if (isShareware() && mi->levelNumber == 0 && ud.recstat != 2) FTA(QUOTE_F1HELP, &ps[myconnectindex]); for (int i = connecthead; i >= 0; i = connectpoint2[i]) { int pn = sector[ps[i].GetActor()->s->sectnum].floorpicnum; if (pn == TILE_HURTRAIL || pn == TILE_FLOORSLIME || pn == TILE_FLOORPLASMA) { resetweapons(i); resetinventory(i); ps[i].gotweapon.Clear(PISTOL_WEAPON); ps[i].ammo_amount[PISTOL_WEAPON] = 0; ps[i].curr_weapon = KNEE_WEAPON; ps[i].okickback_pic = ps[i].kickback_pic = 0; } } resetmys(); everyothertime = 0; global_random = 0; ud.last_level = currentLevel->levelNumber; ps[myconnectindex].over_shoulder_on = 0; clearfrags(); resettimevars(); // Here we go setLevelStarted(mi); if (isRRRA() && ps[screenpeek].sea_sick_stat == 1) { for (int i = 0; i < MAXWALLS; i++) { if (wall[i].picnum == 7873 || wall[i].picnum == 7870) StartInterpolation(i, Interp_Wall_PanX); } } } //--------------------------------------------------------------------------- // // Start a new game from the menu // //--------------------------------------------------------------------------- void GameInterface::NewGame(MapRecord* map, int skill, bool) { for (int i = 0; i != -1; i = connectpoint2[i]) { resetweapons(i); resetinventory(i); } ps[0].last_extra = gs.max_player_health; if (skill == -1) skill = ud.player_skill; else skill++; ud.player_skill = skill; ud.m_respawn_monsters = (skill == 4); ud.m_monsters_off = ud.monsters_off = 0; ud.m_respawn_items = 0; ud.m_respawn_inventory = 0; ud.multimode = 1; donewgame(map, skill); enterlevel(map, 0); PlayerColorChanged(); inputState.ClearAllInput(); gameaction = ga_level; } //--------------------------------------------------------------------------- // // Ideally this will become the only place where map progression gets set up. // //--------------------------------------------------------------------------- bool setnextmap(bool checksecretexit) { MapRecord* map = nullptr;; int from_bonus = 0; if (ud.eog) { } else if (checksecretexit && ud.from_bonus == 0) { if (ud.secretlevel > 0) { int newlevnum = levelnum(volfromlevelnum(currentLevel->levelNumber), ud.secretlevel-1); map = FindMapByLevelNum(newlevnum); if (map) { from_bonus = currentLevel->levelNumber + 1; } } } else if (ud.from_bonus && currentLevel->nextLevel == -1) // if the current level has an explicit link, use that instead of ud.from_bonus. { map = FindMapByLevelNum(ud.from_bonus); } else { map = FindNextMap(currentLevel); } // Make sure these two are cleared in case the map check errors out. ud.from_bonus = 0; ud.secretlevel = 0; if (map) { // If the map doesn't exist, abort with a meaningful message instead of crashing. if (fileSystem.FindFile(map->fileName) < 0) { I_Error("Trying to open non-existent %s", map->fileName.GetChars()); } ud.from_bonus = from_bonus; } CompleteLevel(map); return false; } //--------------------------------------------------------------------------- // // // //--------------------------------------------------------------------------- void exitlevel(MapRecord* nextlevel) { bool endofgame = nextlevel == nullptr; STAT_Update(endofgame); StopCommentary(); SummaryInfo info{}; info.kills = ps[0].actors_killed; info.maxkills = ps[0].max_actors_killed; info.secrets = ps[0].secret_rooms; info.maxsecrets = ps[0].max_secret_rooms; info.time = ps[0].player_par / GameTicRate; info.endofgame = endofgame; Mus_Stop(); if (playerswhenstarted > 1 && ud.coop != 1) { // MP scoreboard ShowScoreboard(playerswhenstarted, [=](bool) { // Clear potentially loaded per-map ART only after the bonus screens. artClearMapArt(); gameaction = ga_level; ud.eog = false; if (endofgame) { auto nextlevel = FindMapByLevelNum(0); if (!nextlevel) { gameaction = ga_startup; return; } else gameaction = ga_nextlevel; } else gameaction = ga_nextlevel; }); } else if (ud.multimode <= 1) { // SP cutscene + summary ShowIntermission(currentLevel, nextlevel, &info, [=](bool) { // Clear potentially loaded per-map ART only after the bonus screens. artClearMapArt(); ud.eog = false; gameaction = endofgame? ga_startup : ga_nextlevel; }); } } END_DUKE_NS