netradiant-custom/plugins/mapq3/plugin.cpp
Garux df02774ff5 tweak StringOutputStream use
auto str = StringOutputStream()(bla) use form was not doing copy elision or move, but copy
2024-01-29 16:54:08 +06:00

664 lines
21 KiB
C++

/*
Copyright (C) 2001-2006, William Joseph.
All Rights Reserved.
This file is part of GtkRadiant.
GtkRadiant 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.
GtkRadiant 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 GtkRadiant; if not, write to the Free Software
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
*/
#include "plugin.h"
#include "iscriplib.h"
#include "ibrush.h"
#include "ipatch.h"
#include "ifiletypes.h"
#include "ieclass.h"
#include "qerplugin.h"
#include "scenelib.h"
#include "string/string.h"
#include "stringio.h"
#include "generic/constant.h"
#include "modulesystem/singletonmodule.h"
#include "parse.h"
#include "write.h"
class MapDoom3Dependencies :
public GlobalRadiantModuleRef,
public GlobalFiletypesModuleRef,
public GlobalScripLibModuleRef,
public GlobalEntityClassManagerModuleRef,
public GlobalSceneGraphModuleRef,
public GlobalBrushModuleRef
{
PatchModuleRef m_patchDef2Doom3Module;
PatchModuleRef m_patchDoom3Module;
public:
MapDoom3Dependencies() :
GlobalEntityClassManagerModuleRef( GlobalRadiant().getRequiredGameDescriptionKeyValue( "entityclass" ) ),
GlobalBrushModuleRef( GlobalRadiant().getRequiredGameDescriptionKeyValue( "brushtypes" ) ),
m_patchDef2Doom3Module( "def2doom3" ),
m_patchDoom3Module( "doom3" ){
}
BrushCreator& getBrushDoom3(){
return GlobalBrushCreator();
}
PatchCreator& getPatchDoom3(){
return *m_patchDoom3Module.getTable();
}
PatchCreator& getPatchDef2Doom3(){
return *m_patchDef2Doom3Module.getTable();
}
};
class MapDoom3API final : public TypeSystemRef, public MapFormat, public PrimitiveParser
{
MapDoom3Dependencies& m_dependencies;
public:
typedef MapFormat Type;
STRING_CONSTANT( Name, "mapdoom3" );
INTEGER_CONSTANT( MapVersion, 2 );
MapDoom3API( MapDoom3Dependencies& dependencies ) : m_dependencies( dependencies ){
GlobalFiletypesModule::getTable().addType( Type::Name, Name, filetype_t( "doom3 maps", "*.map" ) );
GlobalFiletypesModule::getTable().addType( Type::Name, Name, filetype_t( "doom3 region", "*.reg" ) );
}
MapFormat* getTable(){
return this;
}
scene::Node& parsePrimitive( Tokeniser& tokeniser ) const {
const char* primitive = tokeniser.getToken();
if ( primitive != 0 ) {
if ( string_equal( primitive, "patchDef3" ) ) {
return m_dependencies.getPatchDoom3().createPatch();
}
else if ( string_equal( primitive, "patchDef2" ) ) {
return m_dependencies.getPatchDef2Doom3().createPatch();
}
else if ( string_equal( primitive, "brushDef3" ) ) {
return m_dependencies.getBrushDoom3().createBrush();
}
}
Tokeniser_unexpectedError( tokeniser, primitive, "#doom3-primitive" );
return g_nullNode;
}
void readGraph( scene::Node& root, TextInputStream& inputStream, EntityCreator& entityTable ) const {
Tokeniser& tokeniser = GlobalScripLibModule::getTable().m_pfnNewSimpleTokeniser( inputStream );
tokeniser.nextLine();
if ( !Tokeniser_parseToken( tokeniser, "Version" ) ) {
return;
}
int version;
if ( !Tokeniser_getInteger( tokeniser, version ) ) {
return;
}
if ( version != MapVersion ) {
globalErrorStream() << "Doom 3 map version " << MapVersion << " supported, version is " << version << '\n';
return;
}
tokeniser.nextLine();
Map_Read( root, tokeniser, entityTable, *this );
tokeniser.release();
}
void writeGraph( scene::Node& root, GraphTraversalFunc traverse, TextOutputStream& outputStream ) const {
TokenWriter& writer = GlobalScripLibModule::getTable().m_pfnNewSimpleTokenWriter( outputStream );
writer.writeToken( "Version" );
writer.writeInteger( MapVersion );
writer.nextLine();
Map_Write( root, traverse, writer, false );
writer.release();
}
};
typedef SingletonModule<
MapDoom3API,
MapDoom3Dependencies,
DependenciesAPIConstructor<MapDoom3API, MapDoom3Dependencies>
>
MapDoom3Module;
MapDoom3Module g_MapDoom3Module;
class MapQuake4API final : public TypeSystemRef, public MapFormat, public PrimitiveParser
{
MapDoom3Dependencies& m_dependencies;
public:
typedef MapFormat Type;
STRING_CONSTANT( Name, "mapquake4" );
INTEGER_CONSTANT( MapVersion, 3 );
MapQuake4API( MapDoom3Dependencies& dependencies ) : m_dependencies( dependencies ){
GlobalFiletypesModule::getTable().addType( Type::Name, Name, filetype_t( "quake4 maps", "*.map" ) );
GlobalFiletypesModule::getTable().addType( Type::Name, Name, filetype_t( "quake4 region", "*.reg" ) );
}
MapFormat* getTable(){
return this;
}
scene::Node& parsePrimitive( Tokeniser& tokeniser ) const {
const char* primitive = tokeniser.getToken();
if ( primitive != 0 ) {
if ( string_equal( primitive, "patchDef3" ) ) {
return m_dependencies.getPatchDoom3().createPatch();
}
else if ( string_equal( primitive, "patchDef2" ) ) {
return m_dependencies.getPatchDef2Doom3().createPatch();
}
else if ( string_equal( primitive, "brushDef3" ) ) {
return m_dependencies.getBrushDoom3().createBrush();
}
}
Tokeniser_unexpectedError( tokeniser, primitive, "#quake4-primitive" );
return g_nullNode;
}
void readGraph( scene::Node& root, TextInputStream& inputStream, EntityCreator& entityTable ) const {
Tokeniser& tokeniser = GlobalScripLibModule::getTable().m_pfnNewSimpleTokeniser( inputStream );
tokeniser.nextLine();
if ( !Tokeniser_parseToken( tokeniser, "Version" ) ) {
return;
}
int version;
if ( !Tokeniser_getInteger( tokeniser, version ) ) {
return;
}
if ( version != MapVersion ) {
globalErrorStream() << "Quake 4 map version " << MapVersion << " supported, version is " << version << '\n';
return;
}
tokeniser.nextLine();
Map_Read( root, tokeniser, entityTable, *this );
tokeniser.release();
}
void writeGraph( scene::Node& root, GraphTraversalFunc traverse, TextOutputStream& outputStream ) const {
TokenWriter& writer = GlobalScripLibModule::getTable().m_pfnNewSimpleTokenWriter( outputStream );
writer.writeToken( "Version" );
writer.writeInteger( MapVersion );
writer.nextLine();
Map_Write( root, traverse, writer, false );
writer.release();
}
};
typedef SingletonModule<
MapQuake4API,
MapDoom3Dependencies,
DependenciesAPIConstructor<MapQuake4API, MapDoom3Dependencies>
>
MapQuake4Module;
MapQuake4Module g_MapQuake4Module;
class MapDependencies :
public GlobalRadiantModuleRef,
public GlobalBrushModuleRef,
public GlobalPatchModuleRef,
public GlobalFiletypesModuleRef,
public GlobalScripLibModuleRef,
public GlobalEntityClassManagerModuleRef,
public GlobalSceneGraphModuleRef
{
public:
MapDependencies() :
GlobalBrushModuleRef( GlobalRadiant().getRequiredGameDescriptionKeyValue( "brushtypes" ) ),
GlobalPatchModuleRef( GlobalRadiant().getRequiredGameDescriptionKeyValue( "patchtypes" ) ),
GlobalEntityClassManagerModuleRef( GlobalRadiant().getRequiredGameDescriptionKeyValue( "entityclass" ) ){
}
};
class MapQ3API final : public TypeSystemRef, public MapFormat, public PrimitiveParser
{
mutable bool m_formatDetected;
public:
typedef MapFormat Type;
STRING_CONSTANT( Name, "mapq3" );
MapQ3API(){
GlobalFiletypesModule::getTable().addType( Type::Name, Name, filetype_t( "quake3 maps", "*.map", true, true, true ) );
GlobalFiletypesModule::getTable().addType( Type::Name, Name, filetype_t( "quake3 region", "*.reg", true, true, true ) );
GlobalFiletypesModule::getTable().addType( Type::Name, Name, filetype_t( "quake3 compiled maps", "*.bsp", false, true, false ) );
}
MapFormat* getTable(){
return this;
}
scene::Node& parsePrimitive( Tokeniser& tokeniser ) const {
const char* primitive = tokeniser.getToken();
if ( primitive != 0 ) {
if ( string_equal( primitive, "patchDef2" ) ) {
return GlobalPatchModule::getTable().createPatch();
}
if( !m_formatDetected ){
EBrushType detectedFormat;
if ( string_equal( primitive, "brushDef" ) ) {
detectedFormat = eBrushTypeQuake3BP;
globalWarningStream() << "detectedFormat = eBrushTypeQuake3BP\n";
}
else if ( string_equal( primitive, "(" ) && tokeniser.bufferContains( " [ " ) && tokeniser.bufferContains( " ] " ) ) {
detectedFormat = eBrushTypeQuake3Valve220;
globalWarningStream() << "detectedFormat = eBrushTypeQuake3Valve220\n";
}
else if ( string_equal( primitive, "(" ) ) {
detectedFormat = eBrushTypeQuake3;
globalWarningStream() << "detectedFormat = eBrushTypeQuake3\n";
}
else{
globalErrorStream() << "Format is not detected\n";
Tokeniser_unexpectedError( tokeniser, primitive, "#different-brush-format" );
return g_nullNode;
}
m_formatDetected = true;
if( detectedFormat != GlobalBrushCreator().getFormat() ){
GlobalBrushCreator().toggleFormat( detectedFormat );
}
}
switch ( GlobalBrushCreator().getFormat() )
{
case eBrushTypeQuake3:
case eBrushTypeQuake3Valve220:
tokeniser.ungetToken(); // (
// fall through
case eBrushTypeQuake3BP:
return GlobalBrushCreator().createBrush();
default:
break;
}
}
Tokeniser_unexpectedError( tokeniser, primitive, "#quake3-primitive" );
return g_nullNode;
}
void readGraph( scene::Node& root, TextInputStream& inputStream, EntityCreator& entityTable ) const {
Tokeniser& tokeniser = GlobalScripLibModule::getTable().m_pfnNewSimpleTokeniser( inputStream );
m_formatDetected = false;
Map_Read( root, tokeniser, entityTable, *this );
tokeniser.release();
}
void writeGraph( scene::Node& root, GraphTraversalFunc traverse, TextOutputStream& outputStream ) const {
TokenWriter& writer = GlobalScripLibModule::getTable().m_pfnNewSimpleTokenWriter( outputStream );
Map_Write( root, traverse, writer, false );
writer.release();
}
};
typedef SingletonModule<MapQ3API, MapDependencies> MapQ3Module;
MapQ3Module g_MapQ3Module;
class MapQ1API final : public TypeSystemRef, public MapFormat, public PrimitiveParser
{
mutable bool m_formatDetected;
public:
typedef MapFormat Type;
STRING_CONSTANT( Name, "mapq1" );
MapQ1API(){
GlobalFiletypesModule::getTable().addType( Type::Name, Name, filetype_t( "quake maps", "*.map" ) );
GlobalFiletypesModule::getTable().addType( Type::Name, Name, filetype_t( "quake region", "*.reg" ) );
}
MapFormat* getTable(){
return this;
}
scene::Node& parsePrimitive( Tokeniser& tokeniser ) const {
const char* primitive = tokeniser.getToken();
if ( primitive != 0 ) {
if( !m_formatDetected ){
EBrushType detectedFormat;
if ( string_equal( primitive, "brushDef" ) ) {
detectedFormat = eBrushTypeQuake3BP;
globalWarningStream() << "detectedFormat = eBrushTypeQuake3BP\n";
}
else if ( string_equal( primitive, "(" ) && tokeniser.bufferContains( " [ " ) && tokeniser.bufferContains( " ] " ) ) {
detectedFormat = eBrushTypeValve220;
globalWarningStream() << "detectedFormat = eBrushTypeValve220\n";
}
else if ( string_equal( primitive, "(" ) ) {
detectedFormat = eBrushTypeQuake;
globalWarningStream() << "detectedFormat = eBrushTypeQuake\n";
}
else{
globalErrorStream() << "Format is not detected\n";
Tokeniser_unexpectedError( tokeniser, primitive, "#different-brush-format" );
return g_nullNode;
}
m_formatDetected = true;
if( detectedFormat != GlobalBrushCreator().getFormat() ){
GlobalBrushCreator().toggleFormat( detectedFormat );
}
}
switch ( GlobalBrushCreator().getFormat() )
{
case eBrushTypeQuake:
case eBrushTypeValve220:
tokeniser.ungetToken(); // (
// fall through
case eBrushTypeQuake3BP:
return GlobalBrushCreator().createBrush();
default:
break;
}
}
Tokeniser_unexpectedError( tokeniser, primitive, "#quake-primitive" );
return g_nullNode;
}
void readGraph( scene::Node& root, TextInputStream& inputStream, EntityCreator& entityTable ) const {
Tokeniser& tokeniser = GlobalScripLibModule::getTable().m_pfnNewSimpleTokeniser( inputStream );
m_formatDetected = false;
Map_Read( root, tokeniser, entityTable, *this );
tokeniser.release();
}
void writeGraph( scene::Node& root, GraphTraversalFunc traverse, TextOutputStream& outputStream ) const {
TokenWriter& writer = GlobalScripLibModule::getTable().m_pfnNewSimpleTokenWriter( outputStream );
Map_Write( root, traverse, writer, true );
writer.release();
}
};
typedef SingletonModule<MapQ1API, MapDependencies> MapQ1Module;
MapQ1Module g_MapQ1Module;
class MapHalfLifeAPI final : public TypeSystemRef, public MapFormat, public PrimitiveParser
{
public:
typedef MapFormat Type;
STRING_CONSTANT( Name, "maphl" );
MapHalfLifeAPI(){
GlobalFiletypesModule::getTable().addType( Type::Name, Name, filetype_t( "half-life maps", "*.map" ) );
GlobalFiletypesModule::getTable().addType( Type::Name, Name, filetype_t( "half-life region", "*.reg" ) );
}
MapFormat* getTable(){
return this;
}
scene::Node& parsePrimitive( Tokeniser& tokeniser ) const {
const char* primitive = tokeniser.getToken();
if ( primitive != 0 ) {
if ( string_equal( primitive, "(" ) ) {
tokeniser.ungetToken(); // (
return GlobalBrushCreator().createBrush();
}
}
Tokeniser_unexpectedError( tokeniser, primitive, "#halflife-primitive" );
return g_nullNode;
}
void readGraph( scene::Node& root, TextInputStream& inputStream, EntityCreator& entityTable ) const {
Tokeniser& tokeniser = GlobalScripLibModule::getTable().m_pfnNewSimpleTokeniser( inputStream );
Map_Read( root, tokeniser, entityTable, *this );
tokeniser.release();
}
void writeGraph( scene::Node& root, GraphTraversalFunc traverse, TextOutputStream& outputStream ) const {
TokenWriter& writer = GlobalScripLibModule::getTable().m_pfnNewSimpleTokenWriter( outputStream );
Map_Write( root, traverse, writer, true );
writer.release();
}
};
typedef SingletonModule<MapHalfLifeAPI, MapDependencies> MapHalfLifeModule;
MapHalfLifeModule g_MapHalfLifeModule;
class MapQ2API final : public TypeSystemRef, public MapFormat, public PrimitiveParser
{
mutable bool m_formatDetected;
public:
typedef MapFormat Type;
STRING_CONSTANT( Name, "mapq2" );
MapQ2API(){
GlobalFiletypesModule::getTable().addType( Type::Name, Name, filetype_t( "quake2 maps", "*.map" ) );
GlobalFiletypesModule::getTable().addType( Type::Name, Name, filetype_t( "quake2 region", "*.reg" ) );
}
MapFormat* getTable(){
return this;
}
scene::Node& parsePrimitive( Tokeniser& tokeniser ) const {
const char* primitive = tokeniser.getToken();
if ( primitive != 0 ) {
if( !m_formatDetected ){
EBrushType detectedFormat;
if ( string_equal( primitive, "brushDef" ) ) {
detectedFormat = eBrushTypeQuake2BP;
globalWarningStream() << "detectedFormat = eBrushTypeQuake2BP\n";
}
else if ( string_equal( primitive, "(" ) && tokeniser.bufferContains( " [ " ) && tokeniser.bufferContains( " ] " ) ) {
detectedFormat = eBrushTypeQuake2Valve220;
globalWarningStream() << "detectedFormat = eBrushTypeQuake2Valve220\n";
}
else if ( string_equal( primitive, "(" ) ) {
detectedFormat = eBrushTypeQuake2;
globalWarningStream() << "detectedFormat = eBrushTypeQuake2\n";
}
else{
globalErrorStream() << "Format is not detected\n";
Tokeniser_unexpectedError( tokeniser, primitive, "#different-brush-format" );
return g_nullNode;
}
m_formatDetected = true;
if( detectedFormat != GlobalBrushCreator().getFormat() ){
GlobalBrushCreator().toggleFormat( detectedFormat );
}
}
switch ( GlobalBrushCreator().getFormat() )
{
case eBrushTypeQuake2:
case eBrushTypeQuake2Valve220:
tokeniser.ungetToken(); // (
// fall through
case eBrushTypeQuake2BP:
return GlobalBrushCreator().createBrush();
default:
break;
}
}
Tokeniser_unexpectedError( tokeniser, primitive, "#quake2-primitive" );
return g_nullNode;
}
void readGraph( scene::Node& root, TextInputStream& inputStream, EntityCreator& entityTable ) const {
Tokeniser& tokeniser = GlobalScripLibModule::getTable().m_pfnNewSimpleTokeniser( inputStream );
m_formatDetected = false;
Map_Read( root, tokeniser, entityTable, *this );
tokeniser.release();
}
void writeGraph( scene::Node& root, GraphTraversalFunc traverse, TextOutputStream& outputStream ) const {
TokenWriter& writer = GlobalScripLibModule::getTable().m_pfnNewSimpleTokenWriter( outputStream );
Map_Write( root, traverse, writer, true );
writer.release();
}
};
typedef SingletonModule<MapQ2API, MapDependencies> MapQ2Module;
MapQ2Module g_MapQ2Module;
#define PARSE_ERROR "error parsing VMF"
inline void parseToken( Tokeniser& tokeniser, const char* token ){
ASSERT_MESSAGE( Tokeniser_parseToken( tokeniser, token ), "error parsing vmf: token not found: " << makeQuoted( token ) );
}
class VMFBlock
{
public:
using VMFBlockArray = std::vector<const VMFBlock*>;
const char* m_name;
VMFBlockArray m_children;
VMFBlock( const char* name, VMFBlockArray children = VMFBlockArray() ) : m_name( name ), m_children( children ){
}
const char* name() const {
return m_name;
}
using const_iterator = VMFBlockArray::const_iterator;
const_iterator begin() const {
return m_children.begin();
}
const_iterator end() const {
return m_children.end();
}
};
const VMFBlock c_vmfNormals( "normals" );
const VMFBlock c_vmfDistances( "distances" );
const VMFBlock c_vmfOffsets( "offsets" );
const VMFBlock c_vmfOffsetNormals( "offset_normals" );
const VMFBlock c_vmfAlphas( "alphas" );
const VMFBlock c_vmfTriangleTags( "triangle_tags" );
const VMFBlock c_vmfAllowedVerts( "allowed_verts" );
const VMFBlock c_vmfDispInfo( "dispinfo", { &c_vmfNormals, &c_vmfDistances, &c_vmfOffsets, &c_vmfOffsetNormals, &c_vmfAlphas, &c_vmfTriangleTags, &c_vmfAllowedVerts } );
const VMFBlock c_vmfSide( "side", { &c_vmfDispInfo } );
const VMFBlock c_vmfEditor( "editor" );
const VMFBlock c_vmfVersionInfo( "versioninfo" );
const VMFBlock c_vmfViewSettings( "viewsettings" );
const VMFBlock c_vmfCordon( "cordon" );
const VMFBlock c_vmfGroup( "group", { &c_vmfEditor } );
const VMFBlock c_vmfCamera( "camera" );
const VMFBlock c_vmfCameras( "cameras", { &c_vmfCamera } );
VMFBlock c_vmfVisGroup( "visgroup" );
const VMFBlock c_vmfVisGroups( "visgroups", { &c_vmfVisGroup } );
const VMFBlock c_vmfSolid( "solid", { &c_vmfSide, &c_vmfEditor } );
const VMFBlock c_vmfConnections( "connections" );
const VMFBlock c_vmfEntity( "entity", { &c_vmfEditor, &c_vmfSolid, &c_vmfGroup, &c_vmfConnections } );
const VMFBlock c_vmfWorld( "world", { &c_vmfEditor, &c_vmfSolid, &c_vmfGroup } );
const VMFBlock c_vmfRoot( "", { &c_vmfVersionInfo, &c_vmfViewSettings, &c_vmfVisGroups, &c_vmfWorld, &c_vmfEntity, &c_vmfCameras, &c_vmfCordon } );
class VMFInit
{
public:
VMFInit(){
c_vmfVisGroup.m_children = { &c_vmfVisGroup };
}
};
VMFInit g_VMFInit;
int g_vmf_entities;
int g_vmf_brushes;
inline VMFBlock::const_iterator VMFBlock_find( const VMFBlock& block, const char* name ){
for ( VMFBlock::const_iterator i = block.begin(); i != block.end(); ++i )
{
if ( string_equal( name, ( *i )->name() ) ) {
return i;
}
}
return block.end();
}
void VMF_parseBlock( Tokeniser& tokeniser, const VMFBlock& block ){
for (;; )
{
const char* key = tokeniser.getToken();
if ( key == 0 || string_equal( key, "}" ) ) {
tokeniser.ungetToken();
break;
}
CopiedString tmp( key );
tokeniser.nextLine();
const char* value = tokeniser.getToken();
tokeniser.nextLine();
if ( string_equal( value, "{" ) ) {
VMFBlock::const_iterator i = VMFBlock_find( block, tmp.c_str() );
ASSERT_MESSAGE( i != block.end(), "error parsing vmf block " << makeQuoted( block.name() ) << ": unknown block: " << makeQuoted( tmp ) );
if ( string_equal( tmp.c_str(), "solid" ) ) {
++g_vmf_brushes;
}
else if ( string_equal( tmp.c_str(), "entity" ) || string_equal( tmp.c_str(), "world" ) ) {
++g_vmf_entities;
}
VMF_parseBlock( tokeniser, **i );
parseToken( tokeniser, "}" );
tokeniser.nextLine();
}
else
{
// was a pair
}
}
}
void VMF_Read( scene::Node& root, Tokeniser& tokeniser, EntityCreator& entityTable ){
g_vmf_entities = g_vmf_brushes = 0;
VMF_parseBlock( tokeniser, c_vmfRoot );
globalOutputStream() << g_vmf_entities << " entities\n";
globalOutputStream() << g_vmf_brushes << " brushes\n";
}
class MapVMFAPI final : public TypeSystemRef, public MapFormat
{
public:
typedef MapFormat Type;
STRING_CONSTANT( Name, "mapvmf" );
MapVMFAPI(){
GlobalFiletypesModule::getTable().addType( Type::Name, Name, filetype_t( "vmf maps", "*.vmf" ) );
GlobalFiletypesModule::getTable().addType( Type::Name, Name, filetype_t( "vmf region", "*.reg" ) );
}
MapFormat* getTable(){
return this;
}
void readGraph( scene::Node& root, TextInputStream& inputStream, EntityCreator& entityTable ) const {
Tokeniser& tokeniser = GlobalScripLibModule::getTable().m_pfnNewSimpleTokeniser( inputStream );
VMF_Read( root, tokeniser, entityTable );
tokeniser.release();
}
void writeGraph( scene::Node& root, GraphTraversalFunc traverse, TextOutputStream& outputStream ) const {
}
};
typedef SingletonModule<MapVMFAPI, MapDependencies> MapVMFModule;
MapVMFModule g_MapVMFModule;
extern "C" void RADIANT_DLLEXPORT Radiant_RegisterModules( ModuleServer& server ){
initialiseModule( server );
g_MapDoom3Module.selfRegister();
g_MapQuake4Module.selfRegister();
g_MapQ3Module.selfRegister();
g_MapQ1Module.selfRegister();
g_MapQ2Module.selfRegister();
g_MapHalfLifeModule.selfRegister();
g_MapVMFModule.selfRegister();
}