main.cpp 58 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884
  1. /*
  2. Space Construct v0.6-dev
  3. Beanzilla@21:4/110 on FsxNet
  4. Beanzilla@637:1/110 on HappyNet
  5. */
  6. #define VERSION_MAJOR 0
  7. #define VERSION_MINOR 6
  8. #ifndef VERSION_TYPE
  9. #define VERSION_TYPE "dev"
  10. #endif
  11. #define PATH_MAX 256
  12. #define PATH_SEP "/"
  13. //#include <MagiDoor.h> // Sorry Apam... as much as I would like to keep using your stuff it's just not fully functional
  14. #include <OpenDoor.h> // Now using odoors for c++
  15. #include <sqlite3.h>
  16. #if defined(_MSC_VER) || defined(WIN32)
  17. #define snprintf _snprintf
  18. #define strcasecmp _stricmp
  19. #include <winsock2.h>
  20. #ifndef _MSC_VER
  21. #define _MSC_VER 1
  22. #endif
  23. #else
  24. #include <arpa/inet.h>
  25. #endif
  26. // Standard C
  27. #include <time.h>
  28. #include <stdio.h>
  29. #include <string.h>
  30. #include <stdlib.h>
  31. #include <stdarg.h>
  32. #include <ctype.h>
  33. #include <sys/types.h>
  34. #include <sys/stat.h>
  35. #include <unistd.h> // stat
  36. #include <string>
  37. // sc.log ODoors has logging so why not use it!
  38. int inuse = 0; // Are any other copies of us running? (We are a single user door!)
  39. int debug = 0; // Are we in debug mode?
  40. // Please change it for your system!
  41. int allowDev = 1; // Allow "Beanzilla" to not need the password.
  42. std::string sysop_pass = "spaceISbig"; // Case sensitive!
  43. // LOGGING with file output
  44. #include "zf_log.h"
  45. FILE *g_log_file;
  46. static void file_output_callback(const zf_log_message *msg, void *arg) {
  47. (void)arg;
  48. *msg->p = '\n';
  49. fwrite(msg->buf, msg->p - msg->buf + 1, 1, g_log_file);
  50. fflush(g_log_file);
  51. }
  52. static void file_output_close(void) { fclose(g_log_file); }
  53. static int file_output_open(const char *const log_path) {
  54. g_log_file = fopen(log_path, "a");
  55. if (!g_log_file) {
  56. ZF_LOGW("Failed to open log file %s", log_path);
  57. return 0;
  58. }
  59. atexit(file_output_close);
  60. zf_log_set_output_v(ZF_LOG_PUT_STD, 0, file_output_callback);
  61. return 1;
  62. }
  63. void log_flush(void) { fflush(g_log_file); }
  64. // END LOGGING
  65. // Randrange
  66. int randrange(int min, int max){
  67. return min + rand() / (RAND_MAX / (max - min + 1) + 1);
  68. }
  69. class user_info {
  70. // Privates... but why do we allow "friends" to access them too?
  71. int uid, experience, gun, fuel, metal, armor, shield, hitpoints, armorpoints, shieldpoints, shieldsup, laston;
  72. bool dirt;
  73. std::string nick;
  74. std::string real;
  75. public:
  76. void set_up(int uuid, std::string name, std::string realname, int exp, int lo, int a, int ap, int s, int sp, int su, int hp, int g, int m, int f) {
  77. // Setup values
  78. uid = uuid;
  79. nick = name;
  80. real = realname;
  81. experience = exp;
  82. laston = lo;
  83. armor = a;
  84. armorpoints = ap;
  85. shield = s;
  86. shieldpoints = sp;
  87. shieldsup = su;
  88. hitpoints = hp;
  89. gun = g;
  90. metal = m;
  91. fuel = f;
  92. dirt = false;
  93. }
  94. void set_new(std::string name, std::string realname) {
  95. // New User
  96. nick = name;
  97. real = realname;
  98. experience = 0;
  99. laston = 0;
  100. armor = 1;
  101. armorpoints = 8;
  102. shield = 0;
  103. shieldpoints = 0;
  104. shieldsup = 0;
  105. hitpoints = 4;
  106. gun = 1;
  107. metal = 0;
  108. fuel = 10;
  109. dirt = true;
  110. }
  111. bool dirty() {
  112. // Need to save?
  113. return dirt;
  114. }
  115. int calcArmor() {
  116. // How much metal to add one to armor
  117. return (3 * (armor + 1));
  118. }
  119. int calcShield() {
  120. // How much metal to add one to shield
  121. return (6 * (shield + 1));
  122. }
  123. int calcGun() {
  124. // How much metal to add one to gun
  125. return (4 * (gun + 1));
  126. }
  127. int calcHP() {
  128. // our total hitpoints
  129. return (gun + armor + shield + 2);
  130. }
  131. int calcSP() {
  132. // our total shieldpoints
  133. return (shield * 3);
  134. }
  135. int calcAP() {
  136. // our total armorpoints
  137. return (armor * 8);
  138. }
  139. void regen() {
  140. // Process Shields rebooting back online
  141. if(shieldsup != 0) {
  142. dirt = true;
  143. shieldsup -= 1;
  144. } // Process shieldpoints restoring
  145. if(shieldsup != 0) {
  146. dirt = true;
  147. shieldpoints += shield;
  148. } else {
  149. dirt = true;
  150. shieldpoints += 1;
  151. } // Establish limit
  152. if(shieldpoints > calcSP()) {
  153. dirt = true;
  154. shieldpoints = calcSP();
  155. }
  156. }
  157. void repair(bool docked) {
  158. // If we are not docked we are in combat so only repair 2 instead of 4
  159. int amount;
  160. if(docked) {
  161. amount = 4;
  162. } else {
  163. amount = 2;
  164. }
  165. if(armorpoints < calcAP()) {
  166. if(metal != 0) {
  167. dirt = true;
  168. metal -= 1;
  169. armorpoints += amount;
  170. if(armorpoints > calcAP()) {
  171. armorpoints = calcAP();
  172. }
  173. }
  174. }
  175. }
  176. void set_dirty(bool state) {
  177. // To manually change dirty state
  178. dirt = state;
  179. }
  180. int takeDamage(int dmg) {
  181. // Damage System,
  182. // First Shields, Then Armor, Finally Hull.
  183. // Returns any damage left over from target... target died and this is the left over damage
  184. if(shieldsup == 0) {
  185. if(shieldpoints >= dmg) {
  186. // Shields Absorbed all damage
  187. dirt = true;
  188. shieldpoints -= dmg;
  189. return 0;
  190. } else {
  191. // Shields took some damage but more damage to take
  192. dirt = true;
  193. dmg -= shieldpoints;
  194. shieldpoints = 0;
  195. }
  196. }
  197. if(dmg > 0) {
  198. if(armorpoints >= dmg) {
  199. // Armor took all remaining damage
  200. dirt = true;
  201. armorpoints -= dmg;
  202. return 0;
  203. } else {
  204. // Armor took some damage but more damage to take
  205. dirt = true;
  206. dmg -= armorpoints;
  207. armorpoints = 0;
  208. }
  209. }
  210. if(dmg > 0) {
  211. if(hitpoints > dmg) {
  212. // Hull took all remaining damage
  213. dirt = true;
  214. hitpoints -= dmg;
  215. return 0;
  216. } else {
  217. // Hull took some damage but more needed to be taken
  218. // Our ship is :boom: space debris.
  219. dirt = true;
  220. int temp = dmg;
  221. dmg -= hitpoints;
  222. hitpoints -= temp;
  223. temp = 0;
  224. return dmg;
  225. }
  226. }
  227. }
  228. // Exchange Systems
  229. void addEXP(int amount) {
  230. dirt = true;
  231. experience += amount;
  232. }
  233. void rmEXP(int amount) {
  234. dirt = true;
  235. experience -= amount;
  236. }
  237. void addMetal(int amount) {
  238. dirt = true;
  239. metal += amount;
  240. }
  241. void rmMetal(int amount) {
  242. dirt = true;
  243. metal -= amount;
  244. }
  245. void addFuel(int amount) {
  246. dirt = true;
  247. fuel += amount;
  248. }
  249. void rmFuel(int amount) {
  250. dirt = true;
  251. fuel -= amount;
  252. }
  253. void rmLaston(int amount) {
  254. dirt = true;
  255. laston -= amount;
  256. if(laston < 0) {
  257. laston = 0;
  258. }
  259. }
  260. // Another Combat() helper
  261. void addHP(int amount) {
  262. dirt = true;
  263. hitpoints += amount;
  264. if(hitpoints > calcHP()) {
  265. hitpoints = calcHP();
  266. }
  267. }
  268. void addGun(int amount) {
  269. dirt = true;
  270. gun += amount;
  271. addHP(1);
  272. }
  273. void addArmor(int amount) {
  274. dirt = true;
  275. armor += amount;
  276. armorpoints += (amount * calcAP());
  277. addHP(1);
  278. }
  279. void addShield(int amount) {
  280. dirt = true;
  281. shield += amount;
  282. if(shieldsup == 0) {
  283. shieldpoints += (amount * calcSP());
  284. } else {
  285. shieldpoints += amount;
  286. }
  287. addHP(1);
  288. }
  289. void rmSU(int amount) {
  290. dirt = true;
  291. shieldsup -= amount;
  292. if(shieldsup > 0) {
  293. shieldsup = 0;
  294. }
  295. }
  296. // Sets
  297. void set_laston(int lo) {laston = lo;}
  298. void set_uid(int uuid) {uid = uuid;}
  299. void set_nick(std::string name) {nick = name;}
  300. void set_real(std::string name) {real = name;}
  301. void set_experience(int exp) {experience = exp;}
  302. void set_metal(int m) {metal = m;}
  303. void set_fuel(int f) {fuel = f;}
  304. void set_gun(int g) {gun = g;}
  305. void set_armor(int a) {armor = a;}
  306. void set_shield(int s) {shield = s;}
  307. void set_shieldsup(int su) {shieldsup = su;}
  308. void set_armorpoints(int ap) {armorpoints = ap;}
  309. void set_shieldpoints(int sp) {shieldpoints = sp;}
  310. void set_hitpoints(int hp) {hitpoints = hp;}
  311. // Gets
  312. int get_laston() {return laston;}
  313. int get_uid() {return uid;}
  314. std::string get_nick() {return nick;}
  315. std::string get_real() {return real;}
  316. int get_experience() {return experience;}
  317. int get_metal() {return metal;}
  318. int get_fuel() {return fuel;}
  319. int get_gun() {return gun;}
  320. int get_armor() {return armor;}
  321. int get_shield() {return shield;}
  322. int get_armorpoints() {return armorpoints;}
  323. int get_shieldpoints() {return shieldpoints;}
  324. int get_shieldsup() {return shieldsup;}
  325. int get_hitpoints() {return hitpoints;}
  326. };
  327. // DateStamp Structure to provide date difference
  328. typedef struct dateTamp {
  329. int year; // YYYY, or number of years difference
  330. int month; // MM, or number of months difference
  331. int day; // DD, or number of days difference.
  332. int age; // Taking down to aproximate day count.
  333. } dT; // YYYYMMDD it in int form
  334. int dateStamp() {
  335. // YYYYMMDD \o/ In a INT so we can store it and compare against it!
  336. struct tm *time_now;
  337. time_t timen;
  338. int result = 0;
  339. timen = time(NULL);
  340. time_now = localtime(&timen);
  341. result += ((time_now->tm_year + 1900) * 10000);
  342. result += (time_now->tm_mon * 100) + 100;
  343. result += time_now->tm_mday;
  344. return result;
  345. }
  346. int compareDate(int dt) {
  347. // Returns integer of difference, from now.
  348. struct tm *time_now;
  349. time_t timen;
  350. int now = 0;
  351. timen = time(NULL);
  352. time_now = localtime(&timen);
  353. now += ((time_now->tm_year + 1900) * 10000);
  354. now += (time_now->tm_mon * 100) + 100;
  355. now += time_now->tm_mday;
  356. return (now - dt); // 10000 = 1 Year, 100 = 1 Month, 1 = 1 Day
  357. }
  358. dT fromDate(int diff) {
  359. // Reverse process so we can compare exact days or get the structure back out too.
  360. dT result;
  361. result.year = diff / 10000;
  362. result.month = (diff / 100) % 100;
  363. result.day = diff % 100;
  364. result.age = 0;
  365. // Process Aprox day count
  366. result.age += (365 * result.year);
  367. result.age += (30 * result.month);
  368. result.age += result.day;
  369. return result;
  370. }
  371. int db_test() {
  372. // Displays all users in the database
  373. sqlite3 *db;
  374. sqlite3_stmt *stmt;
  375. char sqlbuffer[256];
  376. char strbuffer[256];
  377. int sizeofdb = 0;
  378. int rc = sqlite3_open("spaceconstruct.db3", &db);
  379. if(rc) { // Did we do a successful open?
  380. ZF_LOGF("Failed opening database %s (%d)", sqlite3_errmsg(db), rc);
  381. sqlite3_close(db);
  382. od_exit(-1, FALSE);
  383. }
  384. sqlite3_busy_timeout(db, 5000);
  385. // DB open
  386. strcpy(sqlbuffer, "SELECT * FROM user;");
  387. sqlite3_prepare_v2(db, sqlbuffer, strlen(sqlbuffer), &stmt, NULL);
  388. while(sqlite3_step(stmt) == SQLITE_ROW) {
  389. ZF_LOGV("`white`%s=%d %s=%s %s=%s %s=%d %s=%d %s=%d %s=%d %s=%d %s=%d %s=%d %s=%d %s=%d %s=%d %s=%d\r\n",
  390. sqlite3_column_name(stmt, 0), // int, uid
  391. sqlite3_column_int(stmt, 0),
  392. sqlite3_column_name(stmt, 1), // text, nick
  393. sqlite3_column_text(stmt, 1),
  394. sqlite3_column_name(stmt, 2), // text, real
  395. sqlite3_column_text(stmt, 2),
  396. sqlite3_column_name(stmt, 3), // int, experiece
  397. sqlite3_column_int(stmt, 3),
  398. sqlite3_column_name(stmt, 4), // int, metal
  399. sqlite3_column_int(stmt, 4),
  400. sqlite3_column_name(stmt, 5), // int, fuel
  401. sqlite3_column_int(stmt, 5),
  402. sqlite3_column_name(stmt, 6), // int, gun
  403. sqlite3_column_int(stmt, 6),
  404. sqlite3_column_name(stmt, 7), // int, armor
  405. sqlite3_column_int(stmt, 7),
  406. sqlite3_column_name(stmt, 8), // int, shield
  407. sqlite3_column_int(stmt, 8),
  408. sqlite3_column_name(stmt, 9), // int, armorpoints
  409. sqlite3_column_int(stmt, 9),
  410. sqlite3_column_name(stmt, 10), // int, shieldpoints
  411. sqlite3_column_int(stmt, 10),
  412. sqlite3_column_name(stmt, 11), // int, hitpoints
  413. sqlite3_column_int(stmt, 11),
  414. sqlite3_column_name(stmt, 12), // int, shieldsup
  415. sqlite3_column_int(stmt, 12),
  416. sqlite3_column_name(stmt, 13), // int, laston
  417. sqlite3_column_int(stmt, 13)
  418. );
  419. sizeofdb += 1;
  420. } // Clean up database
  421. ZF_LOGV("There are %d users", sizeofdb);
  422. sqlite3_finalize(stmt);
  423. sqlite3_close(db);
  424. return sizeofdb;
  425. }
  426. int check_lock() {
  427. // Checks stats of lock, (1 = Lock is in effect, 0 = No Lock established)
  428. struct stat s;
  429. if (stat("lock.flg", &s) == 0) {
  430. return 1;
  431. } else {
  432. return 0;
  433. }
  434. }
  435. int grab_lock() {
  436. // Attempt to grab lock, (0 = Already gotten, 1 = Success, -1 = Error)
  437. int valid = check_lock();
  438. FILE *fhandle;
  439. if (valid == 0) {
  440. fhandle = fopen("lock.flg", "w");
  441. if(!fhandle) {
  442. ZF_LOGE("Unable to make lock.flg!");
  443. ZF_LOGE("Something went wrong perhaps we don't have permissions?");
  444. return -1;
  445. }
  446. fprintf(fhandle, "I am in use already!\n");
  447. fclose(fhandle);
  448. ZF_LOGV("Lock Established");
  449. return 1;
  450. } else {
  451. ZF_LOGW("Lock already established!");
  452. return 0;
  453. }
  454. }
  455. void rel_lock() {
  456. // Attempt to release lock
  457. int valid = check_lock();
  458. if (valid == 1) {
  459. if (unlink("lock.flg") != 0) {
  460. ZF_LOGE("Unable to release lock.flg!");
  461. ZF_LOGE("Something went wrong! Players might not be able to play now!");
  462. }
  463. } else {
  464. ZF_LOGW("Lock already released!");
  465. }
  466. }
  467. void log_drop() {
  468. // Spits out info from Drop File:
  469. ZF_LOGV("`white`Name=%s Alias=%s TimeLeft=%d SecLevel=%d Location=%s Node=%d Sysop=%s\r\n",
  470. od_control.user_name,
  471. od_control.user_handle,
  472. od_control.user_timelimit,
  473. od_control.user_security,
  474. od_control.user_location,
  475. od_control.od_node,
  476. od_control.sysop_name
  477. );
  478. }
  479. void paws() {
  480. // Aaah, DRY
  481. od_printf("`white`Press any key to continue...");
  482. od_get_key(true);
  483. od_printf("\r\n");
  484. }
  485. int yesNo() {
  486. char ch;
  487. int done = 0;
  488. od_printf("`bright black`(`bright white`Y`bright black`)`white`es `bright black`(`bright red`N`bright black`)`red`o`white`");
  489. ch = od_get_answer("YyNn\r");
  490. od_printf("\r\n");
  491. switch(tolower(ch)) {
  492. case 'y':
  493. done = 1;
  494. break;
  495. case '\r':
  496. case 'n':
  497. break;
  498. }
  499. return done;
  500. }
  501. int locate_player(char name[]) {
  502. // returns user id for given real name and 0 for no record found
  503. sqlite3 *db;
  504. sqlite3_stmt *stmt;
  505. char sqlbuffer[256];
  506. int result = 0;
  507. int rc = sqlite3_open("spaceconstruct.db3", &db);
  508. if(rc) { // Did we do a successful open?
  509. ZF_LOGF("Failed opening database %s (%d)", sqlite3_errmsg(db), rc);
  510. sqlite3_close(db);
  511. od_exit(-1, FALSE);
  512. }
  513. sqlite3_busy_timeout(db, 5000);
  514. //od_printf("realname = '%s'\r\n", name);
  515. // Locating user with given name
  516. strcpy(sqlbuffer, "SELECT * from user where real=? COLLATE NOCASE;");
  517. sqlite3_prepare_v2(db, sqlbuffer, strlen(sqlbuffer) + 1, &stmt, NULL);
  518. sqlite3_bind_text(stmt, 1, name, strlen(name), SQLITE_STATIC);
  519. rc = sqlite3_step(stmt);
  520. if (rc == SQLITE_ROW) {
  521. // Return user id
  522. result = sqlite3_column_int(stmt, 0);
  523. } else {
  524. // User not found
  525. ZF_LOGW("Unable to locate user %s", name);
  526. result = 0;
  527. } // Clean Up, return results
  528. sqlite3_finalize(stmt);
  529. sqlite3_close(db);
  530. return result;
  531. }
  532. void check_database() {
  533. // If the table users does not exist make it.
  534. sqlite3 *db;
  535. sqlite3_stmt *stmt;
  536. char sqlbuffer[256];
  537. int rc = sqlite3_open("spaceconstruct.db3", &db);
  538. if(rc) {
  539. ZF_LOGF("Failed opening database %s (%d)", sqlite3_errmsg(db), rc);
  540. sqlite3_close(db);
  541. od_exit(-1, FALSE);
  542. }
  543. sqlite3_busy_timeout(db, 5000);
  544. strcpy(sqlbuffer, "SELCECT COUNT(*) FROM user;");
  545. sqlite3_prepare_v2(db, sqlbuffer, strlen(sqlbuffer) + 1, &stmt, NULL);
  546. rc = sqlite3_step(stmt);
  547. if(rc == SQLITE_ROW) {
  548. // Good
  549. } else {
  550. // Bad, Create table
  551. char *errmsg;
  552. rc = sqlite3_exec(db, "CREATE TABLE IF NOT EXISTS \"user\" (`uid` INTEGER PRIMARY KEY AUTOINCREMENT,`nick` TEXT,`real` TEXT,`experience` INTEGER,`metal` INTEGER,`fuel` INTEGER,`gun` INTEGER,`armor` INTEGER,`shield` INTEGER,`armorpoints` INTEGER,`shieldpoints` INTEGER,`hitpoints` INTEGER,`shieldsup` INTEGER,`laston` INTEGER);", NULL, NULL, &errmsg);
  553. ZF_LOGW("Users table did not exist created");
  554. }
  555. sqlite3_finalize(stmt);
  556. sqlite3_close(db);
  557. }
  558. user_info* load_player(int uuid) {
  559. // Returns a player Structure from database
  560. sqlite3 *db;
  561. sqlite3_stmt *stmt;
  562. char sqlbuffer[256];
  563. user_info *result;
  564. int rc = sqlite3_open("spaceconstruct.db3", &db);
  565. if(rc) { // Did we do a successful open?
  566. ZF_LOGF("Failed opening database %s (%d)", sqlite3_errmsg(db), rc);
  567. sqlite3_close(db);
  568. od_exit(-1, FALSE);
  569. }
  570. sqlite3_busy_timeout(db, 5000);
  571. strcpy(sqlbuffer, "SELECT * FROM user WHERE uid=?;");
  572. sqlite3_prepare_v2(db, sqlbuffer, strlen(sqlbuffer) + 1, &stmt, NULL);
  573. sqlite3_bind_int(stmt, 1, uuid);
  574. rc = sqlite3_step(stmt);
  575. if (rc == SQLITE_ROW) {
  576. std::string val1(sqlite3_column_text(stmt, 1));
  577. std::string val2(sqlite3_column_text(stmt, 2));
  578. result->set_up(sqlite3_column_int(stmt, 0), val1, val2, sqlite3_column_int(stmt, 3), sqlite3_column_int(stmt, 13), sqlite3_column_int(stmt, 7), sqlite3_column_int(stmt, 9), sqlite3_column_int(stmt, 8), sqlite3_column_int(stmt, 10), sqlite3_column_int(stmt, 12), sqlite3_column_int(stmt, 11), sqlite3_column_int(stmt, 6), sqlite3_column_int(stmt, 4), sqlite3_column_int(stmt, 5));
  579. } else {
  580. sqlite3_finalize(stmt);
  581. sqlite3_close(db);
  582. result->set_uid(0);
  583. return result;
  584. }
  585. sqlite3_finalize(stmt);
  586. sqlite3_close(db);
  587. return result;
  588. }
  589. user_info* highest_player() {
  590. // Used to find if someone may have fought the Unknown One and won the game already!
  591. int done = 0;
  592. user_info *test;
  593. int high = 0;
  594. int high_exp = 0;
  595. int id = 1;
  596. while(!done) {
  597. test = load_player(id);
  598. if(test->get_uid() != 0) {
  599. if(test->get_experience() > high_exp) {
  600. // Update highest player id and experience
  601. high_exp = test->get_experience();
  602. high = id;
  603. }
  604. } else {
  605. done = 1;
  606. }
  607. id += 1;
  608. } // Ok highest experiencing player found!
  609. test = load_player(high);
  610. return test;
  611. }
  612. void update_player(user_info *data) {
  613. sqlite3 *db;
  614. sqlite3_stmt *stmt;
  615. char sqlbuffer[1024];
  616. int rc = sqlite3_open("spaceconstruct.db3", &db);
  617. if(rc) { // Did we do a successful open?
  618. ZF_LOGF("Failed opening database %s (%d)", sqlite3_errmsg(db), rc);
  619. sqlite3_close(db);
  620. od_exit(-1, FALSE);
  621. }
  622. sqlite3_busy_timeout(db, 5000);
  623. // Bad, don't do this... opens to SQL injection!
  624. //snprintf(sqlbuffer, 1024, "UPDATE user SET nick = '%s', experience = %d, metal = %d, fuel = %d, guns = %d, armors = %d, shields = %d, armorpoints = %d, shieldpoints = %d, hitpoints = %d WHERE uid=%d;",
  625. // data.nick, data.experience, data.metal, data.fuel, data.guns, data.armors, data.shields, data.armorpoints, data.shieldpoints, data.hitpoints, data.uid);
  626. strcpy(sqlbuffer, "UPDATE user SET nick=?, experience=?, metal=?, fuel=?, gun=?, armor=?, shield=?, armorpoints=?, shieldpoints=?, hitpoints=?, shieldsup=?, laston=? WHERE uid=?;");
  627. sqlite3_prepare_v2(db, sqlbuffer, strlen(sqlbuffer) + 1, &stmt, NULL);
  628. // Bind All data values
  629. sqlite3_bind_text(stmt, 1, data->get_nick().c_str(), data->get_nick().length(), SQLITE_STATIC);
  630. sqlite3_bind_int(stmt, 2, data->get_experience());
  631. sqlite3_bind_int(stmt, 3, data->get_metal());
  632. sqlite3_bind_int(stmt, 4, data->get_fuel());
  633. sqlite3_bind_int(stmt, 5, data->get_gun());
  634. sqlite3_bind_int(stmt, 6, data->get_armor());
  635. sqlite3_bind_int(stmt, 7, data->get_shield());
  636. sqlite3_bind_int(stmt, 8, data->get_armorpoints());
  637. sqlite3_bind_int(stmt, 9, data->get_shieldpoints());
  638. sqlite3_bind_int(stmt, 10, data->get_hitpoints());
  639. sqlite3_bind_int(stmt, 11, data->get_shieldsup());
  640. sqlite3_bind_int(stmt, 12, data->get_laston());
  641. sqlite3_bind_int(stmt, 13, data->get_uid());
  642. // Execute
  643. rc = sqlite3_step(stmt);
  644. if(rc != SQLITE_DONE) {
  645. ZF_LOGF("Failed updating player %s (%d)", sqlite3_errmsg(db), rc);
  646. sqlite3_finalize(stmt);
  647. sqlite3_close(db);
  648. od_exit(-1, FALSE);
  649. }
  650. sqlite3_finalize(stmt);
  651. sqlite3_close(db);
  652. }
  653. int create_player(user_info *data) {
  654. sqlite3 *db;
  655. sqlite3_stmt *stmt;
  656. char sqlbuffer[1024];
  657. int rc = sqlite3_open("spaceconstruct.db3", &db);
  658. if(rc) { // Did we do a successful open?
  659. ZF_LOGF("Failed opening database %s (%d)", sqlite3_errmsg(db), rc);
  660. sqlite3_close(db);
  661. od_exit(-1, FALSE);
  662. }
  663. sqlite3_busy_timeout(db, 5000);
  664. strcpy(sqlbuffer, "INSERT INTO user (nick, real, experience, metal, fuel, gun, armor, shield, armorpoints, shieldpoints, hitpoints, shieldsup, laston) VALUES(?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?);");
  665. sqlite3_prepare_v2(db, sqlbuffer, strlen(sqlbuffer) + 1, &stmt, NULL);
  666. sqlite3_bind_text(stmt, 1, data->get_nick().c_str(), data->get_nick().length(), SQLITE_STATIC);
  667. sqlite3_bind_text(stmt, 2, data->get_real().c_str(), data->get_real().length(), SQLITE_STATIC);
  668. sqlite3_bind_int(stmt, 3, data->get_experience());
  669. sqlite3_bind_int(stmt, 4, data->get_metal());
  670. sqlite3_bind_int(stmt, 5, data->get_fuel());
  671. sqlite3_bind_int(stmt, 6, data->get_gun());
  672. sqlite3_bind_int(stmt, 7, data->get_armor());
  673. sqlite3_bind_int(stmt, 8, data->get_shield());
  674. sqlite3_bind_int(stmt, 9, data->get_armorpoints());
  675. sqlite3_bind_int(stmt, 10, data->get_shieldpoints());
  676. sqlite3_bind_int(stmt, 11, data->get_hitpoints());
  677. sqlite3_bind_int(stmt, 12, data->get_shieldsup());
  678. sqlite3_bind_int(stmt, 13, data->get_laston());
  679. rc = sqlite3_step(stmt);
  680. if(rc != SQLITE_DONE) {
  681. ZF_LOGF("Failed inserting player %s (%d)", sqlite3_errmsg(db), rc);
  682. sqlite3_finalize(stmt);
  683. sqlite3_close(db);
  684. od_exit(-1, FALSE);
  685. }
  686. sqlite3_finalize(stmt);
  687. sqlite3_close(db);
  688. return 1; // Good
  689. }
  690. void delete_player(int uuid) {
  691. sqlite3 *db;
  692. sqlite3_stmt *stmt;
  693. char sqlbuffer[256];
  694. int rc = sqlite3_open("spaceconstruct.db3", &db);
  695. if(rc) { // Did we do a successful open?
  696. ZF_LOGF("Failed opening database %s (%d)", sqlite3_errmsg(db), rc);
  697. sqlite3_close(db);
  698. od_exit(-1, FALSE);
  699. }
  700. sqlite3_busy_timeout(db, 5000);
  701. strcpy(sqlbuffer, "DELETE FROM user WHERE uid=?;");
  702. sqlite3_prepare_v2(db, sqlbuffer, strlen(sqlbuffer) + 1, &stmt, NULL);
  703. sqlite3_bind_int(stmt, 1, uuid);
  704. rc = sqlite3_step(stmt);
  705. if (rc != SQLITE_DONE){
  706. ZF_LOGF("Unable to locate user with id=%d got %s (%d)", uuid, sqlite3_errmsg(db), rc);
  707. sqlite3_finalize(stmt);
  708. sqlite3_close(db);
  709. od_exit(-1, FALSE);
  710. }
  711. sqlite3_finalize(stmt);
  712. sqlite3_close(db);
  713. }
  714. void delete_players() {
  715. sqlite3 *db;
  716. sqlite3_stmt *stmt;
  717. char sqlbuffer[256];
  718. int rc = sqlite3_open("spaceconstruct.db3", &db);
  719. if(rc) { // Did we do a successful open?
  720. ZF_LOGF("Failed opening database %s (%d)", sqlite3_errmsg(db), rc);
  721. sqlite3_close(db);
  722. od_exit(-1, FALSE);
  723. }
  724. sqlite3_busy_timeout(db, 5000);
  725. strcpy(sqlbuffer, "DELETE FROM user;");
  726. sqlite3_prepare_v2(db, sqlbuffer, strlen(sqlbuffer) + 1, &stmt, NULL);
  727. rc = sqlite3_step(stmt);
  728. if (rc != SQLITE_DONE){
  729. ZF_LOGF("Delete_players 1/2, Unexpected error %s (%d)", sqlite3_errmsg(db), rc);
  730. sqlite3_finalize(stmt);
  731. sqlite3_close(db);
  732. od_exit(-1, FALSE);
  733. }
  734. strcpy(sqlbuffer, "UPDATE sqlite_sequence SET seq = 0 WHERE name='user';");
  735. sqlite3_prepare_v2(db, sqlbuffer, strlen(sqlbuffer) + 1, &stmt, NULL);
  736. rc = sqlite3_step(stmt);
  737. if (rc != SQLITE_DONE){
  738. ZF_LOGF("Delete_players 2/2, Unexpected error %s (%d)", sqlite3_errmsg(db), rc);
  739. sqlite3_finalize(stmt);
  740. sqlite3_close(db);
  741. od_exit(-1, FALSE);
  742. }
  743. sqlite3_finalize(stmt);
  744. sqlite3_close(db);
  745. }
  746. int display_all_players() {
  747. sqlite3 *db;
  748. sqlite3_stmt *stmt;
  749. char sqlbuffer[1024];
  750. int rc = sqlite3_open("spaceconstruct.db3", &db);
  751. if(rc) { // Did we do a successful open?
  752. ZF_LOGF("Failed opening database %s (%d)", sqlite3_errmsg(db), rc);
  753. sqlite3_close(db);
  754. od_exit(-1, FALSE);
  755. }
  756. sqlite3_busy_timeout(db, 5000);
  757. strcpy(sqlbuffer, "SELECT * from user ORDER BY experience DESC, nick DESC;");
  758. sqlite3_prepare_v2(db, sqlbuffer, strlen(sqlbuffer) + 1, &stmt, NULL);
  759. int total = 0;
  760. od_printf("`bright green`___ The Players ___\r\n");
  761. while(sqlite3_step(stmt) == SQLITE_ROW) {
  762. od_printf("`bright white` %s (%d)\r\n", sqlite3_column_text(stmt, 1), sqlite3_column_int(stmt, 3));
  763. total += 1;
  764. }
  765. if(total > 1) {
  766. od_printf("`green`There are a total of `bright green`%d`green` players\r\n", total);
  767. } else if(total == 0) {
  768. od_printf("`green`There are a total of `bright green`%d`green` players\r\n", total);
  769. } else {
  770. od_printf("`green`There are a total of `bright green`%d`green` player\r\n", total);
  771. }
  772. sqlite3_finalize(stmt);
  773. sqlite3_close(db);
  774. return total;
  775. }
  776. int display_all_opponents(int us) {
  777. sqlite3 *db;
  778. sqlite3_stmt *stmt;
  779. char sqlbuffer[1024];
  780. int rc = sqlite3_open("spaceconstruct.db3", &db);
  781. if(rc) { // Did we do a successful open?
  782. ZF_LOGF("Failed opening database %s (%d)", sqlite3_errmsg(db), rc);
  783. sqlite3_close(db);
  784. od_exit(-1, FALSE);
  785. }
  786. sqlite3_busy_timeout(db, 5000);
  787. strcpy(sqlbuffer, "SELECT * from user ORDER BY experience DESC, nick DESC;");
  788. sqlite3_prepare_v2(db, sqlbuffer, strlen(sqlbuffer) + 1, &stmt, NULL);
  789. int total = 1;
  790. od_printf("`bright green`___ Target List ___\r\n");
  791. while(sqlite3_step(stmt) == SQLITE_ROW) {
  792. if(total != us) {
  793. od_printf("`bright black`(`bright white`%d`bright black`) `bright white`%s (%d)\r\n", total, sqlite3_column_text(stmt, 1), sqlite3_column_int(stmt, 3));
  794. }
  795. total += 1;
  796. }
  797. total -= 1;
  798. sqlite3_finalize(stmt);
  799. sqlite3_close(db);
  800. return total;
  801. }
  802. user_info* build_menu(user_info *my) {
  803. int done = 0;
  804. char ch;
  805. if(my->get_metal() != 0) {
  806. while(!done) {
  807. od_clr_scr();
  808. od_send_file("ansis/sc_build.ans");
  809. // Paint over sc_build.ans file
  810. od_set_cursor(3, 15);
  811. od_printf("%5d", my->calcGun());
  812. od_set_cursor(4, 15);
  813. od_printf("%5d", my->calcArmor());
  814. od_set_cursor(5, 15);
  815. od_printf("%5d", my->calcShield());
  816. od_set_cursor(9, 1);
  817. // Resuming normal operations
  818. od_printf("`bright yellow`You have %8d metal\r\n", my->get_metal());
  819. ch = od_get_answer("GgAaSsCcRr\r");
  820. switch(tolower(ch)) {
  821. case 'g':
  822. if(my->get_metal() >= my->calcGun()) {
  823. my->rmMetal(my->calcGun());
  824. my->addGun(1);
  825. od_printf("`bright green`Added a new gun!\r\n`white`");
  826. paws();
  827. } else {
  828. od_printf("`bright red`You don't have the %d metal for this!\r\n`white`", my->calcGun());
  829. paws();
  830. }
  831. break;
  832. case 'a':
  833. if(my->get_metal() >= my->calcArmor()) {
  834. my->rmMetal(my->calcArmor());
  835. my->addArmor(1);
  836. od_printf("`bright green`Added a new armor!\r\n`white`");
  837. paws();
  838. } else {
  839. od_printf("`bright red`You don't have the %d metal for this!\r\n`white`", my->calcArmor());
  840. paws();
  841. }
  842. break;
  843. case 's':
  844. if(my->get_metal() >= my->calcShield()) {
  845. my->rmMetal(my->calcShield());
  846. my->addShield(1);
  847. od_printf("`bright green`Added a new shield!\r\n`white`");
  848. paws();
  849. } else {
  850. od_printf("`bright red`You don't have the %d metal for this!\r\n`white`", my->calcShield());
  851. paws();
  852. }
  853. break;
  854. case '\r':
  855. case 'c':
  856. done = 1;
  857. break;
  858. case 'r':
  859. if(my->get_experience() >= 25) {
  860. if(my->get_metal() >= 25) {
  861. my->rmEXP(25);
  862. my->rmMetal(25);
  863. my->addFuel(5);
  864. od_printf("`bright green`Made 5 fuel using 25 experience and 25 metal!\r\n");
  865. paws();
  866. } else {
  867. od_printf("`bright red`You need 25 experience and 25 metal to make fuel!\r\n");
  868. paws();
  869. }
  870. } else {
  871. od_printf("`bright red`You need 25 experience and 25 metal to make fuel!\r\n");
  872. paws();
  873. }
  874. break;
  875. }
  876. }
  877. update_player(my);
  878. return my;
  879. } else {
  880. return my;
  881. }
  882. }
  883. void pvp_menu(user_info *my, user_info *targ) {
  884. int done = 0;
  885. int playerWon = 0;
  886. char ch;
  887. // We need to set the targ's hit points and such to max and enable us to "restore" them
  888. int temp_hp = targ->get_hitpoints();
  889. int temp_ap = targ->get_armorpoints();
  890. int temp_sp = targ->get_shieldpoints();
  891. int temp_su = targ->get_shieldsup();
  892. // Set Offline to max values
  893. int targ_dmg = (targ->get_gun() + 1); // Add bonus damage for offline
  894. targ->set_hitpoints(targ->calcHP());
  895. targ->set_armorpoints(targ->calcAP());
  896. targ->set_shieldpoints(targ->calcSP());
  897. targ->set_shieldsup(0);
  898. // Prep Combat values
  899. int fire = 0; // Fire Guns
  900. int flee = 0; // Run away
  901. int regen = 0; // Shields
  902. int test = 0;
  903. int test2 = 0;
  904. while(!done) {
  905. // Process Shields up
  906. if(temp_su != 0) {
  907. temp_su -= 1;
  908. }
  909. if(my->get_shieldsup() != 0) {
  910. my->rmSU(1);
  911. }
  912. // Process Crew repairing hitpoints
  913. if(targ->get_hitpoints() < targ->calcHP()) {
  914. regen = randrange(0, 100);
  915. if(regen >= 85) { // 15%, offline get a bonus 5% on repairing hull.
  916. targ->addHP(1);
  917. od_printf("`bright green`%s`green`'s crew repaired their hull for 1 point!\r\n", targ->get_nick());
  918. }
  919. }
  920. if(my->get_hitpoints() < my->calcHP()) {
  921. regen = randrange(0, 100);
  922. if(regen >= 90) { // 10%
  923. my->addHP(1);
  924. od_printf("`bright green`Your crew repaired your hull for 1 point!\r\n");
  925. }
  926. }
  927. // Process Shields regenerating and display player stats
  928. regen = randrange(0, 100);
  929. if(regen >= 75) { // 25%
  930. targ->regen();
  931. }
  932. od_printf("`bright white`%s has `bright cyan`%d`bright white` shields, %d armor and %d hitpoints left\r\n", targ->get_nick(), temp_sp, temp_ap, temp_hp);
  933. // Online Players Shields
  934. regen = randrange(0, 100);
  935. if(regen >= 80) { // 20%
  936. my->regen();
  937. }
  938. od_printf("`bright white`Your ship has `bright cyan`%d`bright white` shields, %d armor and %d hitpoints left\r\n", my->get_shieldpoints(), my->get_armorpoints(), my->get_hitpoints());
  939. // Form combat menu (Not allowing online to repair since offline can't really repair either)
  940. od_printf("`bright black`(`bright white`A`bright black`)`white`ttack\r\n");
  941. od_printf("`bright black`(`bright white`F`bright black`)`white`lee\r\n");
  942. // Online Players action
  943. ch = od_get_answer("AaFf\r");
  944. switch(tolower(ch)) {
  945. case '\r':
  946. case 'a':
  947. fire = randrange(0, 100);
  948. if(fire >= 50) {
  949. // Proccess online dealing damage to offline
  950. test = targ->takeDamage(my->get_gun());
  951. if(test == 0) {
  952. od_printf("`bright red`%s took %d damage!\r\n", targ->get_nick(), my->get_gun());
  953. } else {
  954. od_printf("`bright red`%s took %d damage, and blew up!\r\n", targ->get_nick(), my->get_gun());
  955. playerWon = 1; // Online beat offline
  956. }
  957. } else { // Online missed
  958. od_printf("`white`You missed!\r\n");
  959. }
  960. break;
  961. case 'f':
  962. flee = randrange(0, 100);
  963. if(flee >= 80) { // 20% chance the online flees
  964. playerWon = 3; // Online ran away from Offline
  965. done = 1;
  966. od_printf("`bright green`You ran away from %s!\r\n", targ->get_nick());
  967. } else {
  968. od_printf("`bright red`%s found you!\r\n", targ->get_nick());
  969. }
  970. break;
  971. } // Now for Offline to attack
  972. if(!playerWon) {
  973. fire = randrange(0, 100);
  974. if(fire >= 45) { // 55%, 5% bonus for offline
  975. test2 = my->takeDamage(targ_dmg);
  976. if(test2 == 0) {
  977. od_printf("`bright red`You took %d damage!\r\n", targ_dmg);
  978. } else {
  979. od_printf("`bright red`You took %d damage, and blew up!\r\n", targ_dmg);
  980. playerWon = 2;
  981. }
  982. }
  983. }
  984. // Winning Move?
  985. if(playerWon) {
  986. switch(playerWon) {
  987. case 1: // Online won over offline
  988. ZF_LOGV("%s beat %s in pvp", my->get_nick(), targ->get_nick());
  989. if(targ->get_experience()) {
  990. my->addEXP((targ->get_experience() / 2));
  991. targ->rmEXP((targ->get_experience() / 2));
  992. }
  993. if(targ->get_metal()) {
  994. my->addMetal((targ->get_metal() / 2));
  995. targ->rmMetal((targ->get_metal() / 2));
  996. }
  997. my->rmFuel(4);
  998. if(my->get_fuel() < 0) {
  999. my->set_fuel(0);
  1000. }
  1001. od_printf("`bright green`Congrats on defeating %s!\r\n`white`", targ->get_nick());
  1002. break;
  1003. case 2: // Offline won over online!
  1004. ZF_LOGV("%s beat %s in pvp", targ->get_nick(), my->get_nick());
  1005. if(my->get_experience()) {
  1006. targ->addEXP((my->get_experience() / 2));
  1007. my->rmEXP((my->get_experience() / 2));
  1008. }
  1009. if(my->get_metal()) {
  1010. targ->addMetal((my->get_metal() / 2));
  1011. my->rmMetal((my->get_metal() / 2));
  1012. }
  1013. my->rmFuel(4);
  1014. if(my->get_fuel() < 0) {
  1015. my->set_fuel(0);
  1016. }
  1017. od_printf("`bright red`Sorry for your loss! (We rebuild your ship)\r\n");
  1018. my->set_hitpoints(4);
  1019. my->set_armorpoints(8);
  1020. my->set_shieldpoints(0);
  1021. my->set_shieldsup(0);
  1022. my->set_armor(1);
  1023. my->set_shield(0);
  1024. my->set_gun(1);
  1025. break;
  1026. case 3:
  1027. ZF_LOGV("%s ran away from %s in pvp", my->get_nick(), targ->get_nick());
  1028. my->rmFuel(4);
  1029. if(my->get_fuel() < 0) {
  1030. my->set_fuel(0);
  1031. }
  1032. break;
  1033. }
  1034. }
  1035. // Restore targ values from temps
  1036. targ->set_shieldsup(temp_su);
  1037. targ->set_armorpoints(temp_ap);
  1038. targ->set_hitpoints(temp_hp);
  1039. targ->set_shieldpoints(temp_sp);
  1040. // DOne, save
  1041. update_player(targ);
  1042. update_player(my);
  1043. }
  1044. }
  1045. void combat_menu(user_info *my, int targ_hp, int targ_dmg, int asteroid) {
  1046. int done = 0;
  1047. int playerWon = 0; // Did the player win? (1 = Yes, 2 = No, 3 = Flee)
  1048. char ch;
  1049. int armorRep = 0; // Can armor be repaired?
  1050. int temp = 0;
  1051. int temp1 = 0;
  1052. int temp2 = 0;
  1053. int temp_hp = targ_hp;
  1054. int flee = 0;
  1055. int fire = 0;
  1056. int regen = 0;
  1057. while(!done) {
  1058. if(asteroid == 0) {
  1059. od_printf("`bright white`Pirate has %d hitpoints left\r\n", targ_hp);
  1060. } else if(asteroid == 1) {
  1061. od_printf("`bright white`Asteroid has %d hitpoints left\r\n", targ_hp);
  1062. } else if(asteroid == 2) {
  1063. od_printf("`bright white`Unknown One has %d hitpoints left\r\n", targ_hp);
  1064. }
  1065. // Proccess Shields comming back online
  1066. if(my->get_shieldsup() != 0) {
  1067. my->rmSU(1);
  1068. }
  1069. // Proccess Crew repairing hull
  1070. if(my->get_hitpoints() < my->calcHP()) {
  1071. regen = randrange(0, 100);
  1072. if(regen >= 90) {
  1073. my->addHP(1);
  1074. }
  1075. }
  1076. // Proccess Shields Regen
  1077. regen = randrange(0, 100);
  1078. if(regen >= 80) { // 20%
  1079. my->regen();
  1080. }
  1081. od_printf("`bright white`Your ship has `bright cyan`%d`bright white` shields, %d armor and %d hitpoints left:\r\n", my->get_shieldpoints(), my->get_armorpoints(), my->get_hitpoints());
  1082. // Forming Combat Menu
  1083. od_printf("`bright black`(`bright white`A`bright black`)`white`ttack\r\n");
  1084. if(my->get_armorpoints() < my->calcAP()) {
  1085. od_printf("`bright black`(`bright white`R`bright black`)`white`epair Armor\r\n");
  1086. armorRep = 1; // Yes we can repair armor
  1087. }
  1088. od_printf("`bright black`(`bright white`F`bright black`)`white`lee\r\n");
  1089. // Proccess User action
  1090. ch = od_get_answer("AaRrFf\r");
  1091. switch(tolower(ch)) {
  1092. case '\r':
  1093. case 'a':
  1094. fire = rand() % 100;
  1095. if(asteroid == 1) { // Its a rock so allow player to hit it more frequent
  1096. fire -= 15;
  1097. }
  1098. if(fire >= 50) {
  1099. targ_hp -= my->get_gun();
  1100. if(targ_hp <= 0) {
  1101. od_printf("`bright red`Target Destroyed!\r\n");
  1102. done = 1;
  1103. playerWon = 1;
  1104. } else {
  1105. od_printf("`bright red`Target took %d damage!\r\n", my->get_gun());
  1106. }
  1107. } else {
  1108. od_printf("`white`You Missed!\r\n");
  1109. }
  1110. break;
  1111. case 'r':
  1112. if(armorRep) {
  1113. my->repair(false);
  1114. } else {
  1115. od_printf("`bright yellow`You don't seem to need repairs!\r\n");
  1116. }
  1117. break;
  1118. case 'f':
  1119. flee = rand() % 100;
  1120. if(asteroid == 1) {
  1121. flee = 100; // Always allow fleeing from asteroid
  1122. } else if(asteroid == 2) {
  1123. flee = 0; // Never allow fleeing from Unknown One
  1124. }
  1125. if(flee >= 82) {
  1126. playerWon = 3;
  1127. done = 1;
  1128. od_printf("`bright green`You ran away!\r\n");
  1129. } else {
  1130. od_printf("`bright red`Target found you!\r\n");
  1131. }
  1132. break;
  1133. } // If Enemy is alive proccess incomming attack
  1134. if(targ_hp > 0) {
  1135. targ_dmg;
  1136. fire = rand() % 100;
  1137. if(asteroid == 1) { // Yup it doesn't have gunz cuz it's rock.
  1138. fire = 0;
  1139. } else if(asteroid == 2) { // This is the Unknown One
  1140. fire += 5;
  1141. }
  1142. if(fire >= 40) {
  1143. temp = my->takeDamage(targ_dmg);
  1144. if(temp == 0) {
  1145. od_printf("`bright red`You took %d damage!\r\n", targ_dmg);
  1146. } else {
  1147. od_printf("`bright red`You took %d damage, and blew up!\r\n", targ_dmg);
  1148. playerWon = 2;
  1149. }
  1150. }
  1151. // firing
  1152. }
  1153. // Winning move?
  1154. if(playerWon) {
  1155. if(playerWon == 1) {
  1156. std::string who;
  1157. switch(asteroid) {
  1158. case 0:
  1159. //Pirate
  1160. who = "Pirate";
  1161. break;
  1162. case 1:
  1163. // Asteroid
  1164. who = "Asteroid";
  1165. break;
  1166. case 2:
  1167. // Unknown One
  1168. who = "Unkown One";
  1169. break;
  1170. }
  1171. ZF_LOGV("%s beat %s", my->get_nick(), who);
  1172. // Player won
  1173. my->addEXP((temp_hp + targ_dmg));
  1174. my->addMetal(temp_hp);
  1175. if(asteroid == 1) { // If asteroid reward more experience and metal!
  1176. my->addEXP(2);
  1177. my->addMetal(2);
  1178. } else if(asteroid == 2) { // If it was the boss!
  1179. my->addEXP(6000); // Ensure the game triggers a reset!
  1180. my->set_metal(0); // Stop the player!
  1181. my->set_fuel(2); // Prevent the player from doing anything else
  1182. }
  1183. my->rmFuel(2);
  1184. if(my->get_fuel() < 0) {
  1185. my->set_fuel(0);
  1186. }
  1187. od_printf("`bright green`Congrats on your victory!\r\n`white`");
  1188. } else if(playerWon == 2) {
  1189. std::string who;
  1190. switch(asteroid) {
  1191. case 0:
  1192. //Pirate
  1193. who = "Pirate";
  1194. break;
  1195. case 1:
  1196. // Asteroid
  1197. who = "Asteroid";
  1198. break;
  1199. case 2:
  1200. // Unknown One
  1201. who = "Unkown One";
  1202. break;
  1203. }
  1204. ZF_LOGV("%s beat %s", who, my->get_nick());
  1205. // Target won
  1206. my->rmEXP((temp_hp + targ_dmg));
  1207. my->rmMetal(temp_hp);
  1208. my->rmFuel(3);
  1209. if(my->get_experience() < 0) {
  1210. my->set_experience(0);
  1211. }
  1212. if(my->get_metal() < 0) {
  1213. my->set_metal(0);
  1214. }
  1215. if(my->get_fuel() < 0) {
  1216. my->set_fuel(0);
  1217. }
  1218. od_printf("`bright red`Sorry for your loss! (We rebuild your ship)\r\n");
  1219. my->set_hitpoints(4);
  1220. my->set_armorpoints(8);
  1221. my->set_shieldpoints(0);
  1222. my->set_shieldsup(0);
  1223. my->set_armor(1);
  1224. my->set_shield(0);
  1225. my->set_gun(1);
  1226. } else if(playerWon == 3) {
  1227. std::string who;
  1228. switch(asteroid) {
  1229. case 0:
  1230. //Pirate
  1231. who = "Pirate";
  1232. break;
  1233. case 1:
  1234. // Asteroid
  1235. who = "Asteroid";
  1236. break;
  1237. case 2:
  1238. // Unknown One
  1239. who = "Unkown One";
  1240. break;
  1241. }
  1242. ZF_LOGV("%s ran away from %s", my->get_nick(), who);
  1243. // Chicken!
  1244. my->rmFuel(1);
  1245. }
  1246. }
  1247. }
  1248. update_player(my);
  1249. }
  1250. void about_game() {
  1251. int done = 0;
  1252. int page = 1;
  1253. while(!done) {
  1254. switch(page) {
  1255. case 1:
  1256. od_clr_scr();
  1257. od_send_file("ansis/sc_doc_1.ans");
  1258. paws();
  1259. break;
  1260. case 2:
  1261. od_clr_scr();
  1262. od_send_file("ansis/sc_doc_2.ans");
  1263. paws();
  1264. break;
  1265. case 3:
  1266. od_clr_scr();
  1267. od_send_file("ansis/sc_doc_3.ans");
  1268. paws();
  1269. break;
  1270. case 4:
  1271. od_clr_scr();
  1272. od_send_file("ansis/sc_doc_4.ans");
  1273. paws();
  1274. break;
  1275. case 5:
  1276. od_clr_scr();
  1277. od_send_file("ansis/sc_doc_5.ans");
  1278. paws();
  1279. break;
  1280. case 6:
  1281. done = 1;
  1282. break;
  1283. }
  1284. od_clr_scr();
  1285. page += 1;
  1286. }
  1287. }
  1288. void play_game() {
  1289. int done = 0;
  1290. char ch;
  1291. char ch1[256];
  1292. user_info *myself;
  1293. user_info *myself2;
  1294. user_info *testing;
  1295. int reset = 0;
  1296. int doRefuel = 0;
  1297. int pirate_encounter = 0;
  1298. int pirate_hp = 0; // Reused for asteroid
  1299. int pirate_dmg = 0; // Reused for asteroid
  1300. int asteroid = 0; // 1 means yes we need to remove damage... completely.
  1301. int me = locate_player(od_control.user_name);
  1302. int abort = 0;
  1303. if(me != 0) {
  1304. //myself = load_player(me);
  1305. testing = load_player(me);
  1306. if(testing->get_uid() != 0) {
  1307. myself = load_player(me);
  1308. }
  1309. dT age = fromDate(compareDate(myself->get_laston()));
  1310. // Process player inactivity
  1311. if(age.age >= 30) { // 30 days
  1312. od_printf("`bright green`Since you haven't played a while we have reset your account!\r\n");
  1313. ZF_LOGW("%s was reset for expiring", od_control.user_name);
  1314. me = 0;
  1315. reset = 1;
  1316. }
  1317. // Processing fuel regeneration
  1318. if(age.age != 0) {
  1319. doRefuel = 1;
  1320. }
  1321. }
  1322. if(me != 0) {
  1323. od_printf("`bright white`Welcome back `bright green`%s\r\n", myself->get_nick());
  1324. ZF_LOGV("%s is back again", od_control.user_name);
  1325. dT age = fromDate(compareDate(myself->get_laston()));
  1326. myself->set_laston(dateStamp());
  1327. if(age.day != 0 || age.month != 0 || age.year != 0) {
  1328. od_printf("`bright yellow`Haven't seen you for");
  1329. }
  1330. if(age.day != 0) {
  1331. if(age.day > 1) {
  1332. od_printf(" %d days", age.day);
  1333. } else {
  1334. od_printf(" %d day", age.day);
  1335. }
  1336. }
  1337. if(age.month != 0) {
  1338. if(age.month > 1) {
  1339. od_printf(" %d months", age.month);
  1340. } else {
  1341. od_printf(" %d month", age.month);
  1342. }
  1343. }
  1344. if(age.year != 0) {
  1345. if(age.year > 1) {
  1346. od_printf(" %d years", age.year);
  1347. } else {
  1348. od_printf(" %d year", age.year);
  1349. }
  1350. }
  1351. if(age.day != 0 || age.month != 0 || age.year != 0) {
  1352. od_printf("\r\n");
  1353. }
  1354. if(doRefuel) {
  1355. if(myself->get_fuel() != 0) {
  1356. myself->addFuel((age.age * 2));
  1357. } else {
  1358. myself->addFuel(((age.age * 2) + 10));
  1359. od_printf("`bright green`For using all your fuel last time you get bonus fuel today!\r\n");
  1360. }
  1361. od_printf("`bright green`You now have %d fuel\r\n", myself->get_fuel());
  1362. }
  1363. update_player(myself);
  1364. paws();
  1365. } else {
  1366. od_printf("`bright white`You look new here. (Hit Enter to abort)\r\n");
  1367. ZF_LOGV("%s is a new player", od_control.user_name);
  1368. while(done == 0) {
  1369. od_printf("`bright yellow`What's your name: ");
  1370. od_input_str(ch1, 26, 32, 126);
  1371. od_printf("\r\n");
  1372. if(strlen(ch1) > 0) {
  1373. //od_input_str(ch1, 256, 32, 126);
  1374. myself->set_nick(ch1);
  1375. } else {
  1376. od_clr_scr();
  1377. abort = 1;
  1378. done = 1;
  1379. od_printf("`bright red`Ok, come back later then...`white`\r\n");
  1380. paws();
  1381. }
  1382. if(abort == 0) {
  1383. od_printf("`bright white`Are you sure you want to be called `bright green`%s\r\n");
  1384. done = yesNo();
  1385. if(done) {
  1386. if(reset) { // So we are reseting the user, this wipes all values and uses update instead of create
  1387. myself->set_up(myself->get_uid(), myself->get_nick(), myself->get_real(), 0, dateStamp(), 1, 8, 0, 0, 0, 4, 1, 0, 10);
  1388. update_player(myself);
  1389. } else { // Brand new user
  1390. myself2->set_new(myself->get_nick(), od_control.user_name);
  1391. create_player(myself2);
  1392. myself = load_player(locate_player(od_control.user_name));
  1393. }
  1394. }
  1395. }
  1396. }
  1397. done = 0; // Reset this for our next loop.
  1398. }
  1399. // Ok user is here now lets ask what they want to do... build/attack
  1400. if(abort == 0) {
  1401. od_clr_scr();
  1402. while(!done) {
  1403. od_printf("`bright yellow`You have %d fuel left for today,\r\n", myself->get_fuel());
  1404. if(myself->get_shieldsup() == 0) {
  1405. od_printf("`bright yellow`Your ship has `bright cyan`%d`bright yellow` shields, %d armor and %d hitpoints left:\r\n", myself->get_shieldpoints(), myself->get_armorpoints(), myself->get_hitpoints());
  1406. } else {
  1407. od_printf("`bright yellow`Your ship has %d armor, `bright cyan`%d`bright yellow` shields and %d hitpoints left:\r\n", myself->get_shieldpoints(), myself->get_armorpoints(), myself->get_hitpoints());
  1408. }
  1409. if(myself->get_armorpoints() < myself->calcAP()) {
  1410. od_printf("`bright black`(`bright white`R`bright black`)`white`epair Armor\r\n");
  1411. }
  1412. if(myself->get_shieldpoints() < myself->calcSP() || myself->get_shieldsup() != 0) {
  1413. od_printf("`bright black`(`bright white`G`bright black`)`white`enerate Shields\r\n");
  1414. }
  1415. if(myself->get_fuel() >= 2) {
  1416. od_send_file("ansis/sc_gameh.ans");
  1417. ch = od_get_answer("SsHhBbQqRrGgLlVv");
  1418. } else {
  1419. od_send_file("ansis/sc_gamel.ans");
  1420. ch = od_get_answer("BbQqRrGgLlVv");
  1421. }
  1422. od_clr_scr();
  1423. switch(tolower(ch)) {
  1424. case 's':
  1425. if(myself->get_fuel() >= 1) {
  1426. pirate_encounter = rand() % 100;
  1427. if(pirate_encounter >= 75) { // 25% chance to encounter a Pirate, or Unknown One
  1428. if(myself->get_experience() > 3000) {
  1429. if(pirate_encounter >= 95) { // 5% chance to encounter unknown one
  1430. od_printf("`bright yellow`You encounter the `bright red`Unknown One\r\n`white`");
  1431. asteroid = 2;
  1432. } else {
  1433. od_printf("`bright yellow`You encounter a `bright red`Pirate\r\n`white`");
  1434. asteroid = 0;
  1435. }
  1436. } else { // The player is not skilled enough to fight the unknown one
  1437. od_printf("`bright yellow`You encounter a `bright red`Pirate\r\n`white`");
  1438. asteroid = 0;
  1439. }
  1440. } else { // 75% chance to encounter a Asteroid
  1441. od_printf("`bright yellow`You encounter a `bright red`Asteroid\r\n`white`");
  1442. asteroid = 1;
  1443. }
  1444. if(asteroid == 0 || asteroid == 2) {
  1445. pirate_hp = ((myself->get_hitpoints() + 1) + (randrange(0, 4)));
  1446. if(asteroid == 2) { // Increase boss hitpoints
  1447. pirate_hp += 10;
  1448. }
  1449. // Increase difficulty based on experience...
  1450. pirate_dmg = (myself->get_experience() / 500);
  1451. if(pirate_dmg == 0) { // Always deal damage
  1452. pirate_dmg = 1;
  1453. }
  1454. if(asteroid == 2) { // Increase boss damage
  1455. pirate_dmg += 2;
  1456. }
  1457. // Send off to combat menu
  1458. combat_menu(myself, pirate_hp, pirate_dmg, asteroid);
  1459. } else {
  1460. pirate_hp = (myself->get_hitpoints() + 1) + (rand() % 10);
  1461. // Send off to combat menu
  1462. combat_menu(myself, pirate_hp, 0, asteroid);
  1463. }
  1464. }
  1465. break;
  1466. case 'h':
  1467. if(myself->get_fuel() >= 4) {
  1468. int targs = display_all_opponents(myself->get_uid());
  1469. int targg = 1;
  1470. int done2 = 0;
  1471. if(targg == myself->get_uid()) {
  1472. targg += 1;
  1473. }
  1474. while(!done2) {
  1475. od_printf("`bright red`Targetting: %d\r\n", targg);
  1476. if(yesNo()) {
  1477. done2 = 1;
  1478. } else if(targg < targs){
  1479. targg += 1;
  1480. } else {
  1481. done2 = 1;
  1482. od_printf("`bright red`ABORTED!\r\n");
  1483. break;
  1484. }
  1485. }
  1486. user_info *targ = load_player(targg);
  1487. if(targ->get_uid() != 0) {
  1488. // I need to add some limits in here else someone could attack weaker players!
  1489. int proceed = 0;
  1490. int temp = targ->get_experience() + 1000; // Prevent the person from selecting a player that is higher than them
  1491. int temp1 = targ->get_experience() - 500; // Prevent the person from selecting a player that is lower than them
  1492. if(temp1 > 0) {
  1493. temp1 = 0;
  1494. }
  1495. if(temp > myself->get_experience() && temp1 < myself->get_experience()) {
  1496. proceed = 1;
  1497. }
  1498. if(proceed) {
  1499. pvp_menu(myself, targ);
  1500. } else {
  1501. od_printf("`bright white`Please select a opponent within your range of expertise!\r\n");
  1502. break;
  1503. }
  1504. }
  1505. } else {
  1506. od_printf("`bright white`You need 4 fuel to fight another player!\r\n`white`");
  1507. }
  1508. break;
  1509. case 'b':
  1510. if(myself->get_metal() != 0) {
  1511. // Send off to build menu
  1512. od_clr_scr();
  1513. //od_send_file("ansis/sc_build.ans", FALSE);
  1514. myself = build_menu(myself);
  1515. } else {
  1516. od_printf("`bright white`I am sorry you have no metal go fight.`white`\r\n");
  1517. }
  1518. break;
  1519. case 'q':
  1520. done = 1;
  1521. break;
  1522. case 'r':
  1523. if(myself->get_armorpoints() < (myself->get_armor() * 8)) {
  1524. myself->repair(true);
  1525. update_player(myself);
  1526. } else {
  1527. od_printf("`bright yellow`You don't seem to need repairs!\r\n`white`");
  1528. }
  1529. break;
  1530. case 'g':
  1531. if(myself->get_shieldpoints() < myself->calcSP() || myself->get_shieldsup() != 0) {
  1532. myself->regen();
  1533. update_player(myself);
  1534. } else {
  1535. od_printf("`bright yellow`You don't seem to need to regenerate shields!\r\n`white`");
  1536. }
  1537. break;
  1538. case 'v':
  1539. od_printf("`bright white` ___ %s's Ship Status ___\r\n", myself->get_nick());
  1540. od_printf("`bright white`Items: # Installed: Metal to Upgrade:\r\n");
  1541. od_printf("`bright red`Guns: %8d %8d\r\n", myself->get_gun(), myself->calcGun());
  1542. od_printf("`bright yellow`Armor: %8d %8d\r\n", myself->get_armor(), myself->calcArmor());
  1543. od_printf("`bright cyan`Shields: %8d %8d\r\n", myself->get_shield(), myself->calcShield());
  1544. od_printf("`bright white`\r\n ___ Situation Report ___\r\n");
  1545. od_printf("`bright white`Items: Maximum: Current:\r\n");
  1546. od_printf("`bright green`Hitpoints: %8d %8d\r\n", myself->calcHP(), myself->get_hitpoints());
  1547. od_printf("`bright yellow`Armor: %8d %8d\r\n", myself->calcAP(), myself->get_armorpoints());
  1548. od_printf("`bright cyan`Shields: %8d %8d\r\n", myself->calcSP(), myself->get_shieldpoints());
  1549. od_printf("`bright magenta`Experience: %8d\r\n", myself->get_experience());
  1550. od_printf("`bright magenta`Fuel: %8d\r\n", myself->get_fuel());
  1551. od_printf("`white`Metal: %8d\r\n", myself->get_metal());
  1552. paws();
  1553. od_clr_scr();
  1554. break;
  1555. }
  1556. }
  1557. }
  1558. }
  1559. void sysop_menu() {
  1560. int done = 0;
  1561. char ch;
  1562. int targ_ply = 1;
  1563. int targ_max = 0;
  1564. int done1 = 0;
  1565. user_info *targ;
  1566. while(!done) {
  1567. od_clr_scr();
  1568. od_send_file("ansis/sc_sysop.ans");
  1569. ch = od_get_answer("RrNnQq");
  1570. //od_printf("\r\n");
  1571. switch(tolower(ch)) {
  1572. case 'q':
  1573. done = 1;
  1574. break;
  1575. case 'r':
  1576. od_printf("`bright red`__ Reset ___\r\n");
  1577. od_printf("`bright red`DANGER THIS WILL DELETE ALL PLAYERS IN GAME\r\n");
  1578. if(yesNo() == 0) {
  1579. od_printf("`bright red`ABORTED!\r\n");
  1580. paws();
  1581. break;
  1582. }
  1583. ZF_LOGW("%s issued reset", od_control.user_name);
  1584. delete_players();
  1585. paws();
  1586. break;
  1587. case 'n':
  1588. od_printf("`bright red`__ New Day ___\r\n");
  1589. while(!done1) {
  1590. targ = load_player(targ_ply);
  1591. if(targ->get_uid() != 0) {
  1592. targ->rmLaston(1); // Travel back 1 day
  1593. update_player(targ); // Save
  1594. } else {
  1595. done1 = 1;
  1596. }
  1597. targ_ply += 1;
  1598. targ_max += 1;
  1599. }
  1600. od_printf("`bright red`%d players affected\r\n", (targ_max - 1));
  1601. targ_ply = 1;
  1602. targ_max = 0;
  1603. done1 = 0;
  1604. ZF_LOGW("%s issued new day", od_control.user_name);
  1605. paws();
  1606. break;
  1607. }
  1608. }
  1609. od_clr_scr();
  1610. }
  1611. void cleanMe() {
  1612. rel_lock();
  1613. od_exit(0, FALSE);
  1614. }
  1615. void main_menu() {
  1616. // Main menu for once things are done being initalized
  1617. int done = 0;
  1618. int done1 = 0; // For Reset Game and for New Day... also for Debug
  1619. int targ_ply = 1; // Target player to perform op!
  1620. int targ_max = 0; // Total number of players affected!
  1621. user_info *targ; // Player Struct
  1622. user_info *test;
  1623. dT targ_age; // Date Struct
  1624. dT test2;
  1625. char targ_real[256];
  1626. char ch;
  1627. int auth; // Security
  1628. char sys[48]; // Security
  1629. // 0 means we got the lock first, 1 means someone else has it already!
  1630. while(!done) {
  1631. od_clr_scr();
  1632. od_send_file("ansis/sc_main.ans");
  1633. ch = od_get_answer("PpLlVvQqSsAa");
  1634. od_clr_scr();
  1635. switch(tolower(ch)){
  1636. case 'q':
  1637. // Quit
  1638. done = 1;
  1639. break;
  1640. case 'p':
  1641. // Play Game (Default if the player just hits enter)
  1642. // Check to see if we have a winner if so refuse to let anyone on! (Wait 3 days then reset the game!)
  1643. test = highest_player();
  1644. test2 = fromDate(compareDate(test->get_laston()));
  1645. if(test->get_experience() >= 8000) { // If the player has the minimum of 3000 exp to fight the Unkown One and wins earning 5000 thats 8000 minimum
  1646. if(test2.age >= 3) { // Auto reset the game or not allow the player to play!
  1647. delete_players();
  1648. ZF_LOGW("Auto-Reset Occured");
  1649. od_printf("`bright green`Game has been reset!\r\n");
  1650. paws();
  1651. } else { // And the lucky winner is!
  1652. int resetage = 3 - test2.age;
  1653. od_printf("`bright green`%s has won the game!\r\n", test->get_nick());
  1654. od_printf("`bright green`The Game will reset in %d days!\r\n", resetage);
  1655. ZF_LOGV("Auto-Reset will occur in %d days", resetage);
  1656. paws();
  1657. break;
  1658. }
  1659. }
  1660. play_game();
  1661. break;
  1662. case 'a':
  1663. about_game();
  1664. break;
  1665. case 'l':
  1666. // List Players in the game
  1667. display_all_players();
  1668. paws();
  1669. break;
  1670. case 'v':
  1671. // Version
  1672. od_clr_scr();
  1673. od_printf("`bright yellow`_____________________________\r\n");
  1674. od_printf("`bright yellow` Space Construct v%d.%d-%s\r\n", VERSION_MAJOR, VERSION_MINOR, VERSION_TYPE);
  1675. od_printf("`bright yellow` By: Beanzilla\r\n");
  1676. od_printf("`bright yellow` FsxNet 21:4/110\r\n");
  1677. od_printf("`bright yellow` HappyNet 637:1/110\r\n");
  1678. od_printf("`bright yellow`_____________________________\r\n");
  1679. paws();
  1680. break;
  1681. case 's':
  1682. auth = 0;
  1683. if(allowDev && strcmp(od_control.user_handle, "Beanzilla") == 0) {
  1684. od_printf("`bright green`Access Granted\r\n");
  1685. ZF_LOGV("Dev accessed sysop menu");
  1686. auth = 1;
  1687. } else {
  1688. od_printf("`bright red`Sysop Password: ");
  1689. strcpy(sys, "");
  1690. od_input_str(sys, 32, 32, 126);
  1691. od_clr_scr(); // Add a bit of protecton
  1692. if(sys == sysop_pass) {
  1693. od_printf("`bright green`Access Granted\r\n");
  1694. ZF_LOGV("%s accessed sysop menu", od_control.user_name);
  1695. auth = 1;
  1696. }
  1697. }
  1698. // Give only Sysop and dev access to restricted commands
  1699. if(auth) {
  1700. paws();
  1701. sysop_menu();
  1702. } else {
  1703. od_printf("`bright red`Access Denied\r\n");
  1704. ZF_LOGW("%s tried to access sysop menu", od_control.user_name);
  1705. paws();
  1706. }
  1707. break;
  1708. }
  1709. }
  1710. }
  1711. int main(int argc, char *argv[]) {
  1712. od_parse_cmd_line(argc, argv);
  1713. // Initiate Door
  1714. od_init();
  1715. od_clr_scr();
  1716. // Debug System
  1717. if (debug) {
  1718. ZF_LOGV("=== Debug ===");
  1719. ZF_LOGV("DropFile...\r\n");
  1720. log_drop();
  1721. int test = check_lock();
  1722. if (test == 0) {
  1723. ZF_LOGV("Lock is: Avalible\r\n");
  1724. ZF_LOGV("Database...\r\n");
  1725. db_test();
  1726. } else {
  1727. ZF_LOGV("Lock is: Taken\r\n");
  1728. ZF_LOGV("CANCELED Database Dump\r\n");
  1729. }
  1730. } else {
  1731. ZF_LOGV("--- Debug ---");
  1732. }
  1733. // Normal Operations
  1734. inuse = check_lock();
  1735. if(!inuse) {
  1736. int lock = grab_lock();
  1737. if(!lock) {
  1738. od_printf("`white`Something went wrong getting the lock.\r\n");
  1739. ZF_LOGF("Lock issue");
  1740. paws();
  1741. od_exit(0, FALSE);
  1742. } else {
  1743. atexit(cleanMe);
  1744. check_database(); // Ensure the users table exists.
  1745. main_menu();
  1746. }
  1747. } else {
  1748. od_send_file("ansis/sc_inuse.ans");
  1749. ZF_LOGI("In use");
  1750. paws();
  1751. od_exit(0, FALSE);
  1752. }
  1753. // Odoors testing
  1754. // Screensaving
  1755. /*char screen[4096]; // 4MB buffer
  1756. // Draw box
  1757. if(od_draw_box(5, 1, 20, 15) == true) {
  1758. od_set_cursor(3, 7);
  1759. od_printf("`white`Box is `bright green`GOOD");
  1760. } else {
  1761. od_set_cursor(3, 7);
  1762. od_printf("`white`Box is `bright red`BAD");
  1763. }
  1764. od_set_cursor(25, 1);
  1765. paws();
  1766. od_save_screen(screen);
  1767. od_clr_scr();
  1768. od_set_cursor(1, 1);
  1769. // Popup
  1770. int run = 1;
  1771. while(run)
  1772. {
  1773. switch(od_popup_menu("Main Menu",
  1774. "^Files|^Electronic Mail|^News|E^xit",
  1775. 20, 5, 0, MENU_NORMAL | MENU_KEEP))
  1776. {
  1777. case 1:
  1778. od_set_cursor(25, 1);
  1779. od_printf("Files");
  1780. od_popup_menu("Files Menu",
  1781. "^Search For Files|^Download|^Upload",
  1782. 30, 7, 1, MENU_NORMAL | MENU_ALLOW_CANCEL | MENU_KEEP);
  1783. break;
  1784. case 2:
  1785. od_set_cursor(25, 1);
  1786. od_printf("Email");
  1787. od_popup_menu("EMail Menu",
  1788. "Get ^New Mail|^Send Mail|Send ^Fax",
  1789. 30, 8, 2, MENU_NORMAL | MENU_ALLOW_CANCEL | MENU_KEEP);
  1790. break;
  1791. case 3:
  1792. od_set_cursor(25, 1);
  1793. od_printf("News");
  1794. od_popup_menu("News Menu",
  1795. "Choose News^Group|^Read News|^Post News",
  1796. 30, 9, 3, MENU_NORMAL | MENU_ALLOW_CANCEL | MENU_KEEP);
  1797. break;
  1798. case 4:
  1799. od_set_cursor(25, 1);
  1800. od_printf("Exit");
  1801. od_popup_menu(NULL, NULL, 0, 0, 0, MENU_DESTROY);
  1802. run = 0;
  1803. break;
  1804. case POPUP_ERROR:
  1805. od_set_cursor(25, 1);
  1806. od_printf("ERROR");
  1807. break;
  1808. case POPUP_ESCAPE:
  1809. od_set_cursor(25, 1);
  1810. od_printf("ESCAPE");
  1811. break;
  1812. case POPUP_LEFT:
  1813. od_set_cursor(25, 1);
  1814. od_printf("LEFT");
  1815. break;
  1816. case POPUP_RIGHT:
  1817. od_set_cursor(25, 1);
  1818. od_printf("RIGHT");
  1819. break;
  1820. default:
  1821. od_set_cursor(25, 1);
  1822. od_printf("DEFAULT");
  1823. break;
  1824. }
  1825. }
  1826. paws();
  1827. od_restore_screen(screen); */
  1828. od_exit(0, false);
  1829. }