2015-02-02 15:41:03 +13:00
|
|
|
//
|
|
|
|
// Author: Michael Cameron
|
|
|
|
// Email: chronokun@hotmail.com
|
|
|
|
//
|
2017-04-12 00:18:29 -07:00
|
|
|
#include <vector>
|
|
|
|
#include <string>
|
|
|
|
#include <sstream>
|
|
|
|
#include <fstream>
|
|
|
|
#include <iostream>
|
2017-07-13 03:24:02 -07:00
|
|
|
#include <exception>
|
2017-05-05 19:33:26 -07:00
|
|
|
|
2017-07-02 20:22:09 -07:00
|
|
|
#include "cxxopts.hpp"
|
2017-06-18 07:44:42 -07:00
|
|
|
|
2017-05-05 19:33:26 -07:00
|
|
|
#include "oopless-parser.hpp"
|
2017-04-11 08:27:10 -07:00
|
|
|
#include "brushdef.hpp"
|
2015-02-02 15:41:03 +13:00
|
|
|
|
2017-04-10 14:29:47 -07:00
|
|
|
using namespace std;
|
|
|
|
|
2017-06-30 16:33:32 -07:00
|
|
|
//ease of reading function pointer
|
|
|
|
typedef void (*brushdef) (std::stringstream &,
|
|
|
|
const std::vector<TPlanePoints> &);
|
|
|
|
|
|
|
|
#define ARG_INPUT_SHORTALIAS "i"
|
|
|
|
#define ARG_OUTPUT_SHORTALIAS "o"
|
|
|
|
#define ARG_BRUSHFORMAT "gtk"
|
2017-06-18 07:44:42 -07:00
|
|
|
cxxopts::Options arguments(int ac, char ** av) {
|
|
|
|
vector<string> parg = {"input", "output"};
|
|
|
|
cxxopts::Options o("reflex2q3",
|
|
|
|
"A tool to convert Reflex maps to idTech 3 maps.");
|
|
|
|
o.add_options()
|
|
|
|
("h,help", "displays help")
|
|
|
|
("i,input",
|
|
|
|
"required, input map file, from Reflex",
|
|
|
|
cxxopts::value<string>(), "FILE")
|
|
|
|
("o,output",
|
|
|
|
"required, output map file, for idTech 3 or similar map editors",
|
|
|
|
cxxopts::value<string>(), "FILE")
|
2017-06-30 16:33:32 -07:00
|
|
|
(ARG_BRUSHFORMAT,
|
2017-06-18 07:44:42 -07:00
|
|
|
"optional, output in GTK Radiant's brush definition format")
|
|
|
|
("e,ent",
|
|
|
|
"optional, convert game entities given a table of equivalents",
|
|
|
|
cxxopts::value<string>(), "FILE")
|
|
|
|
;
|
|
|
|
o.parse_positional(parg);
|
|
|
|
o.parse(ac, av);
|
|
|
|
if (o.count("help")) {
|
|
|
|
cout << o.help() << endl;
|
|
|
|
}
|
|
|
|
if (o.count(ARG_INPUT_SHORTALIAS) < 1 ||
|
|
|
|
o.count(ARG_OUTPUT_SHORTALIAS) < 1) {
|
|
|
|
cerr << "error: no input or output file given" << endl;
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
return o;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool convert_worldspawn(const cxxopts::Options &o,
|
2017-07-06 13:52:31 -07:00
|
|
|
vector<vector<string> > &q) {
|
2017-06-18 07:44:42 -07:00
|
|
|
bool is_ok = false;
|
|
|
|
brushdef fn = &brushdef_net;
|
2017-06-30 16:33:32 -07:00
|
|
|
if (o.count(ARG_BRUSHFORMAT)) {
|
2017-06-18 07:44:42 -07:00
|
|
|
fn = &brushdef_gtk;
|
|
|
|
}
|
2017-06-17 03:34:19 -07:00
|
|
|
try {
|
2017-06-18 07:44:42 -07:00
|
|
|
is_ok = convertmap(
|
|
|
|
o[ARG_INPUT_SHORTALIAS].as<string>().c_str(), // in file
|
|
|
|
o[ARG_OUTPUT_SHORTALIAS].as<string>().c_str(), // out file
|
|
|
|
fn, // brush definition
|
|
|
|
q); // queue of entities
|
2017-06-17 03:34:19 -07:00
|
|
|
} catch (exception &e) {
|
|
|
|
cout << e.what() << endl;
|
2017-06-18 07:44:42 -07:00
|
|
|
cout << "exception encountered while converting map geometry"
|
|
|
|
" - unrecoverable failure" << endl;
|
2017-06-17 03:34:19 -07:00
|
|
|
}
|
2017-06-18 07:44:42 -07:00
|
|
|
if (is_ok) {
|
2017-06-16 13:08:46 -07:00
|
|
|
cout << "Successfully converted map "
|
2017-06-18 07:44:42 -07:00
|
|
|
<< o[ARG_INPUT_SHORTALIAS].as<string>()
|
2017-06-16 13:08:46 -07:00
|
|
|
<< " to "
|
2017-06-18 07:44:42 -07:00
|
|
|
<< o[ARG_OUTPUT_SHORTALIAS].as<string>()
|
2017-06-16 13:08:46 -07:00
|
|
|
<< endl;
|
|
|
|
} else {
|
|
|
|
cout << "Failed to convert map." << endl;
|
|
|
|
}
|
2017-06-18 07:44:42 -07:00
|
|
|
return is_ok;
|
|
|
|
}
|
|
|
|
|
2017-07-06 13:52:31 -07:00
|
|
|
void print_entities(vector<vector<string> > &q) {
|
2017-06-30 16:33:32 -07:00
|
|
|
vector<string> entity;
|
2017-06-18 07:44:42 -07:00
|
|
|
do {
|
|
|
|
entity = q.front();
|
2017-06-30 16:33:32 -07:00
|
|
|
cout << "--------------------------" << endl;
|
2017-06-18 07:44:42 -07:00
|
|
|
for (const string &s : entity) {
|
|
|
|
cout << s << endl;
|
|
|
|
}
|
2017-06-30 16:33:32 -07:00
|
|
|
cout << "--------------------------" << endl;
|
|
|
|
} while (!q.empty());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-07-06 17:53:43 -07:00
|
|
|
void convert_entities(const cxxopts::Options &o, const vector<vector<string> > &q) {
|
2017-06-30 16:33:32 -07:00
|
|
|
string entfile = o["e"].as<string>();
|
|
|
|
if (entfile.empty()) {
|
|
|
|
cout << "No entity data file given"
|
|
|
|
"- skipping entity conversion" << endl;
|
|
|
|
return;
|
|
|
|
}
|
2017-07-06 17:53:43 -07:00
|
|
|
EntityConverter e(entfile);
|
|
|
|
e.extractMapInfo(q);
|
2017-06-30 16:33:32 -07:00
|
|
|
ofstream fout(o[ARG_OUTPUT_SHORTALIAS].as<string>(), ios::app);
|
|
|
|
brushdef fn = &brushdef_net;
|
|
|
|
vector<string> entity;
|
|
|
|
if (o.count(ARG_BRUSHFORMAT)) {
|
|
|
|
fn = &brushdef_gtk;
|
|
|
|
}
|
2017-07-06 13:52:31 -07:00
|
|
|
for (const vector<string> &entity : q) {
|
2017-07-13 03:24:02 -07:00
|
|
|
try {
|
|
|
|
write(entity, fout, fn, e);
|
|
|
|
} catch (exception &e) {
|
|
|
|
cerr << e.what() << endl;
|
|
|
|
}
|
2017-07-06 13:52:31 -07:00
|
|
|
}
|
2017-06-18 07:44:42 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
int main(int argc, char** argv)
|
|
|
|
{
|
|
|
|
cxxopts::Options p = arguments(argc, argv);
|
|
|
|
|
2017-07-06 13:52:31 -07:00
|
|
|
vector<vector<string> > entities;
|
2017-06-18 07:44:42 -07:00
|
|
|
bool is_ok = convert_worldspawn(p, entities);
|
2017-06-30 16:33:32 -07:00
|
|
|
// print_entities(entities);
|
2017-06-18 07:44:42 -07:00
|
|
|
convert_entities(p, entities);
|
2015-02-02 15:41:03 +13:00
|
|
|
return(0);
|
2017-06-16 13:08:46 -07:00
|
|
|
}
|