/* 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 GlobalBrushModule::getTable(); } PatchCreator& getPatchDoom3() { return *m_patchDoom3Module.getTable(); } PatchCreator& getPatchDef2Doom3() { return *m_patchDef2Doom3Module.getTable(); } }; class MapDoom3API : 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; } std::size_t version; if(!Tokeniser_getSize(tokeniser, version)) { return; } if(version != MapVersion()) { globalErrorStream() << "Doom 3 map version " << MapVersion() << " supported, version is " << Unsigned(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 : 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; } std::size_t version; if(!Tokeniser_getSize(tokeniser, version)) { return; } if(version != MapVersion()) { globalErrorStream() << "Quake 4 map version " << MapVersion() << " supported, version is " << Unsigned(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 : public TypeSystemRef, public MapFormat, public PrimitiveParser { mutable bool detectedFormat; public: typedef MapFormat Type; STRING_CONSTANT(Name, "mapq3"); MapQ3API() { GlobalFiletypesModule::getTable().addType(Type::Name(), Name(), filetype_t("quake3 maps", "*.map")); GlobalFiletypesModule::getTable().addType(Type::Name(), Name(), filetype_t("quake3 region", "*.reg")); } 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(GlobalBrushModule::getTable().useAlternativeTextureProjection()) { if(string_equal(primitive, "brushDef")) { detectedFormat = true; return GlobalBrushModule::getTable().createBrush(); } else if(!detectedFormat && string_equal(primitive, "(")) { detectedFormat = true; wrongFormat = true; Tokeniser_unexpectedError(tokeniser, primitive, "#quake3-switch-to-texdef"); return g_nullNode; } } else { if(string_equal(primitive, "(")) { detectedFormat = true; tokeniser.ungetToken(); // ( return GlobalBrushModule::getTable().createBrush(); } else if(!detectedFormat && string_equal(primitive, "(")) { detectedFormat = true; wrongFormat = true; Tokeniser_unexpectedError(tokeniser, primitive, "#quake3-switch-to-brush-primitives"); return g_nullNode; } } } Tokeniser_unexpectedError(tokeniser, primitive, "#quake3-primitive"); return g_nullNode; } void readGraph(scene::Node& root, TextInputStream& inputStream, EntityCreator& entityTable) const { detectedFormat = false; wrongFormat = false; 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, false); writer.release(); } }; typedef SingletonModule MapQ3Module; MapQ3Module g_MapQ3Module; class MapQ1API : public TypeSystemRef, public MapFormat, public PrimitiveParser { 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(string_equal(primitive, "(")) { tokeniser.ungetToken(); // ( return GlobalBrushModule::getTable().createBrush(); } } 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); 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 : 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 GlobalBrushModule::getTable().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 : public TypeSystemRef, public MapFormat, public PrimitiveParser { 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(string_equal(primitive, "(")) { tokeniser.ungetToken(); // ( return GlobalBrushModule::getTable().createBrush(); } } 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); 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)); } #include "generic/arrayrange.h" class VMFBlock; typedef ArrayConstRange VMFBlockArrayRange; class VMFBlock { public: const char* m_name; VMFBlockArrayRange m_children; typedef const VMFBlock Value; VMFBlock(const char* name, VMFBlockArrayRange children = VMFBlockArrayRange(0, 0)) : m_name(name), m_children(children) { } const char* name() const { return m_name; } typedef Value* const_iterator; const_iterator begin() const { return m_children.first; } const_iterator end() const { return m_children.last; } }; 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_vmfDispInfoChildren[] = { c_vmfNormals, c_vmfDistances, c_vmfOffsets, c_vmfOffsetNormals, c_vmfAlphas, c_vmfTriangleTags, c_vmfAllowedVerts }; const VMFBlock c_vmfDispInfo("dispinfo", ARRAY_RANGE(c_vmfDispInfoChildren)); const VMFBlock c_vmfSideChildren[] = { c_vmfDispInfo }; const VMFBlock c_vmfSide("side", ARRAY_RANGE(c_vmfSideChildren)); const VMFBlock c_vmfEditor("editor"); const VMFBlock c_vmfVersionInfo("versioninfo"); const VMFBlock c_vmfViewSettings("viewsettings"); const VMFBlock c_vmfCordon("cordon"); const VMFBlock c_vmfGroupChildren[] = { c_vmfEditor }; const VMFBlock c_vmfGroup("group", ARRAY_RANGE(c_vmfGroupChildren)); const VMFBlock c_vmfCamera("camera"); const VMFBlock c_vmfCamerasChildren[] = { c_vmfCamera }; const VMFBlock c_vmfCameras("cameras", ARRAY_RANGE(c_vmfCamerasChildren)); VMFBlock c_vmfVisGroup("visgroup"); VMFBlock c_vmfVisGroups("visgroups", VMFBlockArrayRange(&c_vmfVisGroup, &c_vmfVisGroup+1)); const VMFBlock c_vmfSolidChildren[] = { c_vmfSide, c_vmfEditor }; const VMFBlock c_vmfSolid("solid", ARRAY_RANGE(c_vmfSolidChildren)); const VMFBlock c_vmfConnections("connections"); const VMFBlock c_vmfEntityChildren[] = { c_vmfEditor, c_vmfSolid, c_vmfGroup, c_vmfConnections }; const VMFBlock c_vmfEntity("entity", ARRAY_RANGE(c_vmfEntityChildren)); const VMFBlock c_vmfWorldChildren[] = { c_vmfEditor, c_vmfSolid, c_vmfGroup }; const VMFBlock c_vmfWorld("world", ARRAY_RANGE(c_vmfWorldChildren)); const VMFBlock c_vmfRootChildren[] = { c_vmfVersionInfo, c_vmfViewSettings, c_vmfVisGroups, c_vmfWorld, c_vmfEntity, c_vmfCameras, c_vmfCordon }; const VMFBlock c_vmfRoot("", ARRAY_RANGE(c_vmfRootChildren)); class VMFInit { public: VMFInit() { c_vmfVisGroup.m_children = VMFBlockArrayRange(&c_vmfVisGroup, &c_vmfVisGroup+1); } }; 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.c_str())); 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 : 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(); }