1
0
mirror of https://github.com/zenorogue/hyperrogue.git synced 2024-11-27 14:37:16 +00:00
hyperrogue/hyper.cpp

462 lines
14 KiB
C++
Raw Normal View History

2015-08-08 13:57:52 +00:00
// Hyperbolic Rogue
// Copyright (C) 2011 Zeno Rogue
// This program is free software; you can redistribute it and/or
// modify it under the terms of the GNU General Public License
// as published by the Free Software Foundation; either version 2
// of the License, or (at your option) any later version.
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
// You should have received a copy of the GNU General Public License
// along with this program; if not, write to the Free Software
// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
2017-07-22 23:33:27 +00:00
#ifdef MAC
#define ISMAC 1
#endif
#ifdef LINUX
#define ISLINUX 1
#endif
2016-08-26 09:58:03 +00:00
2017-07-22 23:33:27 +00:00
#ifdef WINDOWS
#define ISWINDOWS 1
#endif
#if ISSTEAM
2017-03-23 10:53:57 +00:00
#define NOLICENSE
2015-08-08 13:57:52 +00:00
#endif
2017-03-23 10:53:57 +00:00
#include "init.cpp"
2015-08-08 13:57:52 +00:00
2017-07-22 23:33:27 +00:00
#if ISLINUX
2016-08-26 09:58:03 +00:00
#include <sys/resource.h>
void moreStack() {
const rlim_t kStackSize = 1 << 28; // 28;
struct rlimit rl;
int result;
result = getrlimit(RLIMIT_STACK, &rl);
if(result == 0) {
if(rl.rlim_cur < kStackSize) {
// rl.rlim_cur = 1 << 19; // kStackSize;
result = setrlimit(RLIMIT_STACK, &rl);
if (result != 0) {
fprintf(stderr, "setrlimit returned result = %d\n", result);
}
}
}
2015-08-08 13:57:52 +00:00
}
2016-08-26 09:58:03 +00:00
#endif
2015-08-08 13:57:52 +00:00
2017-03-23 10:53:57 +00:00
eLand readland(const char *s) {
string ss = s;
if(ss == "II") return laCrossroads2;
if(ss == "III") return laCrossroads3;
if(ss == "IV") return laCrossroads4;
if(ss == "V") return laCrossroads5;
for(int l=0; l<landtypes; l++) if(strstr(linf[l].name, s) != NULL) {
return eLand(l);
break;
}
return laNone;
}
2015-08-08 13:57:52 +00:00
2017-03-23 10:53:57 +00:00
eItem readItem(const char *s) {
string ss = s;
for(int i=0; i<ittypes; i++) if(strstr(iinf[i].name, s) != NULL) {
return eItem(i);
break;
}
return itNone;
}
2016-08-26 09:58:03 +00:00
2017-06-09 01:41:33 +00:00
eMonster readMonster(const char *s) {
string ss = s;
for(int i=0; i<motypes; i++) if(strstr(minf[i].name, s) != NULL) {
return eMonster(i);
break;
}
return moNone;
}
2017-03-23 10:53:57 +00:00
void initializeCLI() {
2016-08-26 09:58:03 +00:00
printf("HyperRogue by Zeno Rogue <zeno@attnam.com>, version " VER "\n");
2016-01-02 10:09:13 +00:00
2015-08-08 13:57:52 +00:00
#ifndef NOLICENSE
printf("released under GNU General Public License version 2 and thus\n");
printf("comes with absolutely no warranty; see COPYING for details\n");
#endif
#ifdef FHS
2017-03-23 10:53:57 +00:00
static string sbuf, cbuf;
2015-08-08 13:57:52 +00:00
if(getenv("HOME")) {
2017-03-23 10:53:57 +00:00
sbuf = getenv("HOME"); sbuf += "/."; sbuf += scorefile;
cbuf = getenv("HOME"); cbuf += "/."; cbuf += conffile;
scorefile = sbuf.c_str();
conffile = cbuf.c_str();
2015-08-08 13:57:52 +00:00
}
#endif
2017-03-23 10:53:57 +00:00
}
2016-01-02 10:09:13 +00:00
2017-03-23 10:53:57 +00:00
int arg::readCommon() {
if(argis("-c")) { PHASE(1); shift(); conffile = args(); }
else if(argis("-s")) { PHASE(1); shift(); scorefile = args(); }
else if(argis("-m")) { PHASE(1); shift(); musicfile = args(); }
2017-07-22 23:33:27 +00:00
#if CAP_SDLAUDIO
2017-03-23 10:53:57 +00:00
else if(argis("-se")) { PHASE(1); shift(); wheresounds = args(); }
#endif
2017-07-22 23:33:27 +00:00
#if CAP_EDIT
2017-03-23 10:53:57 +00:00
else if(argis("-lev")) { shift(); levelfile = args(); }
else if(argis("-pic")) { shift(); picfile = args(); }
else if(argis("-load")) { PHASE(3); shift(); mapstream::loadMap(loadlevel); }
2016-08-26 09:58:03 +00:00
#endif
2017-03-23 10:53:57 +00:00
else if(argis("-canvas")) {
2017-08-06 12:50:16 +00:00
firstland = specialland = laCanvas;
2017-03-23 10:53:57 +00:00
shift();
if(args()[1] == 0) mapeditor::whichCanvas = args()[0];
else mapeditor::canvasback = strtol(args(), NULL, 16);
}
else if(argis("-back")) {
shift(); backcolor = strtol(args(), NULL, 16);
}
2017-07-04 13:38:33 +00:00
else if(argis("-borders")) {
shift(); bordcolor = strtol(args(), NULL, 16);
}
else if(argis("-fore")) {
shift(); forecolor = strtol(args(), NULL, 16);
}
2017-03-23 10:53:57 +00:00
else if(argis("-W2")) {
shift(); cheatdest = readland(args()); autocheat = true;
2017-08-06 12:50:16 +00:00
showstartmenu = false;
2017-03-23 10:53:57 +00:00
}
else if(argis("-W")) {
2017-08-06 12:50:16 +00:00
shift(); firstland = specialland = readland(args());
showstartmenu = false;
2017-03-23 10:53:57 +00:00
}
else if(argis("-I")) {
PHASE(3) cheater++; timerghost = false;
shift(); eItem i = readItem(args());
shift(); items[i] = argi();
}
2017-06-09 01:41:33 +00:00
else if(argis("-M")) {
PHASE(3) cheater++; timerghost = false;
shift(); eMonster m = readMonster(args());
shift(); int q = argi();
printf("m = %s q = %d\n", dnameof(m), q);
restoreGolems(q, m, 7);
}
2017-03-23 10:53:57 +00:00
else if(argis("-L")) {
printf("Treasures:\n");
for(int i=1; i<ittypes; i++)
if(itemclass(eItem(i)) == IC_TREASURE)
printf(" %s\n", iinf[i].name);
printf("\n");
printf("Orbs:\n");
for(int i=1; i<ittypes; i++)
if(itemclass(eItem(i)) == IC_ORB)
printf(" %s\n", iinf[i].name);
printf("\n");
printf("Other items:\n");
for(int i=1; i<ittypes; i++)
if(itemclass(eItem(i)) == IC_OTHER)
printf(" %s\n", iinf[i].name);
printf("\n");
printf("Monsters:\n");
for(int i=1; i<motypes; i++)
printf(" %s\n", minf[i].name);
printf("\n");
printf("Lands:\n");
for(int i=1; i<landtypes; i++)
printf(" %s\n", linf[i].name);
printf("\n");
printf("Walls:\n");
for(int i=0; i<walltypes; i++)
printf(" %s\n", winf[i].name);
printf("\n");
exit(0);
2015-08-08 13:57:52 +00:00
}
2016-08-26 09:58:03 +00:00
2017-07-04 13:38:33 +00:00
else if(argis("-aa")) { PHASEFROM(2); shift(); vid.antialias = argi(); }
else if(argis("-lw")) { PHASEFROM(2); shift(); vid.linewidth = argf(); }
else if(argis("-wm")) { PHASEFROM(2); vid.wallmode = argi(); }
else if(argis("-mm")) { PHASEFROM(2); vid.monmode = argi(); }
2017-03-23 10:53:57 +00:00
#define TOGGLE(x, param, act) \
2017-08-06 12:50:16 +00:00
else if(args()[0] == '-' && args()[1] == x && !args()[2]) { showstartmenu = false; if(curphase == 3) {act;} else {PHASE(2); param = !param;} } \
else if(args()[0] == '-' && args()[1] == x && args()[2] == '1') { showstartmenu = false; if(curphase == 3 && !param) {act;} else {PHASE(2); param = true;} } \
else if(args()[0] == '-' && args()[1] == x && args()[2] == '0') { showstartmenu = false; if(curphase == 3 && param) {act;} else {PHASE(2); param = false;} }
2017-03-23 10:53:57 +00:00
TOGGLE('o', vid.usingGL, switchGL())
TOGGLE('C', chaosmode, restartGame('C'))
TOGGLE('7', purehepta, restartGame('7'))
TOGGLE('f', vid.full, switchFullscreen())
TOGGLE('T', tactic::on, restartGame('t'))
TOGGLE('S', shmup::on, restartGame('s'))
TOGGLE('H', hardcore, switchHardcore())
TOGGLE('R', randomPatternsMode, restartGame('r'))
2017-08-06 12:50:16 +00:00
TOGGLE('i', inv::on, restartGame('i'))
2015-08-08 13:57:52 +00:00
2017-08-06 12:50:16 +00:00
else if(argis("-peace")) {
peace::otherpuzzles = true;
if(curphase == 3) restartGame('P');
else peace::on = true;
}
else if(argis("-pmem")) {
peace::otherpuzzles = false;
if(curphase == 3) restartGame('P');
else peace::on = true;
}
else if(argis("-geo")) {
if(curphase == 3) {
shift(); targetgeometry = (eGeometry) argi();
restartGame('g');
}
else {
PHASE(2); shift(); geometry = targetgeometry = (eGeometry) argi();
}
}
2017-03-23 10:53:57 +00:00
else if(argis("-qs")) {
autocheat = true;
shift(); fp43.qpaths.push_back(args());
2016-01-02 10:09:13 +00:00
}
2017-03-23 10:53:57 +00:00
else if(argis("-fix")) {
fixseed = true; autocheat = true;
}
2017-07-16 21:00:55 +00:00
else if(argis("-fixx")) {
fixseed = true; autocheat = true;
shift(); startseed = argi();
}
else if(argis("-steplimit")) {
fixseed = true; autocheat = true;
shift(); steplimit = argi();
}
2017-03-23 10:53:57 +00:00
else if(argis("-qpar")) {
int p;
shift(); sscanf(args(), "%d,%d,%d",
&p, &quotientspace::rvadd, &quotientspace::rvdir
);
autocheat = true;
fp43.init(p);
}
2017-06-09 01:41:33 +00:00
else if(argis("-tpar")) {
shift(); sscanf(args(), "%d,%d,%d",
&torusconfig::qty,
&torusconfig::dx,
&torusconfig::dy
);
}
2017-03-23 10:53:57 +00:00
else if(argis("-cs")) {
shift();
fieldpattern::matrix M = fp43.strtomatrix(args());
fieldpattern::subpathid = fp43.matcode[M];
fieldpattern::subpathorder = fp43.order(M);
autocheat = true;
}
else if(argis("-csp")) {
autocheat = true;
fp43.findsubpath();
}
else if(argis("-fi")) {
fieldpattern::info();
exit(0);
}
else if(argis("-P")) {
PHASE(2); shift();
vid.scfg.players = argi();
}
else if(argis("-PM")) {
PHASEFROM(2); shift(); pmodel = eModel(argi());
2017-03-23 10:53:57 +00:00
}
else if(argis("-offline")) offlineMode = true;
else if(argis("-debugf")) {
debugfile = fopen("hyperrogue-debug.txt", "w");
shift(); debugflags = argi();
}
else if(argis("-debuge")) {
debugfile = stderr;
shift(); debugflags = argi();
}
else if(argis("-ch")) { autocheat = true; }
else if(argis("-zoom")) {
PHASEFROM(2); shift(); vid.scale = argf();
2017-03-23 10:53:57 +00:00
}
else if(argis("-Y")) {
yendor::on = true;
shift(); yendor::challenge = argi();
}
else if(argis("-r")) {
PHASEFROM(2);
2017-03-23 10:53:57 +00:00
shift();
int clWidth=0, clHeight=0, clFont=0;
sscanf(args(), "%dx%dx%d", &clWidth, &clHeight, &clFont);
if(clWidth) vid.xres = clWidth;
if(clHeight) vid.yres = clHeight;
if(clFont) vid.fsize = clFont;
}
else if(argis("--version") || argis("-v")) {
printf("HyperRogue version " VER "\n");
exit(0);
}
else if(argis("--run")) {
PHASE(3); mainloop(); quitmainloop = false;
}
2017-07-22 23:33:27 +00:00
#if CAP_TOUR
2017-04-08 15:18:29 +00:00
else if(argis("--tour")) {
PHASE(3); tour::start();
}
else if(argis("--presentation")) {
PHASE(3); tour::texts = false;
tour::start();
}
#endif
2017-03-23 10:53:57 +00:00
else if(argis("--draw")) {
PHASE(3); drawscreen();
}
else if(argis("--exit")) {
PHASE(3); printf("Success.\n");
exit(0);
}
else if(argis("-gencells")) {
PHASE(3); shift();
printf("Generating %d cells...\n", argi());
celllister cl(cwt.c, 50, argi(), NULL);
printf("Cells generated: %d\n", size(cl.lst));
for(int i=0; i<size(cl.lst); i++)
setdist(cl.lst[i], 7, NULL);
}
else if(argis("-sr")) {
PHASEFROM(2);
2017-03-23 10:53:57 +00:00
shift(); sightrange = argi();
}
2017-07-22 23:33:27 +00:00
#if CAP_SDL
2017-03-23 10:53:57 +00:00
else if(argis("-pngshot")) {
PHASE(3); shift();
printf("saving PNG screenshot to %s\n", args());
saveHighQualityShot(args());
}
#endif
2017-03-23 10:53:57 +00:00
else if(argis("-svgsize")) {
shift(); sscanf(args(), "%d/%d", &svg::svgsize, &svg::divby);
}
2017-06-18 16:51:46 +00:00
else if(argis("-svgfont")) {
shift(); svg::font = args();
// note: use '-svgfont latex' to produce text output as: \myfont{size}{text}
// (this is helpful with Inkscape's PDF+TeX output feature; define \myfont yourself)
}
2017-03-23 10:53:57 +00:00
else if(argis("-pngsize")) {
shift(); sscanf(args(), "%d", &pngres);
}
else if(argis("-pngformat")) {
shift(); sscanf(args(), "%d", &pngformat);
}
2017-03-23 10:53:57 +00:00
else if(argis("-svggamma")) {
shift(); svg::gamma = argf();
}
else if(argis("-svgshot")) {
PHASE(3); shift();
printf("saving SVG screenshot to %s\n", args());
svg::render(args());
}
else if(argis("--help") || argis("-h")) {
printf("Press F1 while playing to get ingame options.\n\n");
printf("HyperRogue accepts the following command line options:\n");
printf(" -c FILE - use the specified configuration file\n");
printf(" -s FILE - use the specified highscore file\n");
printf(" -m FILE - use the specified soundtrack (music)\n");
printf(" -se DIR - the directory containing sound effects\n");
printf(" -lev FILE - use the specified filename for the map editor (without loading)\n");
printf(" -load FILE - use the specified filename for the map editor\n");
printf(" -canvas COLOR - set background color or pattern code for the canvas\n");
printf(" --version, -v - show the version number\n");
printf(" --help, -h - show the commandline options\n");
printf(" -f* - toggle fullscreen mode\n");
printf(" -wm n, -mm n - start in the given wallmode or monmode\n");
printf(" -r WxHxF - use the given resolution and font size\n");
printf(" -o* - toggle the OpenGL mode\n");
printf(" -W LAND - start in the given land (cheat)\n");
printf(" -W2 LAND - make the given land easy to find (also turns on autocheat)\n");
printf(" -ch - auto-enable cheat mode\n");
printf(" -geo n - switch geometry (1=Euclidean, 2=spherical, 3=elliptic, 4/5=quotient)\n");
printf(" -qs <desc> - fieldpattern: quotient by the given <desc> (must be followed by qpar)\n");
printf(" -qpar <prime> - fieldpattern: use the given prime instead of 43\n");
printf(" -cs <desc> - fieldpattern: set subpath to the given <desc> (cannot be followed by qpar)\n");
printf(" -csp - fieldpattern: find the subpath of order <prime> (cannot be followed by qpar)\n");
printf(" -S* - toggle Shmup\n");
printf(" -P n - switch Shmup number of players (n=1..7)\n");
printf(" -PM - switch the model index\n");
printf(" -H* - toggle Hardcore\n");
printf(" -T* - toggle Tactical\n");
printf(" -7* - toggle heptagonal mode\n");
printf(" -C* - toggle Chaos mode\n");
printf(" -R* - toggle Random Pattern\n");
printf(" -Y id - enable Yendor, level id\n");
printf(" -D - disable all the special game modes\n");
printf(" -L - list of features\n");
printf(" -debugf 7 - output debugging information to hyperrogue-debug.txt\n");
printf(" -debuge 7 - output debugging information to stderr\n");
printf(" -offline - don't connect to Steam (for Steam versions)\n");
printf(" -I ITEM n - start with n of ITEM (activates cheat and disables ghosts)\n");
printf(" -fix - fix the seed\n");
printf("Toggles: -o0 disables, -o1 enables, -o switches");
printf("Not all options are documented, see hyper.cpp");
exit(0);
}
else if(ca::readArg()) ;
else return 1;
return 0;
}
2015-08-08 13:57:52 +00:00
2017-07-10 18:47:38 +00:00
hookset<bool(int argc, char** argv)> *hooks_main;
2017-07-04 13:38:33 +00:00
#ifndef NOMAIN
2017-03-23 10:53:57 +00:00
int main(int argc, char **argv) {
2017-07-10 18:47:38 +00:00
if(callhandlers(false, hooks_main, argc, argv)) return 0;
2017-07-22 23:33:27 +00:00
#if !ISWEB
#if ISLINUX
2017-03-23 10:53:57 +00:00
moreStack();
#endif
arg::init(argc, argv);
initializeCLI();
2016-08-26 09:58:03 +00:00
#endif
2017-03-23 10:53:57 +00:00
initAll();
arg::read(3);
2017-08-06 12:50:16 +00:00
if(showstartmenu)
pushScreen(showStartMenu);
2016-08-26 09:58:03 +00:00
mainloop();
2017-03-23 10:53:57 +00:00
finishAll();
profile_info();
2015-08-08 13:57:52 +00:00
return 0;
}
2017-07-04 13:38:33 +00:00
#endif
2017-03-23 10:53:57 +00:00
2017-07-22 23:33:27 +00:00
#if CAP_COMMANDLINE
2017-07-10 18:47:38 +00:00
purehookset hooks_config;
hookset<int()> *hooks_args;
2017-03-23 10:53:57 +00:00
namespace arg {
int argc; char **argv;
2017-07-10 18:47:38 +00:00
auto ah = addHook(hooks_args, 0, readCommon);
2017-03-23 10:53:57 +00:00
void read(int phase) {
curphase = phase;
2017-07-10 18:47:38 +00:00
callhooks(hooks_config);
2017-03-23 10:53:57 +00:00
while(argc) {
2017-07-10 18:47:38 +00:00
for(auto& h: *hooks_args) {
int r = h.second(); if(r == 2) return; if(r == 0) { lshift(); goto cont; }
}
2017-03-23 10:53:57 +00:00
printf("Unknown option: %s\n", args());
exit(3);
2017-07-10 18:47:38 +00:00
cont: ;
2017-03-23 10:53:57 +00:00
}
}
}
#endif