1200 lines
26 KiB
C++
1200 lines
26 KiB
C++
//========= Copyright Valve Corporation, All rights reserved. ============//
|
|
//
|
|
// Purpose:
|
|
//
|
|
// $NoKeywords: $
|
|
//=============================================================================//
|
|
|
|
#include "stdafx.h"
|
|
#include <io.h>
|
|
#include "hammer.h"
|
|
#include "GlobalFunctions.h"
|
|
#include "MapErrorsDlg.h"
|
|
#include "Options.h"
|
|
#include "MapDoc.h"
|
|
#include "MapEntity.h"
|
|
#include "MapFace.h"
|
|
#include "MapGroup.h"
|
|
#include "MapSolid.h"
|
|
#include "MapStudioModel.h"
|
|
#include "MapWorld.h"
|
|
#include "progdlg.h"
|
|
#include "TextureSystem.h"
|
|
#include "MapDisp.h"
|
|
|
|
// memdbgon must be the last include file in a .cpp file!!!
|
|
#include <tier0/memdbgon.h>
|
|
|
|
#pragma optimize("g", off)
|
|
|
|
#pragma warning(disable: 4748) // buffer overrung with optimizations off - remove if we turn "g" back on
|
|
|
|
#define TEXTURE_NAME_LEN 128
|
|
|
|
// All files are opened in binary, and we want to save CR/LF
|
|
#define ENDLINE "\r\n"
|
|
|
|
|
|
enum
|
|
{
|
|
fileOsError = -1, // big error!
|
|
fileError = -2, // problem
|
|
fileOk = -3, // loaded ok
|
|
fileDone = -4 // got not-my-kind of line
|
|
};
|
|
|
|
|
|
BOOL bSaveVisiblesOnly;
|
|
|
|
|
|
static BOOL bErrors;
|
|
static int nInvalidSolids;
|
|
static CProgressDlg *pProgDlg;
|
|
|
|
static MAPFORMAT MapFormat;
|
|
|
|
static BOOL bStuffed;
|
|
static char szStuffed[255];
|
|
|
|
static UINT uMapVersion = 0;
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
// Input : buf -
|
|
//-----------------------------------------------------------------------------
|
|
static void StuffLine(char * buf)
|
|
{
|
|
Assert(!bStuffed);
|
|
V_strcpy_safe(szStuffed, buf);
|
|
bStuffed = TRUE;
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
// Input : file -
|
|
// buf -
|
|
//-----------------------------------------------------------------------------
|
|
static void GetLine(std::fstream& file, char *buf)
|
|
{
|
|
if(bStuffed)
|
|
{
|
|
if(buf)
|
|
strcpy(buf, szStuffed);
|
|
bStuffed = FALSE;
|
|
return;
|
|
}
|
|
|
|
char szBuf[1024];
|
|
|
|
while(1)
|
|
{
|
|
file >> std::ws;
|
|
file.getline(szBuf, 512);
|
|
if(file.eof())
|
|
return;
|
|
if(!strncmp(szBuf, "//", 2))
|
|
continue;
|
|
file >> std::ws;
|
|
if(buf)
|
|
{
|
|
// char *p = strchr(szBuf, '\n');
|
|
// if(p) p[0] = 0;
|
|
// p = strchr(szBuf, '\r');
|
|
// if(p) p[0] = 0;
|
|
strcpy(buf, szBuf);
|
|
}
|
|
return;
|
|
}
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
// Input : pObject -
|
|
// file -
|
|
// pIntersecting -
|
|
// Output : int
|
|
//-----------------------------------------------------------------------------
|
|
static int SaveSolidChildrenOf(CMapClass *pObject, std::fstream& file, BoundBox *pIntersecting = NULL)
|
|
{
|
|
CMapWorld *pWorld = (CMapWorld*) CMapClass::GetWorldObject(pObject);
|
|
|
|
//
|
|
// If we are only saving visible objects and this object isn't visible, don't save it.
|
|
//
|
|
if (bSaveVisiblesOnly && (pObject != pWorld) && !pObject->IsVisible())
|
|
{
|
|
return fileOk; // not an error - return ok
|
|
}
|
|
|
|
//
|
|
// If we are only saving objects within a particular bounding box and this object isn't, don't save it.
|
|
//
|
|
if (pIntersecting && !pObject->IsIntersectingBox(pIntersecting->bmins, pIntersecting->bmaxs))
|
|
{
|
|
return fileOk;
|
|
}
|
|
|
|
|
|
const CMapObjectList *pChildren = pObject->GetChildren();
|
|
FOR_EACH_OBJ( *pChildren, pos )
|
|
{
|
|
int iRvl = -1;
|
|
CMapClass *pChild = pChildren->Element(pos);
|
|
|
|
if (!pIntersecting || pChild->IsIntersectingBox(pIntersecting->bmins, pIntersecting->bmaxs))
|
|
{
|
|
if (pChild->IsMapClass(MAPCLASS_TYPE(CMapSolid)))
|
|
{
|
|
if (!bSaveVisiblesOnly || pChild->IsVisible())
|
|
{
|
|
iRvl = pChild->SerializeMAP(file, TRUE);
|
|
}
|
|
}
|
|
else if (pChild->IsMapClass(MAPCLASS_TYPE(CMapGroup)))
|
|
{
|
|
iRvl = SaveSolidChildrenOf(pChild, file, pIntersecting);
|
|
}
|
|
|
|
// return error if there is an error
|
|
if (iRvl != -1 && iRvl != fileOk)
|
|
{
|
|
return iRvl;
|
|
}
|
|
}
|
|
}
|
|
|
|
return fileOk; // ok.
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
// Input : pObject -
|
|
// file -
|
|
// pIntersecting -
|
|
// Output : int
|
|
//-----------------------------------------------------------------------------
|
|
static int SaveEntityChildrenOf(CMapClass *pObject, std::fstream& file, BoundBox *pIntersecting)
|
|
{
|
|
CMapWorld *pWorld = (CMapWorld *)CMapClass::GetWorldObject(pObject);
|
|
|
|
if (bSaveVisiblesOnly && pObject != pWorld && !pObject->IsVisible())
|
|
{
|
|
return fileOk; // no error
|
|
}
|
|
|
|
if (pIntersecting && !pObject->IsIntersectingBox(pIntersecting->bmins, pIntersecting->bmaxs))
|
|
{
|
|
return fileOk;
|
|
}
|
|
|
|
|
|
const CMapObjectList *pChildren = pObject->GetChildren();
|
|
FOR_EACH_OBJ( *pChildren, pos )
|
|
{
|
|
int iRvl = -1;
|
|
|
|
CMapClass *pChild = pChildren->Element(pos);
|
|
|
|
if (!pIntersecting || pChild->IsIntersectingBox(pIntersecting->bmins, pIntersecting->bmaxs))
|
|
{
|
|
if (pChild->IsMapClass(MAPCLASS_TYPE(CMapEntity)))
|
|
{
|
|
if (!bSaveVisiblesOnly || pChild->IsVisible())
|
|
{
|
|
iRvl = pChild->SerializeMAP(file, TRUE);
|
|
}
|
|
}
|
|
else if (pChild->IsMapClass(MAPCLASS_TYPE(CMapGroup)))
|
|
{
|
|
iRvl = SaveEntityChildrenOf(pChild, file, pIntersecting);
|
|
}
|
|
|
|
// return error if there is an error
|
|
if (iRvl != -1 && iRvl != fileOk)
|
|
{
|
|
return iRvl;
|
|
}
|
|
}
|
|
}
|
|
|
|
return fileOk; // ok.
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
// Input : *pObject -
|
|
// file -
|
|
// Output : int
|
|
//-----------------------------------------------------------------------------
|
|
static int ReadSolids(CMapClass *pObject, std::fstream& file)
|
|
{
|
|
int nSolids = 0;
|
|
char szBuf[128];
|
|
|
|
while(1)
|
|
{
|
|
GetLine(file, szBuf);
|
|
if(szBuf[0] != '{')
|
|
{
|
|
StuffLine(szBuf);
|
|
break;
|
|
}
|
|
|
|
CMapSolid *pSolid = new CMapSolid;
|
|
int iRvl = pSolid->SerializeMAP(file, FALSE);
|
|
if(iRvl == fileError)
|
|
{
|
|
// delete the solid
|
|
delete pSolid;
|
|
++nInvalidSolids;
|
|
}
|
|
else if(iRvl == fileOsError)
|
|
{
|
|
// big problem
|
|
delete pSolid;
|
|
return fileOsError;
|
|
}
|
|
else
|
|
pObject->AddChild(pSolid);
|
|
|
|
++nSolids;
|
|
}
|
|
|
|
return nSolids;
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
// Input : file -
|
|
// Output : int
|
|
//-----------------------------------------------------------------------------
|
|
static int PeekChar(std::fstream& file)
|
|
{
|
|
if(bStuffed) // stuffed.. return first char
|
|
return szStuffed[0];
|
|
char szBuf[1024];
|
|
szBuf[0] = 0;
|
|
// get next line
|
|
GetLine(file, szBuf);
|
|
|
|
// still blank? return eof
|
|
if(szBuf[0] == 0)
|
|
return EOF;
|
|
|
|
// to get it next call to getline
|
|
StuffLine(szBuf);
|
|
|
|
return szBuf[0];
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Sets the MAP format for saving.
|
|
// Input : mf - MAP format to use when saving.
|
|
//-----------------------------------------------------------------------------
|
|
void SetMapFormat(MAPFORMAT mf)
|
|
{
|
|
Assert((mf == mfHalfLife) || (mf == mfHalfLife2));
|
|
MapFormat = mf;
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
// Input : file -
|
|
// fIsStoring -
|
|
// Output : int
|
|
//-----------------------------------------------------------------------------
|
|
int CMapClass::SerializeMAP(std::fstream& file, BOOL fIsStoring)
|
|
{
|
|
// no info stored in MAPs ..
|
|
return fileOk;
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
// Input : file -
|
|
// fIsStoring -
|
|
// Output : int
|
|
//-----------------------------------------------------------------------------
|
|
int CMapFace::SerializeMAP(std::fstream& file, BOOL fIsStoring)
|
|
{
|
|
char szBuf[512];
|
|
|
|
if (fIsStoring)
|
|
{
|
|
char *pszTexture;
|
|
char szTexture[sizeof(texture.texture)+1];
|
|
szTexture[sizeof(texture.texture)] = 0;
|
|
memcpy(szTexture, texture.texture, sizeof texture.texture);
|
|
strlwr(szTexture);
|
|
|
|
if (MapFormat == mfQuake2)
|
|
{
|
|
pszTexture = strstr(szTexture, ".");
|
|
if (pszTexture)
|
|
{
|
|
*pszTexture = 0;
|
|
}
|
|
pszTexture = strstr(szTexture, "textures\\");
|
|
if (pszTexture == NULL)
|
|
{
|
|
pszTexture = szTexture;
|
|
}
|
|
else
|
|
{
|
|
pszTexture += strlen("textures\\");
|
|
}
|
|
}
|
|
else
|
|
{
|
|
pszTexture = szTexture;
|
|
}
|
|
|
|
strupr(szTexture);
|
|
|
|
|
|
//
|
|
// Reverse the slashes -- thank you id.
|
|
//
|
|
for (int i = strlen(pszTexture) - 1; i >= 0; i--)
|
|
{
|
|
if (pszTexture[i] == '\\')
|
|
pszTexture[i] = '/';
|
|
}
|
|
|
|
//
|
|
// Convert the plane points to integers.
|
|
//
|
|
for (int nPlane = 0; nPlane < 3; nPlane++)
|
|
{
|
|
plane.planepts[nPlane][0] = V_rint(plane.planepts[nPlane][0]);
|
|
plane.planepts[nPlane][1] = V_rint(plane.planepts[nPlane][1]);
|
|
plane.planepts[nPlane][2] = V_rint(plane.planepts[nPlane][2]);
|
|
}
|
|
|
|
//
|
|
// Check for duplicate plane points. All three plane points must be unique
|
|
// or it isn't a valid plane. Try to fix it if it isn't valid.
|
|
//
|
|
if (!CheckFace())
|
|
{
|
|
Fix();
|
|
}
|
|
|
|
sprintf(szBuf,
|
|
"( %.0f %.0f %.0f ) ( %.0f %.0f %.0f ) ( %.0f %.0f %.0f ) "
|
|
"%s "
|
|
"[ %g %g %g %g ] "
|
|
"[ %g %g %g %g ] "
|
|
"%g %g %g ",
|
|
|
|
plane.planepts[0][0], plane.planepts[0][1], plane.planepts[0][2],
|
|
plane.planepts[1][0], plane.planepts[1][1], plane.planepts[1][2],
|
|
plane.planepts[2][0], plane.planepts[2][1], plane.planepts[2][2],
|
|
|
|
pszTexture,
|
|
|
|
(double)texture.UAxis[0], (double)texture.UAxis[1], (double)texture.UAxis[2], (double)texture.UAxis[3],
|
|
(double)texture.VAxis[0], (double)texture.VAxis[1], (double)texture.VAxis[2], (double)texture.VAxis[3],
|
|
|
|
(double)texture.rotate,
|
|
(double)texture.scale[0],
|
|
(double)texture.scale[1]);
|
|
|
|
file << szBuf << ENDLINE;
|
|
|
|
return fileOk;
|
|
}
|
|
else
|
|
{
|
|
// load the plane
|
|
GetLine(file, szBuf);
|
|
|
|
if(szBuf[0] != '(')
|
|
{
|
|
StuffLine(szBuf);
|
|
return fileDone;
|
|
}
|
|
|
|
char szTexName[TEXTURE_NAME_LEN];
|
|
DWORD q2contents;
|
|
DWORD q2surface;
|
|
int nLightmapScale;
|
|
int nDummy;
|
|
int nRead;
|
|
|
|
if( uMapVersion >= 340 )
|
|
{
|
|
COMPILE_TIME_ASSERT( ARRAYSIZE(szTexName) == 128 );
|
|
nRead = sscanf(szBuf,
|
|
"( %f %f %f ) ( %f %f %f ) ( %f %f %f ) "
|
|
"%127s "
|
|
"[ %f %f %f %f ] "
|
|
"[ %f %f %f %f ] "
|
|
"%f %f %f "
|
|
"%u %u %u",
|
|
|
|
&plane.planepts[0][0], &plane.planepts[0][1], &plane.planepts[0][2],
|
|
&plane.planepts[1][0], &plane.planepts[1][1], &plane.planepts[1][2],
|
|
&plane.planepts[2][0], &plane.planepts[2][1], &plane.planepts[2][2],
|
|
|
|
szTexName,
|
|
|
|
&texture.UAxis[0], &texture.UAxis[1], &texture.UAxis[2], &texture.UAxis[3],
|
|
&texture.VAxis[0], &texture.VAxis[1], &texture.VAxis[2], &texture.VAxis[3],
|
|
|
|
&texture.rotate,
|
|
&texture.scale[0],
|
|
&texture.scale[1],
|
|
|
|
&q2contents,
|
|
&q2surface,
|
|
&nLightmapScale);
|
|
// Guarantee null-termination to keep /analyze happy.
|
|
szTexName[ ARRAYSIZE(szTexName) - 1 ] = 0;
|
|
|
|
if (nRead < 21)
|
|
{
|
|
bErrors = TRUE;
|
|
}
|
|
else if (nRead == 24)
|
|
{
|
|
// got q2 values - set them here
|
|
texture.q2contents = q2contents;
|
|
texture.q2surface = q2surface;
|
|
texture.nLightmapScale = nLightmapScale;
|
|
if (texture.nLightmapScale == 0)
|
|
{
|
|
texture.nLightmapScale = g_pGameConfig->GetDefaultLightmapScale();
|
|
}
|
|
}
|
|
|
|
//
|
|
// very cheesy HACK!!! -- this will be better when we have chunks
|
|
//
|
|
if( uMapVersion <= 350 )
|
|
{
|
|
if( ( file.peek() != '(' ) && ( file.peek() != '}' ) )
|
|
{
|
|
EditDispHandle_t handle = EditDispMgr()->Create();
|
|
SetDisp( handle );
|
|
|
|
CMapDisp *pDisp = EditDispMgr()->GetDisp( handle );
|
|
pDisp->SerializedLoadMAP( file, this, uMapVersion );
|
|
}
|
|
}
|
|
}
|
|
else if (uMapVersion >= 220 )
|
|
{
|
|
COMPILE_TIME_ASSERT( ARRAYSIZE(szTexName) == 128 );
|
|
nRead = sscanf(szBuf,
|
|
"( %f %f %f ) ( %f %f %f ) ( %f %f %f ) "
|
|
"%127s "
|
|
"[ %f %f %f %f ] "
|
|
"[ %f %f %f %f ] "
|
|
"%f %f %f "
|
|
"%u %u %u",
|
|
|
|
&plane.planepts[0][0], &plane.planepts[0][1], &plane.planepts[0][2],
|
|
&plane.planepts[1][0], &plane.planepts[1][1], &plane.planepts[1][2],
|
|
&plane.planepts[2][0], &plane.planepts[2][1], &plane.planepts[2][2],
|
|
|
|
szTexName,
|
|
|
|
&texture.UAxis[0], &texture.UAxis[1], &texture.UAxis[2], &texture.UAxis[3],
|
|
&texture.VAxis[0], &texture.VAxis[1], &texture.VAxis[2], &texture.VAxis[3],
|
|
|
|
&texture.rotate,
|
|
&texture.scale[0],
|
|
&texture.scale[1],
|
|
|
|
&q2contents,
|
|
&q2surface,
|
|
&nDummy); // Pre-340 didn't have lightmap scale.
|
|
// Guarantee null-termination to keep /analyze happy.
|
|
szTexName[ ARRAYSIZE(szTexName) - 1 ] = 0;
|
|
|
|
if (nRead < 21)
|
|
{
|
|
bErrors = TRUE;
|
|
}
|
|
else if (nRead == 24)
|
|
{
|
|
// got q2 values - set them here
|
|
texture.q2contents = q2contents;
|
|
texture.q2surface = q2surface;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
COMPILE_TIME_ASSERT( ARRAYSIZE(szTexName) == 128 );
|
|
nRead = sscanf(szBuf,
|
|
"( %f %f %f ) ( %f %f %f ) ( %f %f %f ) "
|
|
"%127s "
|
|
"%f %f %f "
|
|
"%f %f %u %u %u",
|
|
|
|
&plane.planepts[0][0], &plane.planepts[0][1], &plane.planepts[0][2],
|
|
&plane.planepts[1][0], &plane.planepts[1][1], &plane.planepts[1][2],
|
|
&plane.planepts[2][0], &plane.planepts[2][1], &plane.planepts[2][2],
|
|
|
|
szTexName,
|
|
|
|
&texture.UAxis[3],
|
|
&texture.VAxis[3],
|
|
&texture.rotate,
|
|
&texture.scale[0],
|
|
&texture.scale[1],
|
|
|
|
&q2contents,
|
|
&q2surface,
|
|
&nDummy); // Pre-340 didn't have lightmap scale.
|
|
// Guarantee null-termination to keep /analyze happy.
|
|
szTexName[ ARRAYSIZE(szTexName) - 1 ] = 0;
|
|
|
|
if (nRead < 15)
|
|
{
|
|
bErrors = TRUE;
|
|
}
|
|
else if (nRead == 18)
|
|
{
|
|
// got q2 values - set them here
|
|
texture.q2contents = q2contents;
|
|
texture.q2surface = q2surface;
|
|
}
|
|
}
|
|
|
|
if (g_pGameConfig->GetTextureFormat() != tfVMT)
|
|
{
|
|
// reverse the slashes -- thank you id
|
|
for (int i = strlen(szTexName) - 1; i >= 0; i--)
|
|
{
|
|
if (szTexName[i] == '/')
|
|
szTexName[i] = '\\';
|
|
}
|
|
}
|
|
|
|
SetTexture(szTexName);
|
|
}
|
|
|
|
if (file.fail())
|
|
{
|
|
return fileOsError;
|
|
}
|
|
return fileOk;
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
// Input : file -
|
|
// fIsStoring -
|
|
// Output : int
|
|
//-----------------------------------------------------------------------------
|
|
int MDkeyvalue::SerializeMAP(std::fstream& file, BOOL fIsStoring)
|
|
{
|
|
// load/save a keyvalue
|
|
char szBuf[1024];
|
|
|
|
if(fIsStoring)
|
|
{
|
|
// save a keyvalue
|
|
sprintf( szBuf,
|
|
"\"%s\" \"%s\"",
|
|
|
|
Key(), Value() );
|
|
|
|
file << szBuf << ENDLINE;
|
|
}
|
|
else
|
|
{
|
|
GetLine(file, szBuf);
|
|
if(szBuf[0] != '\"')
|
|
{
|
|
StuffLine(szBuf);
|
|
return fileDone;
|
|
}
|
|
char *p = strchr(szBuf, '\"');
|
|
p = strchr(p+1, '\"');
|
|
if(!p)
|
|
return fileError;
|
|
p[0] = 0;
|
|
V_strcpy_safe(szKey, szBuf+1);
|
|
|
|
// advance to start of value string
|
|
p = strchr(p+1, '\"');
|
|
if(!p)
|
|
return fileError;
|
|
// ocpy in value
|
|
strcpy(szValue, p+1);
|
|
// kill trailing "
|
|
p = strchr(szValue, '\"');
|
|
if(!p)
|
|
return fileError;
|
|
p[0] = 0;
|
|
}
|
|
|
|
return file.fail() ? fileOsError : fileOk;
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
// Input : file -
|
|
// fIsStoring -
|
|
// Output : int
|
|
//-----------------------------------------------------------------------------
|
|
int CMapSolid::SerializeMAP(std::fstream& file, BOOL fIsStoring)
|
|
{
|
|
CMapClass::SerializeMAP(file, fIsStoring);
|
|
|
|
// load/save a brush
|
|
if (fIsStoring)
|
|
{
|
|
// save the brush
|
|
file << "{" << ENDLINE;
|
|
|
|
// serialize the Faces
|
|
int nFaces = Faces.GetCount();
|
|
for(int i = 0; i < nFaces; i++)
|
|
{
|
|
if(!Faces[i].Points)
|
|
continue;
|
|
if(Faces[i].SerializeMAP(file, fIsStoring) == fileError)
|
|
return fileError;
|
|
}
|
|
|
|
// terminator
|
|
file << "}" << ENDLINE;
|
|
}
|
|
else
|
|
{
|
|
// caller skipped delimiter
|
|
Faces.SetCount(0);
|
|
|
|
// read Faces
|
|
for(int i = 0; ; i++)
|
|
{
|
|
// extract plane
|
|
if (Faces[i].SerializeMAP(file, fIsStoring) == fileDone)
|
|
{
|
|
// when fileDone is returned, no face was loaded
|
|
break;
|
|
}
|
|
|
|
Faces[i].CalcPlane();
|
|
}
|
|
|
|
GetLine(file, NULL); // ignore line
|
|
|
|
if (!file.fail())
|
|
{
|
|
//
|
|
// Create the solid using the planes that were read from the MAP file.
|
|
//
|
|
if (CreateFromPlanes() == FALSE)
|
|
{
|
|
bErrors = TRUE;
|
|
return(fileError);
|
|
}
|
|
|
|
//
|
|
// If we are reading from an old map file, the texture axes will need to be set up.
|
|
// Leave the rotation and shifts alone; they were read from the MAP file.
|
|
//
|
|
if (uMapVersion < 220)
|
|
{
|
|
InitializeTextureAxes(TEXTURE_ALIGN_QUAKE, INIT_TEXTURE_AXES | INIT_TEXTURE_FORCE);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
return(fileOsError);
|
|
}
|
|
|
|
CalcBounds();
|
|
|
|
//
|
|
// Set solid type based on texture name.
|
|
//
|
|
m_eSolidType = HL1SolidTypeFromTextureName(Faces[0].texture.texture);
|
|
}
|
|
|
|
return(file.fail() ? fileOsError : fileOk);
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
// Input : file -
|
|
// fIsStoring -
|
|
// Output : int
|
|
//-----------------------------------------------------------------------------
|
|
int CMapEntity::SerializeMAP(std::fstream &file, BOOL fIsStoring)
|
|
{
|
|
CMapClass::SerializeMAP(file, fIsStoring);
|
|
|
|
// load/save an object
|
|
if (fIsStoring)
|
|
{
|
|
//
|
|
// If it's a solidentity but it doesn't have any solids,
|
|
// don't save it.
|
|
//
|
|
if (!IsPlaceholder() && !m_Children.Count())
|
|
{
|
|
return(fileOk);
|
|
}
|
|
|
|
//
|
|
// Save it.
|
|
//
|
|
file << "{" << ENDLINE;
|
|
|
|
//
|
|
// Save keyvalues & other data.
|
|
//
|
|
CEditGameClass::SerializeMAP(file, fIsStoring);
|
|
|
|
//
|
|
// If this is a placeholder and either has no class or is not a solid class,
|
|
// save our origin.
|
|
//
|
|
if (IsPlaceholder() && (!IsClass() || !IsSolidClass()))
|
|
{
|
|
MDkeyvalue tmpkv;
|
|
strcpy(tmpkv.szKey, "origin");
|
|
|
|
Vector Origin;
|
|
GetOrigin(Origin);
|
|
sprintf(tmpkv.szValue, "%.0f %.0f %.0f", Origin[0], Origin[1], Origin[2]);
|
|
tmpkv.SerializeMAP(file, fIsStoring);
|
|
}
|
|
|
|
if (!(IsPlaceholder()))
|
|
{
|
|
SaveSolidChildrenOf(this, file);
|
|
}
|
|
|
|
file << "}" << ENDLINE;
|
|
}
|
|
else
|
|
{
|
|
// load keyvalues
|
|
CEditGameClass::SerializeMAP(file, fIsStoring);
|
|
|
|
// solids
|
|
if (!ReadSolids(this, file))
|
|
{
|
|
flags |= flagPlaceholder;
|
|
}
|
|
|
|
// skip delimiter
|
|
GetLine(file, NULL);
|
|
}
|
|
|
|
return file.fail() ? fileOsError : fileOk;
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
// Input : file -
|
|
// fIsStoring -
|
|
// Output : int
|
|
//-----------------------------------------------------------------------------
|
|
int CEditGameClass::SerializeMAP(std::fstream& file, BOOL fIsStoring)
|
|
{
|
|
int iRvl;
|
|
|
|
if (!fIsStoring)
|
|
{
|
|
// loading
|
|
if (PeekChar(file) == '\"')
|
|
{
|
|
// read kv pairs
|
|
MDkeyvalue newkv;
|
|
while (1)
|
|
{
|
|
if (newkv.SerializeMAP(file, fIsStoring) != fileOk)
|
|
{
|
|
// fileDone means the keyvalue was not loaded
|
|
break;
|
|
}
|
|
|
|
if (!strcmp(newkv.szKey, "classname"))
|
|
{
|
|
m_KeyValues.SetValue(newkv.szKey, newkv.szValue);
|
|
}
|
|
else if (!strcmp(newkv.szKey, "angle"))
|
|
{
|
|
ImportAngle(atoi(newkv.szValue));
|
|
}
|
|
else if (strcmp(newkv.szKey, "wad"))
|
|
{
|
|
//
|
|
// All other keys are simply added to the keyvalue list.
|
|
//
|
|
m_KeyValues.SetValue(newkv.szKey, newkv.szValue);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// save keyvalues
|
|
MDkeyvalue tmpkv;
|
|
|
|
if (GetKeyValue("classname") == NULL)
|
|
{
|
|
tmpkv.Set("classname", m_szClass);
|
|
tmpkv.SerializeMAP(file, fIsStoring);
|
|
}
|
|
|
|
//
|
|
// Determine whether we have a game data class. This will help us decide which keys
|
|
// to write.
|
|
//
|
|
GDclass *pGameDataClass = NULL;
|
|
if (pGD != NULL)
|
|
{
|
|
pGameDataClass = pGD->ClassForName(m_szClass);
|
|
}
|
|
|
|
//
|
|
// Consider all the keyvalues in this object for serialization.
|
|
//
|
|
for ( int z=m_KeyValues.GetFirst(); z != m_KeyValues.GetInvalidIndex(); z=m_KeyValues.GetNext( z ) )
|
|
{
|
|
MDkeyvalue &KeyValue = m_KeyValues.GetKeyValue(z);
|
|
|
|
iRvl = KeyValue.SerializeMAP(file, fIsStoring);
|
|
if (iRvl != fileOk)
|
|
{
|
|
return(iRvl);
|
|
}
|
|
}
|
|
|
|
//
|
|
// If we have a base class, for each keyvalue in the class definition, write out all keys
|
|
// that are not present in the object and whose defaults are nonzero in the class definition.
|
|
//
|
|
if (pGameDataClass != NULL)
|
|
{
|
|
//
|
|
// For each variable from the base class...
|
|
//
|
|
int nVariableCount = pGameDataClass->GetVariableCount();
|
|
for (int i = 0; i < nVariableCount; i++)
|
|
{
|
|
GDinputvariable *pVar = pGameDataClass->GetVariableAt(i);
|
|
Assert(pVar != NULL);
|
|
|
|
if (pVar != NULL)
|
|
{
|
|
int iIndex;
|
|
MDkeyvalue *pKey;
|
|
LPCTSTR p = m_KeyValues.GetValue(pVar->GetName(), &iIndex);
|
|
|
|
//
|
|
// If the variable is not present in this object, write out the default value.
|
|
//
|
|
if (p == NULL)
|
|
{
|
|
pKey = &tmpkv;
|
|
pVar->ResetDefaults();
|
|
pVar->ToKeyValue(pKey);
|
|
|
|
//
|
|
// Only write the key value if it is non-zero.
|
|
//
|
|
if ((pKey->szKey[0] != 0) && (pKey->szValue[0] != 0) && (stricmp(pKey->szValue, "0")))
|
|
{
|
|
iRvl = pKey->SerializeMAP(file, fIsStoring);
|
|
if (iRvl != fileOk)
|
|
{
|
|
return(iRvl);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return(file.fail() ? fileOsError : fileOk);
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
// Input : file -
|
|
// fIsStoring -
|
|
// pIntersecting -
|
|
// Output : int
|
|
//-----------------------------------------------------------------------------
|
|
int CMapWorld::SerializeMAP(std::fstream &file, BOOL fIsStoring, BoundBox *pIntersecting)
|
|
{
|
|
int iRvl;
|
|
|
|
bStuffed = FALSE;
|
|
bErrors = FALSE;
|
|
nInvalidSolids = 0;
|
|
|
|
// load/save a world
|
|
if (fIsStoring)
|
|
{
|
|
file << "{" << ENDLINE;
|
|
|
|
// save worldobject
|
|
CEditGameClass::SerializeMAP(file, fIsStoring);
|
|
|
|
if (MapFormat != mfQuake2)
|
|
{
|
|
MDkeyvalue tmpkv;
|
|
|
|
strcpy(tmpkv.szKey, "mapversion");
|
|
strcpy(tmpkv.szValue, "360");
|
|
tmpkv.SerializeMAP(file, fIsStoring);
|
|
|
|
// Save wad file line
|
|
strcpy(tmpkv.szKey, "wad");
|
|
|
|
// copy all texfiles into value
|
|
tmpkv.szValue[0] = 0;
|
|
BOOL bFirst = TRUE;
|
|
int nGraphicsFiles = g_Textures.FilesGetCount();
|
|
for (int i = 0; i < nGraphicsFiles; i++)
|
|
{
|
|
char szFile[MAX_PATH];
|
|
GRAPHICSFILESTRUCT gf;
|
|
|
|
g_Textures.FilesGetInfo(&gf, i);
|
|
|
|
//
|
|
// Don't save WAL files - they're never used.
|
|
//
|
|
if (gf.format != tfWAL)
|
|
{
|
|
//
|
|
// Also make sure this is the right kind of WAD file
|
|
// based on the game we're using.
|
|
//
|
|
if (gf.format == g_pGameConfig->GetTextureFormat())
|
|
{
|
|
//
|
|
// Append this WAD file to the WAD list.
|
|
//
|
|
strcpy(szFile, gf.filename);
|
|
|
|
// dvs: Strip off the path. This crashes VIS and QRAD!!
|
|
/*
|
|
char *pszSlash = strrchr(szFile, '\\');
|
|
if (pszSlash == NULL)
|
|
{
|
|
pszSlash = strrchr(szFile, '/');
|
|
}
|
|
|
|
if (pszSlash != NULL)
|
|
{
|
|
pszSlash++;
|
|
}
|
|
else
|
|
{
|
|
pszSlash = szFile;
|
|
}
|
|
*/
|
|
|
|
char *pszSlash = szFile;
|
|
|
|
// Strip off any drive letter.
|
|
if (pszSlash[1] == ':')
|
|
{
|
|
pszSlash += 2;
|
|
}
|
|
|
|
// WAD names are semicolon delimited.
|
|
if (!bFirst)
|
|
{
|
|
strcat(tmpkv.szValue, ";");
|
|
}
|
|
|
|
strcat(tmpkv.szValue, pszSlash);
|
|
bFirst = FALSE;
|
|
}
|
|
}
|
|
}
|
|
|
|
if ( tmpkv.szValue[0] != '\0' )
|
|
{
|
|
tmpkv.SerializeMAP(file, fIsStoring);
|
|
}
|
|
}
|
|
|
|
//
|
|
// Save the brushes.
|
|
//
|
|
if (SaveSolidChildrenOf(this, file, pIntersecting) == fileOsError)
|
|
{
|
|
goto FatalError;
|
|
}
|
|
|
|
file << "}" << ENDLINE;
|
|
|
|
//
|
|
// Save the entities.
|
|
//
|
|
if (SaveEntityChildrenOf(this, file, pIntersecting) == fileOsError)
|
|
{
|
|
goto FatalError;
|
|
}
|
|
|
|
//
|
|
// Save paths (if paths are visible).
|
|
//
|
|
FOR_EACH_OBJ( m_Paths, pos )
|
|
{
|
|
CMapPath *pPath = m_Paths.Element(pos);
|
|
pPath->SerializeMAP(file, TRUE, pIntersecting);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
pProgDlg = new CProgressDlg;
|
|
pProgDlg->Create();
|
|
pProgDlg->SetStep(1);
|
|
|
|
CString caption;
|
|
caption.LoadString(IDS_LOADINGFILE);
|
|
pProgDlg->SetWindowText(caption);
|
|
|
|
m_Render2DBox.ResetBounds();
|
|
|
|
// load world
|
|
GetLine(file, NULL); // ignore delimiter
|
|
CEditGameClass::SerializeMAP(file, fIsStoring);
|
|
|
|
const char* pszMapVersion;
|
|
|
|
pszMapVersion = m_KeyValues.GetValue("mapversion");
|
|
if (pszMapVersion != NULL)
|
|
{
|
|
uMapVersion = atoi(pszMapVersion);
|
|
}
|
|
else
|
|
{
|
|
uMapVersion = 0;
|
|
}
|
|
|
|
// read solids
|
|
if (ReadSolids(this, file) == fileOsError)
|
|
{
|
|
goto FatalError;
|
|
}
|
|
|
|
// skip end-of-entity marker
|
|
GetLine(file, NULL);
|
|
|
|
char szBuf[128];
|
|
|
|
// read entities
|
|
while (1)
|
|
{
|
|
GetLine(file, szBuf);
|
|
if (szBuf[0] != '{')
|
|
{
|
|
StuffLine(szBuf);
|
|
break;
|
|
}
|
|
|
|
if (PeekChar(file) == EOF)
|
|
{
|
|
break;
|
|
}
|
|
|
|
CMapEntity *pEntity;
|
|
|
|
pEntity = new CMapEntity;
|
|
iRvl = pEntity->SerializeMAP(file, fIsStoring);
|
|
AddChild(pEntity);
|
|
|
|
if (iRvl == fileError)
|
|
{
|
|
bErrors = TRUE;
|
|
}
|
|
else if (iRvl == fileOsError)
|
|
{
|
|
goto FatalError;
|
|
}
|
|
}
|
|
|
|
if (bErrors)
|
|
{
|
|
if (nInvalidSolids)
|
|
{
|
|
CString str;
|
|
str.Format("For your information, %d solids were not loaded\n"
|
|
"due to errors in the file.", nInvalidSolids);
|
|
AfxMessageBox(str);
|
|
}
|
|
else if (AfxMessageBox("There was a problem loading the MAP file. Do you\n"
|
|
"want to view the error report?", MB_YESNO) == IDYES)
|
|
{
|
|
CMapErrorsDlg dlg;
|
|
dlg.DoModal();
|
|
}
|
|
}
|
|
|
|
PostloadWorld();
|
|
|
|
if (g_pGameConfig->GetTextureFormat() == tfVMT)
|
|
{
|
|
// do batch search and replace of textures from trans.txt if it exists.
|
|
char translationFilename[MAX_PATH];
|
|
Q_snprintf( translationFilename, sizeof( translationFilename ), "materials/trans.txt" );
|
|
if( CMapDoc::GetActiveMapDoc() )
|
|
{
|
|
FileHandle_t searchReplaceFP = g_pFileSystem->Open( translationFilename, "r" );
|
|
if( searchReplaceFP )
|
|
{
|
|
CMapDoc::GetActiveMapDoc()->BatchReplaceTextures( searchReplaceFP );
|
|
g_pFileSystem->Close( searchReplaceFP );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (pProgDlg)
|
|
{
|
|
pProgDlg->DestroyWindow();
|
|
delete pProgDlg;
|
|
pProgDlg = NULL;
|
|
}
|
|
|
|
return (bErrors && fIsStoring) ? -1 : 0;
|
|
|
|
FatalError:
|
|
|
|
// OS error.
|
|
CString str;
|
|
str.Format("The OS reported an error %s the file: %s", fIsStoring ? "saving" : "loading", strerror(errno));
|
|
AfxMessageBox(str);
|
|
|
|
if (pProgDlg != NULL)
|
|
{
|
|
pProgDlg->DestroyWindow();
|
|
delete pProgDlg;
|
|
pProgDlg = NULL;
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|