2019-09-19 22:42:45 +00:00
|
|
|
//-------------------------------------------------------------------------
|
|
|
|
/*
|
|
|
|
Copyright (C) 2010-2019 EDuke32 developers and contributors
|
|
|
|
Copyright (C) 2019 Nuke.YKT
|
|
|
|
|
|
|
|
This file is part of NBlood.
|
|
|
|
|
|
|
|
NBlood is free software; you can redistribute it and/or
|
|
|
|
modify it under the terms of the GNU General Public License version 2
|
|
|
|
as published by the Free Software Foundation.
|
|
|
|
|
|
|
|
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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
*/
|
|
|
|
//-------------------------------------------------------------------------
|
2019-09-21 18:59:54 +00:00
|
|
|
|
|
|
|
#include "ns.h" // Must come before everything else!
|
|
|
|
|
2019-09-19 22:42:45 +00:00
|
|
|
#include "build.h"
|
|
|
|
#include "common_game.h"
|
2020-07-25 16:48:26 +00:00
|
|
|
#include "zstring.h"
|
2019-11-02 21:10:53 +00:00
|
|
|
#include "m_crc32.h"
|
2020-01-11 11:54:58 +00:00
|
|
|
#include "md4.h"
|
2020-09-06 10:44:58 +00:00
|
|
|
#include "automap.h"
|
2020-10-28 23:08:06 +00:00
|
|
|
#include "raze_sound.h"
|
2021-03-19 19:40:44 +00:00
|
|
|
#include "gamefuncs.h"
|
2021-05-02 22:04:36 +00:00
|
|
|
#include "hw_sections.h"
|
2021-05-03 15:48:35 +00:00
|
|
|
#include "sectorgeometry.h"
|
2022-01-12 22:33:44 +00:00
|
|
|
#include "psky.h"
|
2019-09-19 22:42:45 +00:00
|
|
|
|
2020-12-09 14:56:32 +00:00
|
|
|
#include "blood.h"
|
2019-09-19 22:42:45 +00:00
|
|
|
|
2019-09-22 06:39:22 +00:00
|
|
|
BEGIN_BLD_NS
|
|
|
|
|
2021-12-29 19:33:09 +00:00
|
|
|
int gSkyCount;
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
2021-11-23 23:55:57 +00:00
|
|
|
DBloodActor* InsertSprite(sectortype* pSector, int nStat)
|
2019-09-19 22:42:45 +00:00
|
|
|
{
|
2021-12-29 19:45:55 +00:00
|
|
|
auto act = static_cast<DBloodActor*>(::InsertActor(RUNTIME_CLASS(DBloodActor), pSector, nStat));
|
|
|
|
act->spr.cstat = CSTAT_SPRITE_YCENTER;
|
2022-10-04 17:08:29 +00:00
|
|
|
act->clipdist = 8;
|
2021-12-29 19:45:55 +00:00
|
|
|
act->spr.xrepeat = act->spr.yrepeat = 64;
|
|
|
|
return act;
|
2019-09-19 22:42:45 +00:00
|
|
|
}
|
|
|
|
|
2021-12-04 21:04:16 +00:00
|
|
|
int DeleteSprite(DBloodActor* actor)
|
2019-09-19 22:42:45 +00:00
|
|
|
{
|
2021-08-28 08:38:36 +00:00
|
|
|
#ifdef NOONE_EXTENSIONS
|
2021-12-29 19:45:55 +00:00
|
|
|
for (auto& ctrl : gPlayerCtrl) if (ctrl.qavScene.initiator == actor) ctrl.qavScene.initiator = nullptr;
|
2021-08-28 08:38:36 +00:00
|
|
|
#endif
|
2019-06-29 12:24:23 +00:00
|
|
|
|
2021-12-29 19:45:55 +00:00
|
|
|
actor->Destroy();
|
|
|
|
return 0;
|
2019-09-19 22:42:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-12-04 21:04:16 +00:00
|
|
|
bool gModernMap = false;
|
|
|
|
int gVisibility;
|
2019-09-19 22:42:45 +00:00
|
|
|
|
2021-12-29 19:33:09 +00:00
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
2021-12-29 19:45:55 +00:00
|
|
|
void dbCrypt(char* pPtr, int nLength, int nKey)
|
2019-09-19 22:42:45 +00:00
|
|
|
{
|
2021-12-29 19:45:55 +00:00
|
|
|
for (int i = 0; i < nLength; i++)
|
|
|
|
{
|
|
|
|
pPtr[i] = pPtr[i] ^ nKey;
|
|
|
|
nKey++;
|
|
|
|
}
|
2019-09-19 22:42:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-12-29 19:33:09 +00:00
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
2019-09-19 22:42:45 +00:00
|
|
|
|
2021-12-29 19:45:55 +00:00
|
|
|
unsigned int dbReadMapCRC(const char* pPath)
|
2019-09-19 22:42:45 +00:00
|
|
|
{
|
2021-12-29 19:45:55 +00:00
|
|
|
FString mapname = pPath;
|
|
|
|
DefaultExtension(mapname, ".map");
|
|
|
|
auto fr = fileSystem.OpenFileReader(mapname);
|
|
|
|
|
|
|
|
if (!fr.isOpen())
|
|
|
|
{
|
|
|
|
Printf("Error opening map file %s", pPath);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
MAPSIGNATURE header;
|
|
|
|
fr.Read(&header, 6);
|
|
|
|
if (memcmp(header.signature, "BLM\x1a", 4))
|
|
|
|
{
|
|
|
|
I_Error("%s: Map file corrupted.", mapname.GetChars());
|
|
|
|
}
|
|
|
|
int ver = LittleShort(header.version);
|
|
|
|
if ((ver & 0xff00) == 0x600)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
else if ((ver & 0xff00) == 0x700)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
I_Error("%s: Map file is wrong version.", mapname.GetChars());
|
|
|
|
}
|
|
|
|
fr.Seek(-4, FileReader::SeekEnd);
|
|
|
|
return fr.ReadInt32();
|
2019-09-19 22:42:45 +00:00
|
|
|
}
|
|
|
|
|
2021-12-29 19:33:09 +00:00
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
2020-10-03 16:35:47 +00:00
|
|
|
|
2022-09-24 16:27:02 +00:00
|
|
|
void dbLoadMap(const char* pPath, DVector3& pos, short* pAngle, sectortype** cursect, unsigned int* pCRC, BloodSpawnSpriteDef& sprites)
|
2021-11-19 19:50:02 +00:00
|
|
|
{
|
2021-12-29 19:45:55 +00:00
|
|
|
const int nXSectorSize = 60;
|
|
|
|
const int nXSpriteSize = 56;
|
|
|
|
const int nXWallSize = 24;
|
2021-12-29 19:33:09 +00:00
|
|
|
|
2021-12-29 19:45:55 +00:00
|
|
|
MAPHEADER2 xheader;
|
|
|
|
int gMapRev, gMattId;
|
2021-12-29 19:33:09 +00:00
|
|
|
|
2021-12-29 19:45:55 +00:00
|
|
|
int16_t tpskyoff[256];
|
|
|
|
ClearAutomap();
|
2021-11-19 19:50:02 +00:00
|
|
|
#ifdef NOONE_EXTENSIONS
|
2021-12-29 19:45:55 +00:00
|
|
|
gModernMap = false;
|
2021-11-19 19:50:02 +00:00
|
|
|
#endif
|
2019-11-08 19:57:01 +00:00
|
|
|
|
2021-08-28 08:38:36 +00:00
|
|
|
#ifdef NOONE_EXTENSIONS
|
2021-12-29 19:45:55 +00:00
|
|
|
for (auto& ctrl : gPlayerCtrl) ctrl.qavScene.initiator = nullptr;
|
2021-08-28 08:38:36 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2021-12-29 19:45:55 +00:00
|
|
|
FString mapname = pPath;
|
|
|
|
DefaultExtension(mapname, ".map");
|
|
|
|
auto fr = fileSystem.OpenFileReader(mapname);
|
|
|
|
|
|
|
|
if (!fr.isOpen())
|
|
|
|
{
|
|
|
|
I_Error("Error opening map file %s", mapname.GetChars());
|
|
|
|
}
|
|
|
|
MAPSIGNATURE header;
|
|
|
|
fr.Read(&header, 6);
|
|
|
|
if (memcmp(header.signature, "BLM\x1a", 4))
|
|
|
|
{
|
|
|
|
I_Error("%s: Map file corrupted", mapname.GetChars());
|
|
|
|
}
|
|
|
|
bool encrypted = 0;
|
|
|
|
if ((LittleShort(header.version) & 0xff00) == 0x700) {
|
|
|
|
encrypted = 1;
|
2021-11-19 19:50:02 +00:00
|
|
|
|
|
|
|
#ifdef NOONE_EXTENSIONS
|
2021-12-29 19:45:55 +00:00
|
|
|
// indicate if the map requires modern features to work properly
|
|
|
|
// for maps wich created in PMAPEDIT BETA13 or higher versions. Since only minor version changed,
|
|
|
|
// the map is still can be loaded with vanilla BLOOD / MAPEDIT and should work in other ports too.
|
|
|
|
if ((header.version & 0x00ff) == 0x001) gModernMap = true;
|
2021-11-19 19:50:02 +00:00
|
|
|
#endif
|
2019-10-11 21:59:39 +00:00
|
|
|
|
2021-12-29 19:45:55 +00:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
I_Error("%s: Map file is wrong version", mapname.GetChars());
|
|
|
|
}
|
|
|
|
|
|
|
|
MAPHEADER mapHeader;
|
|
|
|
fr.Read(&mapHeader, 37/* sizeof(mapHeader)*/);
|
|
|
|
if (mapHeader.mattid != 0 && mapHeader.mattid != 0x7474614d && mapHeader.mattid != 0x4d617474) {
|
|
|
|
dbCrypt((char*)&mapHeader, sizeof(mapHeader), 0x7474614d);
|
|
|
|
}
|
|
|
|
|
|
|
|
mapHeader.x = LittleLong(mapHeader.x);
|
|
|
|
mapHeader.y = LittleLong(mapHeader.y);
|
|
|
|
mapHeader.z = LittleLong(mapHeader.z);
|
|
|
|
mapHeader.ang = LittleShort(mapHeader.ang);
|
|
|
|
mapHeader.sect = LittleShort(mapHeader.sect);
|
|
|
|
mapHeader.pskybits = LittleShort(mapHeader.pskybits);
|
|
|
|
mapHeader.visibility = LittleLong(mapHeader.visibility);
|
|
|
|
mapHeader.mattid = LittleLong(mapHeader.mattid);
|
|
|
|
mapHeader.revision = LittleLong(mapHeader.revision);
|
|
|
|
mapHeader.numsectors = LittleShort(mapHeader.numsectors);
|
|
|
|
mapHeader.numwalls = LittleShort(mapHeader.numwalls);
|
|
|
|
mapHeader.numsprites = LittleShort(mapHeader.numsprites);
|
|
|
|
|
2022-08-23 21:36:23 +00:00
|
|
|
pos = { mapHeader.x * maptoworld, mapHeader.y * maptoworld, mapHeader.z * zmaptoworld };
|
2021-12-29 19:45:55 +00:00
|
|
|
*pAngle = mapHeader.ang;
|
|
|
|
gVisibility = g_visibility = mapHeader.visibility;
|
|
|
|
gMattId = mapHeader.mattid;
|
|
|
|
if (encrypted)
|
|
|
|
{
|
|
|
|
if (!(mapHeader.mattid == 0x7474614d || mapHeader.mattid == 0x4d617474 || !mapHeader.mattid))
|
|
|
|
{
|
|
|
|
I_Error("%s: Corrupted Map file", mapname.GetChars());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (mapHeader.mattid)
|
|
|
|
{
|
|
|
|
I_Error("%s: Corrupted Map file", mapname.GetChars());
|
|
|
|
}
|
|
|
|
gMapRev = mapHeader.revision;
|
|
|
|
allocateMapArrays(mapHeader.numwalls, mapHeader.numsectors, mapHeader.numsprites);
|
2022-09-24 16:27:02 +00:00
|
|
|
*cursect = validSectorIndex(mapHeader.sect)? §or[mapHeader.sect] : nullptr;
|
2021-12-29 19:45:55 +00:00
|
|
|
|
|
|
|
if (encrypted)
|
|
|
|
{
|
|
|
|
fr.Read(&xheader, 128);
|
|
|
|
dbCrypt((char*)&xheader, 128, wall.Size());
|
|
|
|
|
|
|
|
xheader.numxsprites = LittleLong(xheader.numxsprites);
|
|
|
|
xheader.numxwalls = LittleLong(xheader.numxwalls);
|
|
|
|
xheader.numxsectors = LittleLong(xheader.numxsectors);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
memset(&xheader, 0, 128);
|
|
|
|
}
|
|
|
|
gSkyCount = 1 << mapHeader.pskybits;
|
|
|
|
fr.Read(tpskyoff, gSkyCount * sizeof(tpskyoff[0]));
|
|
|
|
if (encrypted)
|
|
|
|
{
|
|
|
|
dbCrypt((char*)tpskyoff, gSkyCount * sizeof(tpskyoff[0]), gSkyCount * 2);
|
|
|
|
}
|
|
|
|
for (int i = 0; i < ClipHigh(gSkyCount, MAXPSKYTILES); i++)
|
|
|
|
{
|
2022-01-12 22:33:44 +00:00
|
|
|
tpskyoff[i] = LittleShort(tpskyoff[i]);
|
2021-12-29 19:45:55 +00:00
|
|
|
}
|
|
|
|
|
2022-01-12 22:33:44 +00:00
|
|
|
defineSky(DEFAULTPSKY, mapHeader.pskybits, tpskyoff);
|
|
|
|
|
2021-12-29 19:45:55 +00:00
|
|
|
for (unsigned i = 0; i < sector.Size(); i++)
|
|
|
|
{
|
|
|
|
sectortype* pSector = §or[i];
|
|
|
|
sectortypedisk load;
|
|
|
|
fr.Read(&load, sizeof(sectortypedisk));
|
|
|
|
if (encrypted)
|
|
|
|
{
|
|
|
|
dbCrypt((char*)&load, sizeof(sectortypedisk), gMapRev * sizeof(sectortypedisk));
|
|
|
|
}
|
|
|
|
pSector->wallptr = LittleShort(load.wallptr);
|
|
|
|
pSector->wallnum = LittleShort(load.wallnum);
|
2022-02-03 18:15:46 +00:00
|
|
|
pSector->setzfrommap(LittleLong(load.ceilingz), LittleLong(load.floorz));
|
2021-12-29 19:45:55 +00:00
|
|
|
pSector->ceilingstat = ESectorFlags::FromInt(LittleShort(load.ceilingstat));
|
|
|
|
pSector->floorstat = ESectorFlags::FromInt(LittleShort(load.floorstat));
|
|
|
|
pSector->ceilingpicnum = LittleShort(load.ceilingpicnum);
|
|
|
|
pSector->ceilingheinum = LittleShort(load.ceilingheinum);
|
|
|
|
pSector->floorpicnum = LittleShort(load.floorpicnum);
|
|
|
|
pSector->floorheinum = LittleShort(load.floorheinum);
|
|
|
|
pSector->type = LittleShort(load.type);
|
|
|
|
pSector->hitag = LittleShort(load.hitag);
|
|
|
|
pSector->extra = LittleShort(load.extra);
|
|
|
|
pSector->ceilingshade = load.ceilingshade;
|
|
|
|
pSector->ceilingpal = load.ceilingpal;
|
|
|
|
pSector->ceilingxpan_ = load.ceilingxpanning;
|
|
|
|
pSector->ceilingypan_ = load.ceilingypanning;
|
|
|
|
pSector->floorshade = load.floorshade;
|
|
|
|
pSector->floorpal = load.floorpal;
|
|
|
|
pSector->floorxpan_ = load.floorxpanning;
|
|
|
|
pSector->floorypan_ = load.floorypanning;
|
|
|
|
pSector->visibility = load.visibility;
|
|
|
|
pSector->slopewallofs = load.fogpal;
|
|
|
|
pSector->dirty = EDirty::AllDirty;
|
|
|
|
pSector->exflags = 0;
|
|
|
|
pSector->fogpal = 0;
|
|
|
|
|
|
|
|
if (pSector->extra > 0)
|
|
|
|
{
|
2022-08-04 17:13:55 +00:00
|
|
|
uint8_t pBuffer[nXSectorSize];
|
2021-12-29 19:45:55 +00:00
|
|
|
pSector->allocX();
|
|
|
|
XSECTOR* pXSector = &pSector->xs();
|
|
|
|
int nCount;
|
|
|
|
if (!encrypted)
|
|
|
|
{
|
|
|
|
nCount = nXSectorSize;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
nCount = xheader.numxsectors;
|
|
|
|
}
|
|
|
|
assert(nCount <= nXSectorSize);
|
|
|
|
fr.Read(pBuffer, nCount);
|
|
|
|
BitReader bitReader(pBuffer, nCount);
|
|
|
|
/*pXSector->reference =*/ bitReader.readSigned(14);
|
|
|
|
pXSector->state = bitReader.readUnsigned(1);
|
|
|
|
pXSector->busy = bitReader.readUnsigned(17);
|
|
|
|
pXSector->data = bitReader.readUnsigned(16);
|
|
|
|
pXSector->txID = bitReader.readUnsigned(10);
|
|
|
|
pXSector->busyWaveA = bitReader.readUnsigned(3);
|
|
|
|
pXSector->busyWaveB = bitReader.readUnsigned(3);
|
|
|
|
pXSector->rxID = bitReader.readUnsigned(10);
|
|
|
|
pXSector->command = bitReader.readUnsigned(8);
|
|
|
|
pXSector->triggerOn = bitReader.readUnsigned(1);
|
|
|
|
pXSector->triggerOff = bitReader.readUnsigned(1);
|
|
|
|
pXSector->busyTimeA = bitReader.readUnsigned(12);
|
|
|
|
pXSector->waitTimeA = bitReader.readUnsigned(12);
|
|
|
|
pXSector->restState = bitReader.readUnsigned(1);
|
|
|
|
pXSector->interruptable = bitReader.readUnsigned(1);
|
|
|
|
pXSector->amplitude = bitReader.readSigned(8);
|
|
|
|
pXSector->freq = bitReader.readUnsigned(8);
|
|
|
|
pXSector->reTriggerA = bitReader.readUnsigned(1);
|
|
|
|
pXSector->reTriggerB = bitReader.readUnsigned(1);
|
|
|
|
pXSector->phase = bitReader.readUnsigned(8);
|
|
|
|
pXSector->wave = bitReader.readUnsigned(4);
|
|
|
|
pXSector->shadeAlways = bitReader.readUnsigned(1);
|
|
|
|
pXSector->shadeFloor = bitReader.readUnsigned(1);
|
|
|
|
pXSector->shadeCeiling = bitReader.readUnsigned(1);
|
|
|
|
pXSector->shadeWalls = bitReader.readUnsigned(1);
|
|
|
|
pXSector->shade = bitReader.readSigned(8);
|
|
|
|
pXSector->panAlways = bitReader.readUnsigned(1);
|
|
|
|
pXSector->panFloor = bitReader.readUnsigned(1);
|
|
|
|
pXSector->panCeiling = bitReader.readUnsigned(1);
|
|
|
|
pXSector->Drag = bitReader.readUnsigned(1);
|
|
|
|
pXSector->Underwater = bitReader.readUnsigned(1);
|
|
|
|
pXSector->Depth = bitReader.readUnsigned(3);
|
|
|
|
pXSector->panVel = bitReader.readUnsigned(8);
|
2022-09-19 09:04:09 +00:00
|
|
|
pXSector->panAngle = mapangle(bitReader.readUnsigned(11));
|
2021-12-29 19:45:55 +00:00
|
|
|
pXSector->unused1 = bitReader.readUnsigned(1);
|
|
|
|
pXSector->decoupled = bitReader.readUnsigned(1);
|
|
|
|
pXSector->triggerOnce = bitReader.readUnsigned(1);
|
|
|
|
pXSector->isTriggered = bitReader.readUnsigned(1);
|
|
|
|
pXSector->Key = bitReader.readUnsigned(3);
|
|
|
|
pXSector->Push = bitReader.readUnsigned(1);
|
|
|
|
pXSector->Vector = bitReader.readUnsigned(1);
|
|
|
|
pXSector->Reserved = bitReader.readUnsigned(1);
|
|
|
|
pXSector->Enter = bitReader.readUnsigned(1);
|
|
|
|
pXSector->Exit = bitReader.readUnsigned(1);
|
|
|
|
pXSector->Wallpush = bitReader.readUnsigned(1);
|
|
|
|
pXSector->color = bitReader.readUnsigned(1);
|
|
|
|
/*pXSector->unused2 =*/ bitReader.readUnsigned(1);
|
|
|
|
pXSector->busyTimeB = bitReader.readUnsigned(12);
|
|
|
|
pXSector->waitTimeB = bitReader.readUnsigned(12);
|
|
|
|
pXSector->stopOn = bitReader.readUnsigned(1);
|
|
|
|
pXSector->stopOff = bitReader.readUnsigned(1);
|
|
|
|
pXSector->ceilpal = bitReader.readUnsigned(4);
|
2022-09-27 18:26:53 +00:00
|
|
|
pXSector->offCeilZ = bitReader.readSigned(32) * zmaptoworld;
|
|
|
|
pXSector->onCeilZ = bitReader.readSigned(32) * zmaptoworld;
|
|
|
|
pXSector->offFloorZ = bitReader.readSigned(32) * zmaptoworld;
|
|
|
|
pXSector->onFloorZ = bitReader.readSigned(32) * zmaptoworld;
|
2021-12-29 19:45:55 +00:00
|
|
|
/*pXSector->marker0 =*/ bitReader.readUnsigned(16);
|
|
|
|
/*pXSector->marker1 =*/ bitReader.readUnsigned(16);
|
|
|
|
pXSector->Crush = bitReader.readUnsigned(1);
|
|
|
|
pSector->ceilingxpan_ += bitReader.readUnsigned(8) / 256.f;
|
|
|
|
pSector->ceilingypan_ += bitReader.readUnsigned(8) / 256.f;
|
|
|
|
pSector->floorxpan_ += bitReader.readUnsigned(8) / 256.f;
|
|
|
|
pXSector->damageType = bitReader.readUnsigned(3);
|
|
|
|
pXSector->floorpal = bitReader.readUnsigned(4);
|
|
|
|
pSector->floorypan_ += bitReader.readUnsigned(8) / 256.f;
|
|
|
|
pXSector->locked = bitReader.readUnsigned(1);
|
|
|
|
pXSector->windVel = bitReader.readUnsigned(10);
|
2022-09-19 09:04:09 +00:00
|
|
|
pXSector->windAng = mapangle(bitReader.readUnsigned(11));
|
2021-12-29 19:45:55 +00:00
|
|
|
pXSector->windAlways = bitReader.readUnsigned(1);
|
|
|
|
pXSector->dudeLockout = bitReader.readUnsigned(1);
|
|
|
|
pXSector->bobTheta = bitReader.readUnsigned(11);
|
|
|
|
pXSector->bobZRange = bitReader.readUnsigned(5);
|
|
|
|
pXSector->bobSpeed = bitReader.readSigned(12);
|
|
|
|
pXSector->bobAlways = bitReader.readUnsigned(1);
|
|
|
|
pXSector->bobFloor = bitReader.readUnsigned(1);
|
|
|
|
pXSector->bobCeiling = bitReader.readUnsigned(1);
|
|
|
|
pXSector->bobRotate = bitReader.readUnsigned(1);
|
|
|
|
pXSector->busy = IntToFixed(pXSector->state);
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (unsigned i = 0; i < wall.Size(); i++)
|
|
|
|
{
|
|
|
|
walltype* pWall = &wall[i];
|
|
|
|
walltypedisk load;
|
|
|
|
fr.Read(&load, sizeof(walltypedisk));
|
|
|
|
if (encrypted)
|
|
|
|
{
|
|
|
|
dbCrypt((char*)&load, sizeof(walltypedisk), (gMapRev * sizeof(sectortypedisk)) | 0x7474614d);
|
|
|
|
}
|
2022-01-27 17:03:59 +00:00
|
|
|
int x = LittleLong(load.x);
|
|
|
|
int y = LittleLong(load.y);
|
2022-02-03 00:02:48 +00:00
|
|
|
pWall->setPosFromMap(x, y);
|
2021-12-29 19:45:55 +00:00
|
|
|
pWall->point2 = LittleShort(load.point2);
|
|
|
|
pWall->nextwall = LittleShort(load.nextwall);
|
|
|
|
pWall->nextsector = LittleShort(load.nextsector);
|
|
|
|
pWall->cstat = EWallFlags::FromInt(LittleShort(load.cstat));
|
|
|
|
pWall->picnum = EWallFlags::FromInt(LittleShort(load.picnum));
|
|
|
|
pWall->overpicnum = LittleShort(load.overpicnum);
|
|
|
|
pWall->type = LittleShort(load.type);
|
|
|
|
pWall->hitag = LittleShort(load.hitag);
|
|
|
|
pWall->extra = LittleShort(load.extra);
|
|
|
|
pWall->shade = load.shade;
|
|
|
|
pWall->pal = load.pal;
|
|
|
|
pWall->xrepeat = load.xrepeat;
|
|
|
|
pWall->xpan_ = load.xpanning;
|
|
|
|
pWall->yrepeat = load.yrepeat;
|
|
|
|
pWall->ypan_ = load.ypanning;
|
|
|
|
|
|
|
|
if (pWall->extra > 0)
|
|
|
|
{
|
2022-08-04 17:13:55 +00:00
|
|
|
uint8_t pBuffer[nXWallSize];
|
2021-12-29 19:45:55 +00:00
|
|
|
pWall->allocX();
|
|
|
|
XWALL* pXWall = &pWall->xw();
|
|
|
|
int nCount;
|
|
|
|
if (!encrypted)
|
|
|
|
{
|
|
|
|
nCount = nXWallSize;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
nCount = xheader.numxwalls;
|
|
|
|
}
|
|
|
|
assert(nCount <= nXWallSize);
|
|
|
|
fr.Read(pBuffer, nCount);
|
|
|
|
BitReader bitReader(pBuffer, nCount);
|
|
|
|
/*pXWall->reference =*/ bitReader.readSigned(14);
|
|
|
|
pXWall->state = bitReader.readUnsigned(1);
|
|
|
|
pXWall->busy = bitReader.readUnsigned(17);
|
|
|
|
pXWall->data = bitReader.readSigned(16);
|
|
|
|
pXWall->txID = bitReader.readUnsigned(10);
|
|
|
|
bitReader.readUnsigned(6);
|
|
|
|
pXWall->rxID = bitReader.readUnsigned(10);
|
|
|
|
pXWall->command = bitReader.readUnsigned(8);
|
|
|
|
pXWall->triggerOn = bitReader.readUnsigned(1);
|
|
|
|
pXWall->triggerOff = bitReader.readUnsigned(1);
|
|
|
|
pXWall->busyTime = bitReader.readUnsigned(12);
|
|
|
|
pXWall->waitTime = bitReader.readUnsigned(12);
|
|
|
|
pXWall->restState = bitReader.readUnsigned(1);
|
|
|
|
pXWall->interruptable = bitReader.readUnsigned(1);
|
|
|
|
pXWall->panAlways = bitReader.readUnsigned(1);
|
2021-12-30 10:26:39 +00:00
|
|
|
pXWall->panVel.X = bitReader.readSigned(8);
|
2021-12-30 10:27:28 +00:00
|
|
|
pXWall->panVel.Y = bitReader.readSigned(8);
|
2021-12-29 19:45:55 +00:00
|
|
|
pXWall->decoupled = bitReader.readUnsigned(1);
|
|
|
|
pXWall->triggerOnce = bitReader.readUnsigned(1);
|
|
|
|
pXWall->isTriggered = bitReader.readUnsigned(1);
|
|
|
|
pXWall->key = bitReader.readUnsigned(3);
|
|
|
|
pXWall->triggerPush = bitReader.readUnsigned(1);
|
|
|
|
pXWall->triggerVector = bitReader.readUnsigned(1);
|
|
|
|
pXWall->triggerTouch = bitReader.readUnsigned(1);
|
|
|
|
bitReader.readUnsigned(2);
|
|
|
|
pWall->xpan_ += bitReader.readUnsigned(8) / 256.f;
|
|
|
|
pWall->ypan_ += bitReader.readUnsigned(8) / 256.f;
|
|
|
|
pXWall->locked = bitReader.readUnsigned(1);
|
|
|
|
pXWall->dudeLockout = bitReader.readUnsigned(1);
|
|
|
|
bitReader.readUnsigned(4);
|
|
|
|
bitReader.readUnsigned(32);
|
|
|
|
pXWall->busy = IntToFixed(pXWall->state);
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
leveltimer = mapHeader.numsprites;
|
|
|
|
sprites.sprites.Resize(mapHeader.numsprites);
|
|
|
|
sprites.xspr.Resize(mapHeader.numsprites);
|
|
|
|
for (int i = 0; i < mapHeader.numsprites; i++)
|
|
|
|
{
|
|
|
|
spritetypedisk load;
|
|
|
|
fr.Read(&load, sizeof(spritetypedisk));
|
|
|
|
if (encrypted) // What were these people thinking? :(
|
|
|
|
{
|
|
|
|
dbCrypt((char*)&load, sizeof(spritetypedisk), (gMapRev * sizeof(spritetypedisk)) | 0x7474614d);
|
|
|
|
}
|
|
|
|
auto pSprite = &sprites.sprites[i];
|
2021-12-30 16:55:17 +00:00
|
|
|
*pSprite = {};
|
2022-02-02 18:49:57 +00:00
|
|
|
pSprite->SetMapPos(LittleLong(load.x), LittleLong(load.y), LittleLong(load.z));
|
2021-12-29 19:45:55 +00:00
|
|
|
pSprite->cstat = ESpriteFlags::FromInt(LittleShort(load.cstat));
|
|
|
|
pSprite->picnum = LittleShort(load.picnum);
|
|
|
|
int secno = LittleShort(load.sectnum);
|
|
|
|
pSprite->statnum = LittleShort(load.statnum);
|
2022-09-11 11:47:47 +00:00
|
|
|
pSprite->angle = mapangle(LittleShort(load.ang));
|
2022-05-23 22:30:41 +00:00
|
|
|
pSprite->intowner = LittleShort(load.owner);
|
2022-09-01 15:17:06 +00:00
|
|
|
pSprite->xint = LittleShort(load.index);
|
|
|
|
pSprite->yint = LittleShort(load.yvel);
|
2021-12-29 19:45:55 +00:00
|
|
|
pSprite->inittype = LittleShort(load.inittype);
|
|
|
|
pSprite->type = LittleShort(load.type);
|
|
|
|
pSprite->flags = LittleShort(load.hitag);
|
|
|
|
pSprite->extra = LittleShort(load.extra);
|
|
|
|
pSprite->pal = load.pal;
|
|
|
|
pSprite->clipdist = load.clipdist;
|
|
|
|
pSprite->xrepeat = load.xrepeat;
|
|
|
|
pSprite->yrepeat = load.yrepeat;
|
|
|
|
pSprite->xoffset = load.xoffset;
|
|
|
|
pSprite->yoffset = load.yoffset;
|
|
|
|
pSprite->detail = load.detail;
|
|
|
|
pSprite->shade = load.shade;
|
|
|
|
pSprite->blend = 0;
|
|
|
|
validateSprite(*pSprite, secno, i);
|
|
|
|
|
|
|
|
if (pSprite->extra > 0)
|
|
|
|
{
|
2022-08-04 17:13:55 +00:00
|
|
|
uint8_t pBuffer[nXSpriteSize];
|
2021-12-29 19:45:55 +00:00
|
|
|
XSPRITE* pXSprite = &sprites.xspr[i];
|
|
|
|
*pXSprite = {};
|
|
|
|
int nCount;
|
|
|
|
if (!encrypted)
|
|
|
|
{
|
|
|
|
nCount = nXSpriteSize;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
nCount = xheader.numxsprites;
|
|
|
|
}
|
|
|
|
assert(nCount <= nXSpriteSize);
|
|
|
|
fr.Read(pBuffer, nCount);
|
|
|
|
BitReader bitReader(pBuffer, nCount);
|
|
|
|
/*pXSprite->reference =*/ bitReader.readSigned(14);
|
|
|
|
pXSprite->state = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->busy = bitReader.readUnsigned(17);
|
|
|
|
pXSprite->txID = bitReader.readUnsigned(10);
|
|
|
|
pXSprite->rxID = bitReader.readUnsigned(10);
|
|
|
|
pXSprite->command = bitReader.readUnsigned(8);
|
|
|
|
pXSprite->triggerOn = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->triggerOff = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->wave = bitReader.readUnsigned(2);
|
|
|
|
pXSprite->busyTime = bitReader.readUnsigned(12);
|
|
|
|
pXSprite->waitTime = bitReader.readUnsigned(12);
|
|
|
|
pXSprite->restState = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->Interrutable = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->unused1 = bitReader.readUnsigned(2);
|
|
|
|
pXSprite->respawnPending = bitReader.readUnsigned(2);
|
|
|
|
pXSprite->unused2 = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->lT = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->dropMsg = bitReader.readUnsigned(8);
|
|
|
|
pXSprite->Decoupled = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->triggerOnce = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->isTriggered = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->key = bitReader.readUnsigned(3);
|
|
|
|
pXSprite->Push = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->Vector = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->Impact = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->Pickup = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->Touch = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->Sight = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->Proximity = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->unused3 = bitReader.readUnsigned(2);
|
|
|
|
pXSprite->lSkill = bitReader.readUnsigned(5);
|
|
|
|
pXSprite->lS = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->lB = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->lC = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->DudeLockout = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->data1 = bitReader.readSigned(16);
|
|
|
|
pXSprite->data2 = bitReader.readSigned(16);
|
|
|
|
pXSprite->data3 = bitReader.readSigned(16);
|
2022-09-11 11:47:47 +00:00
|
|
|
pXSprite->goalAng = mapangle(bitReader.readUnsigned(11));
|
2021-12-29 19:45:55 +00:00
|
|
|
pXSprite->dodgeDir = bitReader.readSigned(2);
|
|
|
|
pXSprite->locked = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->medium = bitReader.readUnsigned(2);
|
|
|
|
pXSprite->respawn = bitReader.readUnsigned(2);
|
|
|
|
pXSprite->data4 = bitReader.readUnsigned(16);
|
|
|
|
pXSprite->unused4 = bitReader.readUnsigned(6);
|
|
|
|
pXSprite->lockMsg = bitReader.readUnsigned(8);
|
|
|
|
pXSprite->health = bitReader.readUnsigned(12);
|
|
|
|
pXSprite->dudeDeaf = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->dudeAmbush = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->dudeGuard = bitReader.readUnsigned(1);
|
|
|
|
pXSprite->dudeFlag4 = bitReader.readUnsigned(1);
|
|
|
|
/*pXSprite->target_i = */ bitReader.readSigned(16);
|
2022-08-22 16:31:23 +00:00
|
|
|
int tx = bitReader.readSigned(32);
|
|
|
|
int ty = bitReader.readSigned(32);
|
|
|
|
int tz = bitReader.readSigned(32);
|
2022-08-22 16:31:45 +00:00
|
|
|
pXSprite->TargetPos = {tx * maptoworld, ty * maptoworld, tz * zmaptoworld };
|
2021-12-29 19:45:55 +00:00
|
|
|
pXSprite->burnTime = bitReader.readUnsigned(16);
|
|
|
|
/*pXSprite->burnSource =*/ bitReader.readSigned(16);
|
|
|
|
pXSprite->height = bitReader.readUnsigned(16);
|
|
|
|
pXSprite->stateTimer = bitReader.readUnsigned(16);
|
|
|
|
pXSprite->aiState = NULL;
|
|
|
|
bitReader.skipBits(32);
|
|
|
|
pXSprite->busy = IntToFixed(pXSprite->state);
|
|
|
|
if (!encrypted) {
|
|
|
|
pXSprite->lT |= pXSprite->lB;
|
|
|
|
}
|
2019-10-07 19:29:52 +00:00
|
|
|
|
2021-11-19 19:50:02 +00:00
|
|
|
#ifdef NOONE_EXTENSIONS
|
2021-12-29 19:45:55 +00:00
|
|
|
// indicate if the map requires modern features to work properly
|
|
|
|
// for maps wich created in different editors (include vanilla MAPEDIT) or in PMAPEDIT version below than BETA13
|
|
|
|
if (!gModernMap && pXSprite->rxID == kChannelMapModernize && pXSprite->rxID == pXSprite->txID && pXSprite->command == kCmdModernFeaturesEnable)
|
|
|
|
gModernMap = true;
|
2021-11-19 19:50:02 +00:00
|
|
|
#endif
|
2021-12-29 19:45:55 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
fixSectors();
|
|
|
|
|
|
|
|
unsigned int nCRC = fr.ReadUInt32();
|
|
|
|
|
|
|
|
fr.Seek(0, FileReader::SeekSet);
|
|
|
|
auto buffer = fr.Read();
|
|
|
|
uint8_t md4[16];
|
|
|
|
md4once(buffer.Data(), buffer.Size(), md4);
|
|
|
|
loadMapHack(mapname, md4, sprites);
|
|
|
|
|
|
|
|
if (CalcCRC32(buffer.Data(), buffer.Size() - 4) != nCRC)
|
|
|
|
{
|
|
|
|
I_Error("%s: Map File does not match CRC", mapname.GetChars());
|
|
|
|
}
|
|
|
|
if (pCRC)
|
|
|
|
*pCRC = nCRC;
|
|
|
|
if (encrypted)
|
|
|
|
{
|
|
|
|
if (!(gMattId == 0x7474614d || gMattId == 0x4d617474 || !gMattId))
|
|
|
|
{
|
|
|
|
I_Error("%s: Corrupted Map file", mapname.GetChars());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (gMattId != 0)
|
|
|
|
{
|
|
|
|
I_Error("%s: Corrupted Map file", mapname.GetChars());
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((header.version & 0xff00) == 0x600)
|
|
|
|
{
|
|
|
|
switch (header.version & 0xff)
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
for (auto& sect : sector)
|
|
|
|
{
|
|
|
|
sectortype* pSector = §
|
|
|
|
if (pSector->hasX())
|
|
|
|
{
|
|
|
|
XSECTOR* pXSector = &pSector->xs();
|
|
|
|
pXSector->busyTimeB = pXSector->busyTimeA;
|
|
|
|
if (pXSector->busyTimeA > 0)
|
|
|
|
{
|
|
|
|
if (!pXSector->restState)
|
|
|
|
{
|
|
|
|
pXSector->reTriggerA = 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
pXSector->waitTimeB = pXSector->busyTimeA;
|
|
|
|
pXSector->waitTimeA = 0;
|
|
|
|
pXSector->reTriggerB = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
[[fallthrough]];
|
|
|
|
case 1:
|
|
|
|
for (auto& sect : sector)
|
|
|
|
{
|
|
|
|
sectortype* pSector = §
|
|
|
|
if (pSector->hasX())
|
|
|
|
{
|
|
|
|
XSECTOR* pXSector = &pSector->xs();
|
|
|
|
pXSector->freq >>= 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
[[fallthrough]];
|
|
|
|
case 2:
|
|
|
|
break;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
setWallSectors();
|
|
|
|
hw_CreateSections();
|
|
|
|
sectionGeometry.SetSize(sections.Size());
|
|
|
|
wallbackup = wall;
|
|
|
|
sectorbackup = sector;
|
2022-09-24 16:27:02 +00:00
|
|
|
validateStartSector(mapname.GetChars(), pos, cursect, mapHeader.numsectors, true);
|
2019-09-19 22:42:45 +00:00
|
|
|
}
|
2019-06-29 15:37:04 +00:00
|
|
|
|
2019-09-22 06:39:22 +00:00
|
|
|
|
|
|
|
END_BLD_NS
|
2020-10-11 18:57:20 +00:00
|
|
|
|
2021-12-29 19:33:09 +00:00
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
2020-10-11 18:57:20 +00:00
|
|
|
// only used by the backup loader.
|
2021-12-29 19:33:09 +00:00
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
2022-08-23 21:36:23 +00:00
|
|
|
void qloadboard(const char* filename, uint8_t flags, DVector3* dapos, int16_t* daang)
|
2020-10-11 18:57:20 +00:00
|
|
|
{
|
2021-12-29 19:45:55 +00:00
|
|
|
Blood::BloodSpawnSpriteDef sprites;
|
2022-09-24 16:27:02 +00:00
|
|
|
sectortype* sp;
|
2022-08-23 21:36:23 +00:00
|
|
|
Blood::dbLoadMap(filename, *dapos, daang, &sp, nullptr, sprites);
|
2020-10-11 18:57:20 +00:00
|
|
|
}
|