plugin.cpp 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662
  1. /*
  2. Copyright (C) 2001-2006, William Joseph.
  3. All Rights Reserved.
  4. This file is part of GtkRadiant.
  5. GtkRadiant is free software; you can redistribute it and/or modify
  6. it under the terms of the GNU General Public License as published by
  7. the Free Software Foundation; either version 2 of the License, or
  8. (at your option) any later version.
  9. GtkRadiant is distributed in the hope that it will be useful,
  10. but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. GNU General Public License for more details.
  13. You should have received a copy of the GNU General Public License
  14. along with GtkRadiant; if not, write to the Free Software
  15. Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
  16. */
  17. #include "plugin.h"
  18. #include "iscriplib.h"
  19. #include "ibrush.h"
  20. #include "ipatch.h"
  21. #include "ifiletypes.h"
  22. #include "ieclass.h"
  23. #include "qerplugin.h"
  24. #include "scenelib.h"
  25. #include "string/string.h"
  26. #include "stringio.h"
  27. #include "generic/constant.h"
  28. #include "modulesystem/singletonmodule.h"
  29. #include "parse.h"
  30. #include "write.h"
  31. class MapDoom3Dependencies :
  32. public GlobalRadiantModuleRef,
  33. public GlobalFiletypesModuleRef,
  34. public GlobalScripLibModuleRef,
  35. public GlobalEntityClassManagerModuleRef,
  36. public GlobalSceneGraphModuleRef,
  37. public GlobalBrushModuleRef
  38. {
  39. PatchModuleRef m_patchDef2Doom3Module;
  40. PatchModuleRef m_patchDoom3Module;
  41. public:
  42. MapDoom3Dependencies() :
  43. GlobalEntityClassManagerModuleRef(GlobalRadiant().getRequiredGameDescriptionKeyValue("entityclass")),
  44. GlobalBrushModuleRef(GlobalRadiant().getRequiredGameDescriptionKeyValue("brushtypes")),
  45. m_patchDef2Doom3Module("def2doom3"),
  46. m_patchDoom3Module("doom3")
  47. {
  48. }
  49. BrushCreator& getBrushDoom3()
  50. {
  51. return GlobalBrushModule::getTable();
  52. }
  53. PatchCreator& getPatchDoom3()
  54. {
  55. return *m_patchDoom3Module.getTable();
  56. }
  57. PatchCreator& getPatchDef2Doom3()
  58. {
  59. return *m_patchDef2Doom3Module.getTable();
  60. }
  61. };
  62. class MapDoom3API : public TypeSystemRef, public MapFormat, public PrimitiveParser
  63. {
  64. MapDoom3Dependencies& m_dependencies;
  65. public:
  66. typedef MapFormat Type;
  67. STRING_CONSTANT(Name, "mapdoom3");
  68. INTEGER_CONSTANT(MapVersion, 2);
  69. MapDoom3API(MapDoom3Dependencies& dependencies) : m_dependencies(dependencies)
  70. {
  71. GlobalFiletypesModule::getTable().addType(Type::Name(), Name(), filetype_t("doom3 maps", "*.map"));
  72. GlobalFiletypesModule::getTable().addType(Type::Name(), Name(), filetype_t("doom3 region", "*.reg"));
  73. }
  74. MapFormat* getTable()
  75. {
  76. return this;
  77. }
  78. scene::Node& parsePrimitive(Tokeniser& tokeniser) const
  79. {
  80. const char* primitive = tokeniser.getToken();
  81. if(primitive != 0)
  82. {
  83. if(string_equal(primitive, "patchDef3"))
  84. {
  85. return m_dependencies.getPatchDoom3().createPatch();
  86. }
  87. else if(string_equal(primitive, "patchDef2"))
  88. {
  89. return m_dependencies.getPatchDef2Doom3().createPatch();
  90. }
  91. else if(string_equal(primitive, "brushDef3"))
  92. {
  93. return m_dependencies.getBrushDoom3().createBrush();
  94. }
  95. }
  96. Tokeniser_unexpectedError(tokeniser, primitive, "#doom3-primitive");
  97. return g_nullNode;
  98. }
  99. void readGraph(scene::Node& root, TextInputStream& inputStream, EntityCreator& entityTable) const
  100. {
  101. Tokeniser& tokeniser = GlobalScripLibModule::getTable().m_pfnNewSimpleTokeniser(inputStream);
  102. tokeniser.nextLine();
  103. if(!Tokeniser_parseToken(tokeniser, "Version"))
  104. {
  105. return;
  106. }
  107. std::size_t version;
  108. if(!Tokeniser_getSize(tokeniser, version))
  109. {
  110. return;
  111. }
  112. if(version != MapVersion())
  113. {
  114. globalErrorStream() << "Doom 3 map version " << MapVersion() << " supported, version is " << Unsigned(version) << "\n";
  115. return;
  116. }
  117. tokeniser.nextLine();
  118. Map_Read(root, tokeniser, entityTable, *this);
  119. tokeniser.release();
  120. }
  121. void writeGraph(scene::Node& root, GraphTraversalFunc traverse, TextOutputStream& outputStream) const
  122. {
  123. TokenWriter& writer = GlobalScripLibModule::getTable().m_pfnNewSimpleTokenWriter(outputStream);
  124. writer.writeToken("Version");
  125. writer.writeInteger(MapVersion());
  126. writer.nextLine();
  127. Map_Write(root, traverse, writer, false);
  128. writer.release();
  129. }
  130. };
  131. typedef SingletonModule<
  132. MapDoom3API,
  133. MapDoom3Dependencies,
  134. DependenciesAPIConstructor<MapDoom3API, MapDoom3Dependencies>
  135. >
  136. MapDoom3Module;
  137. MapDoom3Module g_MapDoom3Module;
  138. class MapQuake4API : public TypeSystemRef, public MapFormat, public PrimitiveParser
  139. {
  140. MapDoom3Dependencies& m_dependencies;
  141. public:
  142. typedef MapFormat Type;
  143. STRING_CONSTANT(Name, "mapquake4");
  144. INTEGER_CONSTANT(MapVersion, 3);
  145. MapQuake4API(MapDoom3Dependencies& dependencies) : m_dependencies(dependencies)
  146. {
  147. GlobalFiletypesModule::getTable().addType(Type::Name(), Name(), filetype_t("quake4 maps", "*.map"));
  148. GlobalFiletypesModule::getTable().addType(Type::Name(), Name(), filetype_t("quake4 region", "*.reg"));
  149. }
  150. MapFormat* getTable()
  151. {
  152. return this;
  153. }
  154. scene::Node& parsePrimitive(Tokeniser& tokeniser) const
  155. {
  156. const char* primitive = tokeniser.getToken();
  157. if(primitive != 0)
  158. {
  159. if(string_equal(primitive, "patchDef3"))
  160. {
  161. return m_dependencies.getPatchDoom3().createPatch();
  162. }
  163. else if(string_equal(primitive, "patchDef2"))
  164. {
  165. return m_dependencies.getPatchDef2Doom3().createPatch();
  166. }
  167. else if(string_equal(primitive, "brushDef3"))
  168. {
  169. return m_dependencies.getBrushDoom3().createBrush();
  170. }
  171. }
  172. Tokeniser_unexpectedError(tokeniser, primitive, "#quake4-primitive");
  173. return g_nullNode;
  174. }
  175. void readGraph(scene::Node& root, TextInputStream& inputStream, EntityCreator& entityTable) const
  176. {
  177. Tokeniser& tokeniser = GlobalScripLibModule::getTable().m_pfnNewSimpleTokeniser(inputStream);
  178. tokeniser.nextLine();
  179. if(!Tokeniser_parseToken(tokeniser, "Version"))
  180. {
  181. return;
  182. }
  183. std::size_t version;
  184. if(!Tokeniser_getSize(tokeniser, version))
  185. {
  186. return;
  187. }
  188. if(version != MapVersion())
  189. {
  190. globalErrorStream() << "Quake 4 map version " << MapVersion() << " supported, version is " << Unsigned(version) << "\n";
  191. return;
  192. }
  193. tokeniser.nextLine();
  194. Map_Read(root, tokeniser, entityTable, *this);
  195. tokeniser.release();
  196. }
  197. void writeGraph(scene::Node& root, GraphTraversalFunc traverse, TextOutputStream& outputStream) const
  198. {
  199. TokenWriter& writer = GlobalScripLibModule::getTable().m_pfnNewSimpleTokenWriter(outputStream);
  200. writer.writeToken("Version");
  201. writer.writeInteger(MapVersion());
  202. writer.nextLine();
  203. Map_Write(root, traverse, writer, false);
  204. writer.release();
  205. }
  206. };
  207. typedef SingletonModule<
  208. MapQuake4API,
  209. MapDoom3Dependencies,
  210. DependenciesAPIConstructor<MapQuake4API, MapDoom3Dependencies>
  211. >
  212. MapQuake4Module;
  213. MapQuake4Module g_MapQuake4Module;
  214. class MapDependencies :
  215. public GlobalRadiantModuleRef,
  216. public GlobalBrushModuleRef,
  217. public GlobalPatchModuleRef,
  218. public GlobalFiletypesModuleRef,
  219. public GlobalScripLibModuleRef,
  220. public GlobalEntityClassManagerModuleRef,
  221. public GlobalSceneGraphModuleRef
  222. {
  223. public:
  224. MapDependencies() :
  225. GlobalBrushModuleRef(GlobalRadiant().getRequiredGameDescriptionKeyValue("brushtypes")),
  226. GlobalPatchModuleRef(GlobalRadiant().getRequiredGameDescriptionKeyValue("patchtypes")),
  227. GlobalEntityClassManagerModuleRef(GlobalRadiant().getRequiredGameDescriptionKeyValue("entityclass"))
  228. {
  229. }
  230. };
  231. class MapQ3API : public TypeSystemRef, public MapFormat, public PrimitiveParser
  232. {
  233. public:
  234. typedef MapFormat Type;
  235. STRING_CONSTANT(Name, "mapq3");
  236. MapQ3API()
  237. {
  238. GlobalFiletypesModule::getTable().addType(Type::Name(), Name(), filetype_t("quake3 maps", "*.map"));
  239. GlobalFiletypesModule::getTable().addType(Type::Name(), Name(), filetype_t("quake3 region", "*.reg"));
  240. }
  241. MapFormat* getTable()
  242. {
  243. return this;
  244. }
  245. scene::Node& parsePrimitive(Tokeniser& tokeniser) const
  246. {
  247. const char* primitive = tokeniser.getToken();
  248. if(primitive != 0)
  249. {
  250. if(string_equal(primitive, "patchDef2"))
  251. {
  252. return GlobalPatchModule::getTable().createPatch();
  253. }
  254. if(GlobalBrushModule::getTable().useAlternativeTextureProjection())
  255. {
  256. if(string_equal(primitive, "brushDef"))
  257. {
  258. return GlobalBrushModule::getTable().createBrush();
  259. }
  260. }
  261. else
  262. {
  263. if(string_equal(primitive, "("))
  264. {
  265. tokeniser.ungetToken(); // (
  266. return GlobalBrushModule::getTable().createBrush();
  267. }
  268. }
  269. }
  270. Tokeniser_unexpectedError(tokeniser, primitive, "#quake3-primitive");
  271. return g_nullNode;
  272. }
  273. void readGraph(scene::Node& root, TextInputStream& inputStream, EntityCreator& entityTable) const
  274. {
  275. Tokeniser& tokeniser = GlobalScripLibModule::getTable().m_pfnNewSimpleTokeniser(inputStream);
  276. Map_Read(root, tokeniser, entityTable, *this);
  277. tokeniser.release();
  278. }
  279. void writeGraph(scene::Node& root, GraphTraversalFunc traverse, TextOutputStream& outputStream) const
  280. {
  281. TokenWriter& writer = GlobalScripLibModule::getTable().m_pfnNewSimpleTokenWriter(outputStream);
  282. Map_Write(root, traverse, writer, false);
  283. writer.release();
  284. }
  285. };
  286. typedef SingletonModule<MapQ3API, MapDependencies> MapQ3Module;
  287. MapQ3Module g_MapQ3Module;
  288. class MapQ1API : public TypeSystemRef, public MapFormat, public PrimitiveParser
  289. {
  290. public:
  291. typedef MapFormat Type;
  292. STRING_CONSTANT(Name, "mapq1");
  293. MapQ1API()
  294. {
  295. GlobalFiletypesModule::getTable().addType(Type::Name(), Name(), filetype_t("quake maps", "*.map"));
  296. GlobalFiletypesModule::getTable().addType(Type::Name(), Name(), filetype_t("quake region", "*.reg"));
  297. }
  298. MapFormat* getTable()
  299. {
  300. return this;
  301. }
  302. scene::Node& parsePrimitive(Tokeniser& tokeniser) const
  303. {
  304. const char* primitive = tokeniser.getToken();
  305. if(primitive != 0)
  306. {
  307. if(string_equal(primitive, "("))
  308. {
  309. tokeniser.ungetToken(); // (
  310. return GlobalBrushModule::getTable().createBrush();
  311. }
  312. }
  313. Tokeniser_unexpectedError(tokeniser, primitive, "#quake-primitive");
  314. return g_nullNode;
  315. }
  316. void readGraph(scene::Node& root, TextInputStream& inputStream, EntityCreator& entityTable) const
  317. {
  318. Tokeniser& tokeniser = GlobalScripLibModule::getTable().m_pfnNewSimpleTokeniser(inputStream);
  319. Map_Read(root, tokeniser, entityTable, *this);
  320. tokeniser.release();
  321. }
  322. void writeGraph(scene::Node& root, GraphTraversalFunc traverse, TextOutputStream& outputStream) const
  323. {
  324. TokenWriter& writer = GlobalScripLibModule::getTable().m_pfnNewSimpleTokenWriter(outputStream);
  325. Map_Write(root, traverse, writer, true);
  326. writer.release();
  327. }
  328. };
  329. typedef SingletonModule<MapQ1API, MapDependencies> MapQ1Module;
  330. MapQ1Module g_MapQ1Module;
  331. class MapHalfLifeAPI : public TypeSystemRef, public MapFormat, public PrimitiveParser
  332. {
  333. public:
  334. typedef MapFormat Type;
  335. STRING_CONSTANT(Name, "maphl");
  336. MapHalfLifeAPI()
  337. {
  338. GlobalFiletypesModule::getTable().addType(Type::Name(), Name(), filetype_t("half-life maps", "*.map"));
  339. GlobalFiletypesModule::getTable().addType(Type::Name(), Name(), filetype_t("half-life region", "*.reg"));
  340. }
  341. MapFormat* getTable()
  342. {
  343. return this;
  344. }
  345. scene::Node& parsePrimitive(Tokeniser& tokeniser) const
  346. {
  347. const char* primitive = tokeniser.getToken();
  348. if(primitive != 0)
  349. {
  350. if(string_equal(primitive, "("))
  351. {
  352. tokeniser.ungetToken(); // (
  353. return GlobalBrushModule::getTable().createBrush();
  354. }
  355. }
  356. Tokeniser_unexpectedError(tokeniser, primitive, "#halflife-primitive");
  357. return g_nullNode;
  358. }
  359. void readGraph(scene::Node& root, TextInputStream& inputStream, EntityCreator& entityTable) const
  360. {
  361. Tokeniser& tokeniser = GlobalScripLibModule::getTable().m_pfnNewSimpleTokeniser(inputStream);
  362. Map_Read(root, tokeniser, entityTable, *this);
  363. tokeniser.release();
  364. }
  365. void writeGraph(scene::Node& root, GraphTraversalFunc traverse, TextOutputStream& outputStream) const
  366. {
  367. TokenWriter& writer = GlobalScripLibModule::getTable().m_pfnNewSimpleTokenWriter(outputStream);
  368. Map_Write(root, traverse, writer, true);
  369. writer.release();
  370. }
  371. };
  372. typedef SingletonModule<MapHalfLifeAPI, MapDependencies> MapHalfLifeModule;
  373. MapHalfLifeModule g_MapHalfLifeModule;
  374. class MapQ2API : public TypeSystemRef, public MapFormat, public PrimitiveParser
  375. {
  376. public:
  377. typedef MapFormat Type;
  378. STRING_CONSTANT(Name, "mapq2");
  379. MapQ2API()
  380. {
  381. GlobalFiletypesModule::getTable().addType(Type::Name(), Name(), filetype_t("quake2 maps", "*.map"));
  382. GlobalFiletypesModule::getTable().addType(Type::Name(), Name(), filetype_t("quake2 region", "*.reg"));
  383. }
  384. MapFormat* getTable()
  385. {
  386. return this;
  387. }
  388. scene::Node& parsePrimitive(Tokeniser& tokeniser) const
  389. {
  390. const char* primitive = tokeniser.getToken();
  391. if(primitive != 0)
  392. {
  393. if(string_equal(primitive, "("))
  394. {
  395. tokeniser.ungetToken(); // (
  396. return GlobalBrushModule::getTable().createBrush();
  397. }
  398. }
  399. Tokeniser_unexpectedError(tokeniser, primitive, "#quake2-primitive");
  400. return g_nullNode;
  401. }
  402. void readGraph(scene::Node& root, TextInputStream& inputStream, EntityCreator& entityTable) const
  403. {
  404. Tokeniser& tokeniser = GlobalScripLibModule::getTable().m_pfnNewSimpleTokeniser(inputStream);
  405. Map_Read(root, tokeniser, entityTable, *this);
  406. tokeniser.release();
  407. }
  408. void writeGraph(scene::Node& root, GraphTraversalFunc traverse, TextOutputStream& outputStream) const
  409. {
  410. TokenWriter& writer = GlobalScripLibModule::getTable().m_pfnNewSimpleTokenWriter(outputStream);
  411. Map_Write(root, traverse, writer, true);
  412. writer.release();
  413. }
  414. };
  415. typedef SingletonModule<MapQ2API, MapDependencies> MapQ2Module;
  416. MapQ2Module g_MapQ2Module;
  417. #define PARSE_ERROR "error parsing VMF"
  418. inline void parseToken(Tokeniser& tokeniser, const char* token)
  419. {
  420. ASSERT_MESSAGE(Tokeniser_parseToken(tokeniser, token), "error parsing vmf: token not found: " << makeQuoted(token));
  421. }
  422. #include "generic/arrayrange.h"
  423. class VMFBlock;
  424. typedef ArrayConstRange<VMFBlock> VMFBlockArrayRange;
  425. class VMFBlock
  426. {
  427. public:
  428. const char* m_name;
  429. VMFBlockArrayRange m_children;
  430. typedef const VMFBlock Value;
  431. VMFBlock(const char* name, VMFBlockArrayRange children = VMFBlockArrayRange(0, 0)) : m_name(name), m_children(children)
  432. {
  433. }
  434. const char* name() const
  435. {
  436. return m_name;
  437. }
  438. typedef Value* const_iterator;
  439. const_iterator begin() const
  440. {
  441. return m_children.begin;
  442. }
  443. const_iterator end() const
  444. {
  445. return m_children.end;
  446. }
  447. };
  448. const VMFBlock c_vmfNormals("normals");
  449. const VMFBlock c_vmfDistances("distances");
  450. const VMFBlock c_vmfOffsets("offsets");
  451. const VMFBlock c_vmfOffsetNormals("offset_normals");
  452. const VMFBlock c_vmfAlphas("alphas");
  453. const VMFBlock c_vmfTriangleTags("triangle_tags");
  454. const VMFBlock c_vmfAllowedVerts("allowed_verts");
  455. const VMFBlock c_vmfDispInfoChildren[] = { c_vmfNormals, c_vmfDistances, c_vmfOffsets, c_vmfOffsetNormals, c_vmfAlphas, c_vmfTriangleTags, c_vmfAllowedVerts };
  456. const VMFBlock c_vmfDispInfo("dispinfo", ARRAY_RANGE(c_vmfDispInfoChildren));
  457. const VMFBlock c_vmfSideChildren[] = { c_vmfDispInfo };
  458. const VMFBlock c_vmfSide("side", ARRAY_RANGE(c_vmfSideChildren));
  459. const VMFBlock c_vmfEditor("editor");
  460. const VMFBlock c_vmfVersionInfo("versioninfo");
  461. const VMFBlock c_vmfViewSettings("viewsettings");
  462. const VMFBlock c_vmfCordon("cordon");
  463. const VMFBlock c_vmfGroupChildren[] = { c_vmfEditor };
  464. const VMFBlock c_vmfGroup("group", ARRAY_RANGE(c_vmfGroupChildren));
  465. const VMFBlock c_vmfCamera("camera");
  466. const VMFBlock c_vmfCamerasChildren[] = { c_vmfCamera };
  467. const VMFBlock c_vmfCameras("cameras", ARRAY_RANGE(c_vmfCamerasChildren));
  468. VMFBlock c_vmfVisGroup("visgroup");
  469. VMFBlock c_vmfVisGroups("visgroups", VMFBlockArrayRange(&c_vmfVisGroup, &c_vmfVisGroup+1));
  470. const VMFBlock c_vmfSolidChildren[] = { c_vmfSide, c_vmfEditor };
  471. const VMFBlock c_vmfSolid("solid", ARRAY_RANGE(c_vmfSolidChildren));
  472. const VMFBlock c_vmfConnections("connections");
  473. const VMFBlock c_vmfEntityChildren[] = { c_vmfEditor, c_vmfSolid, c_vmfGroup, c_vmfConnections };
  474. const VMFBlock c_vmfEntity("entity", ARRAY_RANGE(c_vmfEntityChildren));
  475. const VMFBlock c_vmfWorldChildren[] = { c_vmfEditor, c_vmfSolid, c_vmfGroup };
  476. const VMFBlock c_vmfWorld("world", ARRAY_RANGE(c_vmfWorldChildren));
  477. const VMFBlock c_vmfRootChildren[] = { c_vmfVersionInfo, c_vmfViewSettings, c_vmfVisGroups, c_vmfWorld, c_vmfEntity, c_vmfCameras, c_vmfCordon };
  478. const VMFBlock c_vmfRoot("", ARRAY_RANGE(c_vmfRootChildren));
  479. class VMFInit
  480. {
  481. public:
  482. VMFInit()
  483. {
  484. c_vmfVisGroup.m_children = VMFBlockArrayRange(&c_vmfVisGroup, &c_vmfVisGroup+1);
  485. }
  486. };
  487. VMFInit g_VMFInit;
  488. int g_vmf_entities;
  489. int g_vmf_brushes;
  490. inline VMFBlock::const_iterator VMFBlock_find(const VMFBlock& block, const char* name)
  491. {
  492. for(VMFBlock::const_iterator i = block.begin(); i != block.end(); ++i)
  493. {
  494. if(string_equal(name, (*i).name()))
  495. {
  496. return i;
  497. }
  498. }
  499. return block.end();
  500. }
  501. void VMF_parseBlock(Tokeniser& tokeniser, const VMFBlock& block)
  502. {
  503. for(;;)
  504. {
  505. const char* key = tokeniser.getToken();
  506. if(key == 0 || string_equal(key, "}"))
  507. {
  508. tokeniser.ungetToken();
  509. break;
  510. }
  511. CopiedString tmp(key);
  512. tokeniser.nextLine();
  513. const char* value = tokeniser.getToken();
  514. tokeniser.nextLine();
  515. if(string_equal(value, "{"))
  516. {
  517. VMFBlock::const_iterator i = VMFBlock_find(block, tmp.c_str());
  518. ASSERT_MESSAGE(i != block.end(), "error parsing vmf block " << makeQuoted(block.name()) << ": unknown block: " << makeQuoted(tmp.c_str()));
  519. if(string_equal(tmp.c_str(), "solid"))
  520. {
  521. ++g_vmf_brushes;
  522. }
  523. else if(string_equal(tmp.c_str(), "entity") || string_equal(tmp.c_str(), "world"))
  524. {
  525. ++g_vmf_entities;
  526. }
  527. VMF_parseBlock(tokeniser, *i);
  528. parseToken(tokeniser, "}");
  529. tokeniser.nextLine();
  530. }
  531. else
  532. {
  533. // was a pair
  534. }
  535. }
  536. }
  537. void VMF_Read(scene::Node& root, Tokeniser& tokeniser, EntityCreator& entityTable)
  538. {
  539. g_vmf_entities = g_vmf_brushes = 0;
  540. VMF_parseBlock(tokeniser, c_vmfRoot);
  541. globalOutputStream() << g_vmf_entities << " entities\n";
  542. globalOutputStream() << g_vmf_brushes << " brushes\n";
  543. }
  544. class MapVMFAPI : public TypeSystemRef, public MapFormat
  545. {
  546. public:
  547. typedef MapFormat Type;
  548. STRING_CONSTANT(Name, "mapvmf");
  549. MapVMFAPI()
  550. {
  551. GlobalFiletypesModule::getTable().addType(Type::Name(), Name(), filetype_t("vmf maps", "*.vmf"));
  552. GlobalFiletypesModule::getTable().addType(Type::Name(), Name(), filetype_t("vmf region", "*.reg"));
  553. }
  554. MapFormat* getTable()
  555. {
  556. return this;
  557. }
  558. void readGraph(scene::Node& root, TextInputStream& inputStream, EntityCreator& entityTable) const
  559. {
  560. Tokeniser& tokeniser = GlobalScripLibModule::getTable().m_pfnNewSimpleTokeniser(inputStream);
  561. VMF_Read(root, tokeniser, entityTable);
  562. tokeniser.release();
  563. }
  564. void writeGraph(scene::Node& root, GraphTraversalFunc traverse, TextOutputStream& outputStream) const
  565. {
  566. }
  567. };
  568. typedef SingletonModule<MapVMFAPI, MapDependencies> MapVMFModule;
  569. MapVMFModule g_MapVMFModule;
  570. extern "C" void RADIANT_DLLEXPORT Radiant_RegisterModules(ModuleServer& server)
  571. {
  572. GlobalErrorStream::instance().setOutputStream(server.getErrorStream());
  573. GlobalOutputStream::instance().setOutputStream(server.getOutputStream());
  574. GlobalDebugMessageHandler::instance().setHandler(server.getDebugMessageHandler());
  575. GlobalModuleServer::instance().set(server);
  576. g_MapDoom3Module.selfRegister();
  577. g_MapQuake4Module.selfRegister();
  578. g_MapQ3Module.selfRegister();
  579. g_MapQ1Module.selfRegister();
  580. g_MapQ2Module.selfRegister();
  581. g_MapHalfLifeModule.selfRegister();
  582. g_MapVMFModule.selfRegister();
  583. }