-
Notifications
You must be signed in to change notification settings - Fork 13
/
Copy pathMapFile.h
246 lines (203 loc) · 8.47 KB
/
MapFile.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
/*****************************************************************************
The Dark Mod GPL Source Code
This file is part of the The Dark Mod Source Code, originally based
on the Doom 3 GPL Source Code as published in 2011.
The Dark Mod Source Code 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 3 of the License,
or (at your option) any later version. For details, see LICENSE.TXT.
Project: The Dark Mod (http://www.thedarkmod.com/)
******************************************************************************/
#ifndef __MAPFILE_H__
#define __MAPFILE_H__
/*
===============================================================================
Reads or writes the contents of .map files into a standard internal
format, which can then be moved into private formats for collision
detection, map processing, or editor use.
No validation (duplicate planes, null area brushes, etc) is performed.
There are no limits to the number of any of the elements in maps.
The order of entities, brushes, and sides is maintained.
===============================================================================
*/
const int OLD_MAP_VERSION = 1;
const int CURRENT_MAP_VERSION = 2;
const int DEFAULT_CURVE_SUBDIVISION = 4;
const float DEFAULT_CURVE_MAX_ERROR = 4.0f;
const float DEFAULT_CURVE_MAX_ERROR_CD = 24.0f;
const float DEFAULT_CURVE_MAX_LENGTH = -1.0f;
const float DEFAULT_CURVE_MAX_LENGTH_CD = -1.0f;
class idMapPrimitive {
public:
enum { TYPE_INVALID = -1, TYPE_BRUSH, TYPE_PATCH };
idDict epairs;
idMapPrimitive( void ) { type = TYPE_INVALID; }
virtual ~idMapPrimitive( void ) { }
int GetType( void ) const { return type; }
protected:
int type;
};
class idMapBrushSide {
friend class idMapBrush;
public:
idMapBrushSide( void );
~idMapBrushSide( void ) { }
const char * GetMaterial( void ) const { return material; }
void SetMaterial( const char *p ) { material = p; }
const idPlane & GetPlane( void ) const { return plane; }
void SetPlane( const idPlane &p ) { plane = p; }
void SetTextureMatrix( const idVec3 mat[2] ) { texMat[0] = mat[0]; texMat[1] = mat[1]; }
void GetTextureMatrix( idVec3 &mat1, idVec3 &mat2 ) { mat1 = texMat[0]; mat2 = texMat[1]; }
void GetTextureVectors( idVec4 v[2] ) const;
protected:
idStr material;
idPlane plane;
idVec3 texMat[2];
idVec3 origin;
};
ID_INLINE idMapBrushSide::idMapBrushSide( void ) {
plane.Zero();
texMat[0].Zero();
texMat[1].Zero();
origin.Zero();
}
class idMapBrush : public idMapPrimitive {
public:
idMapBrush( void ) { type = TYPE_BRUSH; sides.Resize( 8, 4 ); }
~idMapBrush( void ) { sides.DeleteContents( true ); }
static idMapBrush * Parse( idLexer &src, const idVec3 &origin, bool newFormat = true, float version = CURRENT_MAP_VERSION );
static idMapBrush * ParseQ3( idLexer &src, const idVec3 &origin );
bool Write( idFile *fp, int primitiveNum, const idVec3 &origin ) const;
int GetNumSides( void ) const { return sides.Num(); }
int AddSide( idMapBrushSide *side ) { return sides.Append( side ); }
idMapBrushSide * GetSide( int i ) const { return sides[i]; }
unsigned int GetGeometryCRC( void ) const;
protected:
int numSides;
idList<idMapBrushSide*> sides;
};
class idMapPatch : public idMapPrimitive, public idSurface_Patch {
public:
idMapPatch( void );
idMapPatch( int maxPatchWidth, int maxPatchHeight );
~idMapPatch( void ) { }
static idMapPatch * Parse( idLexer &src, const idVec3 &origin, bool patchDef3 = true, float version = CURRENT_MAP_VERSION );
bool Write( idFile *fp, int primitiveNum, const idVec3 &origin ) const;
const char * GetMaterial( void ) const { return material; }
void SetMaterial( const char *p ) { material = p; }
int GetHorzSubdivisions( void ) const { return horzSubdivisions; }
int GetVertSubdivisions( void ) const { return vertSubdivisions; }
bool GetExplicitlySubdivided( void ) const { return explicitSubdivisions; }
void SetHorzSubdivisions( int n ) { horzSubdivisions = n; }
void SetVertSubdivisions( int n ) { vertSubdivisions = n; }
void SetExplicitlySubdivided( bool b ) { explicitSubdivisions = b; }
unsigned int GetGeometryCRC( void ) const;
protected:
idStr material;
int horzSubdivisions;
int vertSubdivisions;
bool explicitSubdivisions;
};
ID_INLINE idMapPatch::idMapPatch( void ) {
type = TYPE_PATCH;
horzSubdivisions = vertSubdivisions = 0;
explicitSubdivisions = false;
width = height = 0;
maxWidth = maxHeight = 0;
expanded = false;
}
ID_INLINE idMapPatch::idMapPatch( int maxPatchWidth, int maxPatchHeight ) {
type = TYPE_PATCH;
horzSubdivisions = vertSubdivisions = 0;
explicitSubdivisions = false;
width = height = 0;
maxWidth = maxPatchWidth;
maxHeight = maxPatchHeight;
verts.SetNum( maxWidth * maxHeight );
expanded = false;
}
class idMapEntity {
friend class idMapFile;
public:
idDict epairs;
public:
idMapEntity( void ) { epairs.SetHashSize( 64 ); }
~idMapEntity( void ) { primitives.DeleteContents( true ); }
static idMapEntity * Parse( idLexer &src, bool worldSpawn = false, float version = CURRENT_MAP_VERSION );
bool Write( idFile *fp, int entityNum ) const;
int GetNumPrimitives( void ) const { return primitives.Num(); }
idMapPrimitive * GetPrimitive( int i ) const { return primitives[i]; }
void AddPrimitive( idMapPrimitive *p ) { primitives.Append( p ); }
unsigned int GetGeometryCRC( void ) const;
void RemovePrimitiveData();
bool NeedsReload(const idMapEntity *oldEntity) const;
protected:
idList<idMapPrimitive*> primitives;
};
struct idMapReloadInfo;
class idMapFile {
public:
idMapFile( void );
~idMapFile( void );
// filename does not require an extension
// normally this will use a .reg file instead of a .map file if it exists,
// which is what the game and dmap want, but the editor will want to always
// load a .map file
bool Parse( const char *filename, bool ignoreRegion = false, bool osPath = false );
bool Write( const char *fileName, const char *ext, bool fromBasePath = true );
// get the number of entities in the map
int GetNumEntities( void ) const { return entities.Num(); }
// get the specified entity
idMapEntity * GetEntity( int i ) const { return entities[i]; }
// get the name without file extension
const char * GetName( void ) const { return name; }
// get the name of file used to load map via idFilesystem
const char * GetFileName( void ) const { return fileName; }
// get the file time
ID_TIME_T GetFileTime( void ) const { return fileTime; }
// get CRC for the map geometry
// texture coordinates and entity key/value pairs are not taken into account
unsigned int GetGeometryCRC( void ) const { return geometryCRC; }
// returns true if the file on disk changed
bool NeedsReload();
// reload a map, returning information about changed entities
// used for hot reload from .map file (note: can return "fail")
idMapReloadInfo TryReload();
// apply specified diff to this map, and return information about changes
// used for incremental hot reload from DarkRadiant
idMapReloadInfo ApplyDiff(const char *text);
int AddEntity( idMapEntity *mapentity );
idMapEntity * FindEntity( const char *name );
void RemoveEntity( idMapEntity *mapEnt );
void RemoveEntities( const char *classname );
void RemoveAllEntities();
void RemovePrimitiveData();
bool HasPrimitiveData() { return hasPrimitiveData; }
int GetTotalPrimitivesNum() const;
protected:
float version;
ID_TIME_T fileTime;
unsigned int geometryCRC;
idList<idMapEntity *> entities;
idStr name, fileName;
bool hasPrimitiveData;
private:
void SetGeometryCRC( void );
};
//stgatilov: used to detect which entities changed during idMapFile::Reload
struct idMapReloadInfo {
struct NameAndIdx {
const char *name;
int idx;
static int Cmp(const NameAndIdx *a, const NameAndIdx *b);
};
bool mapInvalid = false;
bool cannotReload = true;
idList<NameAndIdx> modifiedEntities;
idList<NameAndIdx> respawnEntities; //modified entities which are forced to respawn
idList<NameAndIdx> addedEntities;
idList<NameAndIdx> removedEntities;
//note: this map may miss nonmodified entities
std::unique_ptr<idMapFile> oldMap;
};
#endif /* !__MAPFILE_H__ */