/* 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 > 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 > 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 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 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 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 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 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 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(); }