2017-04-17 11:33:19 +00:00
|
|
|
/*
|
|
|
|
** g_levellocals.h
|
|
|
|
** The static data for a level
|
|
|
|
**
|
|
|
|
**---------------------------------------------------------------------------
|
|
|
|
** Copyright 1998-2016 Randy Heit
|
|
|
|
** Copyright 2005-2017 Christoph Oelckers
|
|
|
|
** All rights reserved.
|
|
|
|
**
|
|
|
|
** Redistribution and use in source and binary forms, with or without
|
|
|
|
** modification, are permitted provided that the following conditions
|
|
|
|
** are met:
|
|
|
|
**
|
|
|
|
** 1. Redistributions of source code must retain the above copyright
|
|
|
|
** notice, this list of conditions and the following disclaimer.
|
|
|
|
** 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
** notice, this list of conditions and the following disclaimer in the
|
|
|
|
** documentation and/or other materials provided with the distribution.
|
|
|
|
** 3. The name of the author may not be used to endorse or promote products
|
|
|
|
** derived from this software without specific prior written permission.
|
|
|
|
**
|
|
|
|
** THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
|
|
|
|
** IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
|
|
|
|
** OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
|
|
|
** IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
|
|
** INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
|
|
|
|
** NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
|
|
** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
|
|
** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
|
|
|
|
** THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
**---------------------------------------------------------------------------
|
|
|
|
**
|
|
|
|
*/
|
|
|
|
|
2017-01-08 17:45:30 +00:00
|
|
|
#pragma once
|
|
|
|
|
2017-03-17 11:49:43 +00:00
|
|
|
#include "doomdata.h"
|
2017-01-08 17:45:30 +00:00
|
|
|
#include "g_level.h"
|
|
|
|
#include "r_defs.h"
|
2018-11-30 10:22:34 +00:00
|
|
|
#include "r_sky.h"
|
2017-03-10 01:22:42 +00:00
|
|
|
#include "portal.h"
|
2017-03-17 13:24:21 +00:00
|
|
|
#include "p_blockmap.h"
|
2018-10-28 22:08:38 +00:00
|
|
|
#include "p_local.h"
|
|
|
|
#include "p_destructible.h"
|
2017-01-08 17:45:30 +00:00
|
|
|
|
2017-11-15 16:06:21 +00:00
|
|
|
|
2019-04-04 12:43:49 +00:00
|
|
|
extern int i_compatflags, i_compatflags2;
|
|
|
|
|
2018-12-19 17:41:53 +00:00
|
|
|
struct FLevelData
|
|
|
|
{
|
2017-03-17 11:09:38 +00:00
|
|
|
TArray<vertex_t> vertexes;
|
|
|
|
TArray<sector_t> sectors;
|
2017-03-17 11:49:43 +00:00
|
|
|
TArray<line_t*> linebuffer; // contains the line lists for the sectors.
|
2017-03-17 11:09:38 +00:00
|
|
|
TArray<line_t> lines;
|
|
|
|
TArray<side_t> sides;
|
|
|
|
TArray<seg_t> segs;
|
|
|
|
TArray<subsector_t> subsectors;
|
|
|
|
TArray<node_t> nodes;
|
|
|
|
TArray<subsector_t> gamesubsectors;
|
|
|
|
TArray<node_t> gamenodes;
|
|
|
|
node_t *headgamenode;
|
2017-03-17 11:49:43 +00:00
|
|
|
TArray<uint8_t> rejectmatrix;
|
2018-12-19 17:41:53 +00:00
|
|
|
TArray<zone_t> Zones;
|
2017-01-08 17:45:30 +00:00
|
|
|
|
2017-01-14 15:05:40 +00:00
|
|
|
TArray<FSectorPortal> sectorPortals;
|
2017-03-17 11:09:38 +00:00
|
|
|
|
2018-10-28 22:08:38 +00:00
|
|
|
// [ZZ] Destructible geometry information
|
|
|
|
TMap<int, FHealthGroup> healthGroups;
|
|
|
|
|
2017-03-17 13:24:21 +00:00
|
|
|
FBlockmap blockmap;
|
|
|
|
|
2017-03-17 11:09:38 +00:00
|
|
|
// These are copies of the loaded map data that get used by the savegame code to skip unaltered fields
|
|
|
|
// Without such a mechanism the savegame format would become too slow and large because more than 80-90% are normally still unaltered.
|
|
|
|
TArray<sector_t> loadsectors;
|
|
|
|
TArray<line_t> loadlines;
|
|
|
|
TArray<side_t> loadsides;
|
2017-01-14 15:05:40 +00:00
|
|
|
|
2017-03-17 11:49:43 +00:00
|
|
|
// Maintain single and multi player starting spots.
|
|
|
|
TArray<FPlayerStart> deathmatchstarts;
|
|
|
|
FPlayerStart playerstarts[MAXPLAYERS];
|
|
|
|
TArray<FPlayerStart> AllPlayerStarts;
|
|
|
|
|
2018-12-19 17:41:53 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
struct FLevelLocals : public FLevelData
|
|
|
|
{
|
|
|
|
void Tick();
|
|
|
|
void Mark();
|
|
|
|
void AddScroller(int secnum);
|
|
|
|
void SetInterMusic(const char *nextmap);
|
|
|
|
void SetMusicVolume(float v);
|
|
|
|
|
2019-04-04 12:43:49 +00:00
|
|
|
int li_compatflags = i_compatflags;
|
|
|
|
int li_compatflags2 = i_compatflags2;
|
|
|
|
|
2018-12-19 17:41:53 +00:00
|
|
|
uint8_t md5[16]; // for savegame validation. If the MD5 does not match the savegame won't be loaded.
|
|
|
|
int time; // time in the hub
|
|
|
|
int maptime; // time in the map
|
|
|
|
int totaltime; // time in the game
|
|
|
|
int starttime;
|
|
|
|
int partime;
|
|
|
|
int sucktime;
|
2018-12-24 09:18:25 +00:00
|
|
|
uint32_t spawnindex;
|
2018-12-19 17:41:53 +00:00
|
|
|
|
|
|
|
level_info_t *info;
|
|
|
|
int cluster;
|
|
|
|
int clusterflags;
|
|
|
|
int levelnum;
|
|
|
|
int lumpnum;
|
|
|
|
FString LevelName;
|
|
|
|
FString MapName; // the lump name (E1M1, MAP01, etc)
|
|
|
|
FString NextMap; // go here when using the regular exit
|
|
|
|
FString NextSecretMap; // map to go to when used secret exit
|
|
|
|
FString F1Pic;
|
|
|
|
EMapType maptype;
|
|
|
|
|
|
|
|
uint64_t ShaderStartTime = 0; // tell the shader system when we started the level (forces a timer restart)
|
|
|
|
|
|
|
|
static const int BODYQUESIZE = 32;
|
|
|
|
TObjPtr<AActor*> bodyque[BODYQUESIZE];
|
|
|
|
int bodyqueslot;
|
2017-01-14 15:05:40 +00:00
|
|
|
|
2017-03-08 17:47:52 +00:00
|
|
|
uint32_t flags;
|
|
|
|
uint32_t flags2;
|
|
|
|
uint32_t flags3;
|
2017-01-08 17:45:30 +00:00
|
|
|
|
2017-03-08 17:47:52 +00:00
|
|
|
uint32_t fadeto; // The color the palette fades to (usually black)
|
|
|
|
uint32_t outsidefog; // The fog for sectors with sky ceilings
|
2017-01-08 17:45:30 +00:00
|
|
|
|
2017-03-08 17:47:52 +00:00
|
|
|
uint32_t hazardcolor; // what color strife hazard blends the screen color as
|
|
|
|
uint32_t hazardflash; // what color strife hazard flashes the screen color as
|
2017-03-05 14:58:07 +00:00
|
|
|
|
2017-01-08 17:45:30 +00:00
|
|
|
FString Music;
|
|
|
|
int musicorder;
|
|
|
|
int cdtrack;
|
|
|
|
unsigned int cdid;
|
|
|
|
FTextureID skytexture1;
|
|
|
|
FTextureID skytexture2;
|
|
|
|
|
|
|
|
float skyspeed1; // Scrolling speed of sky textures, in pixels per ms
|
|
|
|
float skyspeed2;
|
|
|
|
|
|
|
|
int total_secrets;
|
|
|
|
int found_secrets;
|
|
|
|
|
|
|
|
int total_items;
|
|
|
|
int found_items;
|
|
|
|
|
|
|
|
int total_monsters;
|
|
|
|
int killed_monsters;
|
|
|
|
|
|
|
|
double gravity;
|
|
|
|
double aircontrol;
|
|
|
|
double airfriction;
|
|
|
|
int airsupply;
|
|
|
|
int DefaultEnvironment; // Default sound environment.
|
|
|
|
|
|
|
|
TArray<DVector2> Scrolls; // NULL if no DScrollers in this level
|
|
|
|
|
2017-03-08 17:47:52 +00:00
|
|
|
int8_t WallVertLight; // Light diffs for vert/horiz walls
|
|
|
|
int8_t WallHorizLight;
|
2017-01-08 17:45:30 +00:00
|
|
|
|
|
|
|
bool FromSnapshot; // The current map was restored from a snapshot
|
2018-05-01 07:02:24 +00:00
|
|
|
bool HasHeightSecs; // true if some Transfer_Heights effects are present in the map. If this is false, some checks in the renderer can be shortcut.
|
2019-01-28 19:15:48 +00:00
|
|
|
uint8_t frozenstate;
|
2017-01-08 17:45:30 +00:00
|
|
|
|
|
|
|
double teamdamage;
|
|
|
|
|
2017-03-14 12:54:24 +00:00
|
|
|
// former OpenGL-exclusive properties that should also be usable by the true color software renderer.
|
|
|
|
int fogdensity;
|
|
|
|
int outsidefogdensity;
|
|
|
|
int skyfog;
|
|
|
|
|
2018-11-24 19:32:12 +00:00
|
|
|
FName deathsequence;
|
2017-12-29 07:42:03 +00:00
|
|
|
float pixelstretch;
|
2018-03-24 12:06:37 +00:00
|
|
|
float MusicVolume;
|
2017-03-14 12:54:24 +00:00
|
|
|
|
2019-01-01 18:35:55 +00:00
|
|
|
FDynamicLight *lights;
|
|
|
|
|
2017-01-08 17:45:30 +00:00
|
|
|
bool IsJumpingAllowed() const;
|
|
|
|
bool IsCrouchingAllowed() const;
|
|
|
|
bool IsFreelookAllowed() const;
|
2017-03-17 00:42:37 +00:00
|
|
|
|
|
|
|
node_t *HeadNode() const
|
|
|
|
{
|
2017-03-18 21:44:03 +00:00
|
|
|
return nodes.Size() == 0? nullptr : &nodes[nodes.Size() - 1];
|
2017-03-17 00:42:37 +00:00
|
|
|
}
|
|
|
|
node_t *HeadGamenode() const
|
|
|
|
{
|
2017-03-17 11:09:38 +00:00
|
|
|
return headgamenode;
|
2017-03-17 00:42:37 +00:00
|
|
|
}
|
2018-03-24 12:06:37 +00:00
|
|
|
|
2018-06-11 08:55:49 +00:00
|
|
|
// Returns true if level is loaded from saved game or is being revisited as a part of a hub
|
|
|
|
bool IsReentering() const
|
|
|
|
{
|
|
|
|
return savegamerestore
|
|
|
|
|| (info != nullptr && info->Snapshot.mBuffer != nullptr && info->isValid());
|
|
|
|
}
|
2019-01-28 19:15:48 +00:00
|
|
|
|
|
|
|
int isFrozen()
|
|
|
|
{
|
|
|
|
return frozenstate;
|
|
|
|
}
|
2017-01-08 17:45:30 +00:00
|
|
|
};
|
|
|
|
|
2018-12-19 17:41:53 +00:00
|
|
|
#ifndef NO_DEFINE_LEVEL
|
|
|
|
|
2017-01-08 17:45:30 +00:00
|
|
|
extern FLevelLocals level;
|
|
|
|
|
2017-01-08 23:46:16 +00:00
|
|
|
inline int vertex_t::Index() const
|
|
|
|
{
|
|
|
|
return int(this - &level.vertexes[0]);
|
|
|
|
}
|
|
|
|
|
2017-01-08 17:45:30 +00:00
|
|
|
inline int side_t::Index() const
|
|
|
|
{
|
|
|
|
return int(this - &level.sides[0]);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline int line_t::Index() const
|
|
|
|
{
|
|
|
|
return int(this - &level.lines[0]);
|
|
|
|
}
|
|
|
|
|
2017-03-16 20:33:13 +00:00
|
|
|
inline int seg_t::Index() const
|
|
|
|
{
|
|
|
|
return int(this - &level.segs[0]);
|
|
|
|
}
|
|
|
|
|
2017-03-16 23:22:52 +00:00
|
|
|
inline int subsector_t::Index() const
|
|
|
|
{
|
|
|
|
return int(this - &level.subsectors[0]);
|
|
|
|
}
|
|
|
|
|
2017-03-17 00:42:37 +00:00
|
|
|
inline int node_t::Index() const
|
|
|
|
{
|
|
|
|
return int(this - &level.nodes[0]);
|
|
|
|
}
|
|
|
|
|
2017-01-14 15:05:40 +00:00
|
|
|
inline FSectorPortal *line_t::GetTransferredPortal()
|
|
|
|
{
|
|
|
|
return portaltransferred >= level.sectorPortals.Size() ? (FSectorPortal*)nullptr : &level.sectorPortals[portaltransferred];
|
|
|
|
}
|
|
|
|
|
2017-01-08 17:45:30 +00:00
|
|
|
inline int sector_t::Index() const
|
|
|
|
{
|
|
|
|
return int(this - &level.sectors[0]);
|
|
|
|
}
|
2017-01-14 15:05:40 +00:00
|
|
|
|
|
|
|
inline FSectorPortal *sector_t::GetPortal(int plane)
|
|
|
|
{
|
|
|
|
return &level.sectorPortals[Portals[plane]];
|
|
|
|
}
|
|
|
|
|
|
|
|
inline double sector_t::GetPortalPlaneZ(int plane)
|
|
|
|
{
|
|
|
|
return level.sectorPortals[Portals[plane]].mPlaneZ;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline DVector2 sector_t::GetPortalDisplacement(int plane)
|
|
|
|
{
|
|
|
|
return level.sectorPortals[Portals[plane]].mDisplacement;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline int sector_t::GetPortalType(int plane)
|
|
|
|
{
|
|
|
|
return level.sectorPortals[Portals[plane]].mType;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline int sector_t::GetOppositePortalGroup(int plane)
|
|
|
|
{
|
|
|
|
return level.sectorPortals[Portals[plane]].mDestination->PortalGroup;
|
|
|
|
}
|
2017-03-10 01:22:42 +00:00
|
|
|
|
|
|
|
inline bool sector_t::PortalBlocksView(int plane)
|
|
|
|
{
|
|
|
|
if (GetPortalType(plane) != PORTS_LINKEDPORTAL) return false;
|
|
|
|
return !!(planes[plane].Flags & (PLANEF_NORENDER | PLANEF_DISABLED | PLANEF_OBSTRUCTED));
|
|
|
|
}
|
|
|
|
|
|
|
|
inline bool sector_t::PortalBlocksSight(int plane)
|
|
|
|
{
|
|
|
|
return PLANEF_LINKED != (planes[plane].Flags & (PLANEF_NORENDER | PLANEF_NOPASS | PLANEF_DISABLED | PLANEF_OBSTRUCTED | PLANEF_LINKED));
|
|
|
|
}
|
|
|
|
|
|
|
|
inline bool sector_t::PortalBlocksMovement(int plane)
|
|
|
|
{
|
|
|
|
return PLANEF_LINKED != (planes[plane].Flags & (PLANEF_NOPASS | PLANEF_DISABLED | PLANEF_OBSTRUCTED | PLANEF_LINKED));
|
|
|
|
}
|
|
|
|
|
|
|
|
inline bool sector_t::PortalBlocksSound(int plane)
|
|
|
|
{
|
|
|
|
return PLANEF_LINKED != (planes[plane].Flags & (PLANEF_BLOCKSOUND | PLANEF_DISABLED | PLANEF_OBSTRUCTED | PLANEF_LINKED));
|
|
|
|
}
|
|
|
|
|
|
|
|
inline bool sector_t::PortalIsLinked(int plane)
|
|
|
|
{
|
|
|
|
return (GetPortalType(plane) == PORTS_LINKEDPORTAL);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline FLinePortal *line_t::getPortal() const
|
|
|
|
{
|
|
|
|
return portalindex >= linePortals.Size() ? (FLinePortal*)NULL : &linePortals[portalindex];
|
|
|
|
}
|
|
|
|
|
|
|
|
// returns true if the portal is crossable by actors
|
|
|
|
inline bool line_t::isLinePortal() const
|
|
|
|
{
|
|
|
|
return portalindex >= linePortals.Size() ? false : !!(linePortals[portalindex].mFlags & PORTF_PASSABLE);
|
|
|
|
}
|
|
|
|
|
|
|
|
// returns true if the portal needs to be handled by the renderer
|
|
|
|
inline bool line_t::isVisualPortal() const
|
|
|
|
{
|
|
|
|
return portalindex >= linePortals.Size() ? false : !!(linePortals[portalindex].mFlags & PORTF_VISIBLE);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline line_t *line_t::getPortalDestination() const
|
|
|
|
{
|
|
|
|
return portalindex >= linePortals.Size() ? (line_t*)NULL : linePortals[portalindex].mDestination;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline int line_t::getPortalAlignment() const
|
|
|
|
{
|
|
|
|
return portalindex >= linePortals.Size() ? 0 : linePortals[portalindex].mAlign;
|
|
|
|
}
|
2018-11-30 10:22:34 +00:00
|
|
|
|
|
|
|
inline bool line_t::hitSkyWall(AActor* mo) const
|
|
|
|
{
|
|
|
|
return backsector &&
|
|
|
|
backsector->GetTexture(sector_t::ceiling) == skyflatnum &&
|
|
|
|
mo->Z() >= backsector->ceilingplane.ZatPoint(mo->PosRelative(this));
|
|
|
|
}
|
2018-12-19 17:41:53 +00:00
|
|
|
#endif
|