// HyperRogue menus // Copyright (C) 2011-2018 Zeno Rogue, see 'hyper.cpp' for details #include "dialogs.cpp" // -- overview -- #define BLACKISH 0x404040 #define REDDISH 0x400000 ld whatever = 0; int PREC(ld x) { ld sh = shiftmul; if(sh > -.2 && sh < .2) x = 10.01; return int(shiftmul * x); } void showOverview() { cmode = sm::ZOOMABLE | sm::OVERVIEW; DEBB(DF_GRAPH, (debugfile,"show overview\n")); if(dialog::infix != "") mouseovers = dialog::infix; else { mouseovers = XLAT("world overview"); mouseovers += " "; mouseovers += XLAT(" kills: %1/%2", its(tkills()), its(killtypes())); mouseovers += XLAT(" $$$: %1", its(gold())); if(hellUnlocked()) { int i1, i2; countHyperstoneQuest(i1, i2); mouseovers += XLAT(" Hyperstone: %1/%2", its(i1), its(i2)); } else mouseovers += XLAT(" Hell: %1/9", its(orbsUnlocked())); } bool pages; generateLandList(isLandIngame); if(dialog::infix != "") { vector filtered; for(eLand l: landlist) { string s = dnameof(l); s += "@"; s += dnameof(treasureType(l)); s += "@"; s += dnameof(nativeOrbType(l)); if(dialog::hasInfix(s)) filtered.push_back(l); } if(filtered.size()) landlist = filtered; } int nl = size(landlist), nlm; int lstart = 0; if(nl > 30) { pages = true; lstart += dialog::handlePage(nl, nlm, (nl+1)/2); } else nlm = nl; int vf = min((vid.yres-64-vid.fsize*2) / nlm, vid.xres/40); eLand curland = getLandForList(cwt.c); getcstat = '0'; for(int i=0; i= 25) col = 0xFFD500; else if(lv >= 10) col = 0x00D500; else if(items[it]) col = 0xC0C0C0; else col = BLACKISH; int c8 = (vf+2)/3; if(displayfrZ(xr*24-c8*6, i0, 1, vf-4, its(items[it]), col, 16)) getcstat = 2000+it; if(!cheater) if(displayfrZ(xr*24, i0, 1, vf-4, its(hiitems[modecode()][it]), col, 16)) getcstat = 2000+it; if(items[it]) col = iinf[it].color; else col = BLACKISH; if(displayfrZ(xr*24+c8*4, i0, 1, vf-4, s0 + iinf[it].glyph, col, 16)) getcstat = 2000+it; if(displayfrZ(xr*24+c8*5, i0, 1, vf-4, XLAT1(iinf[it].name), col, 0)) getcstat = 2000+it; eItem io = nativeOrbType(l); if(io == itShard) { if(items[it] >= 10) col = winf[waMirror].color; else col = BLACKISH; if(displayfrZ(xr*46, i0, 1, vf-4, XLAT1(winf[waMirror].name), col, 0)) getcstat = 3000+waMirror; if(getcstat == 3000+waMirror) mouseovers = XLAT( olrDescriptions[getOLR(io, cwt.c->land)], cwt.c->land, it, treasureTypeUnlock(curland, io)); } else if(io) { if(lv >= 25) col = 0xFFD500; else if(lv >= 10) col = 0xC0C0C0; else col = BLACKISH; if(displayfrZ(xr*46-c8*4, i0, 1, vf-4, its(items[io]), col, 16)) getcstat = 2000+io; if(lv >= 10) col = iinf[io].color; else col = BLACKISH; if(displayfrZ(xr*46-c8, i0, 1, vf-4, s0 + iinf[io].glyph, col, 16)) getcstat = 2000+io; if(displayfrZ(xr*46, i0, 1, vf-4, XLAT1(iinf[io].name), col, 0)) getcstat = 2000+io; if(getcstat == 2000+io) mouseovers = XLAT( olrDescriptions[getOLR(io, curland)], curland, it, treasureTypeUnlock(curland, io)); } } dialog::displayPageButtons(3, pages); keyhandler = [] (int sym, int uni) { int umod = uni % 1000; int udiv = uni / 1000; if(udiv == 1 && umod < landtypes) { const eLand l = eLand(umod); gotoHelp(""); gotoHelpFor(l); if(cheater) { help_extension hex; hex.key = 't'; hex.text = XLAT("teleport"); hex.action = [l] () { cheater++; bool princ = (l == laPrincessQuest); if(princ) { if(kills[moVizier] == 0) kills[moVizier] = 1; princess::forceMouse = true; princess::gotoPrincess = true; cheatMoveTo(laPalace); } else cheatMoveTo(l); canmove = true; if(princ) fullcenter(); popScreen(); popScreen(); }; help_extensions.push_back(hex); } } else if(udiv == 2 && umod < ittypes) { gotoHelp(generateHelpForItem(eItem(umod))); if(cheater) { dialog::helpToEdit(items[umod], 0, 200, 10, 10); dialog::reaction = [] () { if(hardcore) canmove = true; else checkmove(); cheater++; }; } } else if(udiv == 3 && umod < walltypes) gotoHelp(generateHelpForWall(eWall(umod))); else if(uni == SDLK_F1) gotoHelp( "This displays all lands available in the game. " "Bonus lands (available only in separate challenges) " "are not included. Lands written in dark have to be " "unlocked, and lands in dark red are unavailable " "because of using special options. Click on any " "land or item to get information about it. Hover over " "an Orb to know its relation to the current land. " "Cheaters can click to move between lands, and use the " "mousewheel to gain or lose treasures and orbs quickly (Ctrl = precise, Shift = reverse)." ); else if(dialog::handlePageButtons(uni)) ; else if(dialog::editInfix(uni)) ; else if(doexiton(sym, uni)) popScreen(); }; } // -- main menu -- void showMainMenu() { gamescreen(2); getcstat = ' '; dialog::init(XLAT("HyperRogue %1", VER), 0xC00000, 200, 100); dialog::addItem(XLAT("basic configuration"), 'b'); dialog::addItem(XLAT("graphics configuration"), 'g'); dialog::addItem(XLAT("special display modes"), 'd'); dialog::addItem(XLAT("special game modes"), 'm'); #if CAP_SAVE dialog::addItem(XLAT("local highscores"), 't'); #endif dialog::addItem(XLAT("help"), 'h'); dialog::lastItem().keycaption += " / F1"; if(cheater) dialog::addItem(XLAT("cheats"), 'c'); else dialog::addBreak(100); dialog::addItem(XLAT("restart game"), 'r'); dialog::lastItem().keycaption += " / F5"; dialog::addItem(XLAT(inSpecialMode() ? "reset special modes" : "back to the start menu"), 'R'); string q; #if ISMOBILE==1 dialog::addItem(XLAT("visit the website"), 'q'); #else q = quitsaves() ? "save" : "quit"; q = q + " the game"; dialog::addItem(XLAT(q), 'q'); dialog::lastItem().keycaption += " / F10"; #endif if(canmove) q = "review your quest"; else q = "game over screen"; dialog::addItem(XLAT(q), SDLK_ESCAPE); dialog::addItem(XLAT("world overview"), 'o'); if(inv::on) dialog::addItem(XLAT("inventory"), 'i'); #if CAP_ROGUEVIZ dialog::addItem(XLAT("rogueviz menu"), 'u'); #endif #if ISMOBILE==1 #if CAP_ACHIEVE dialog::addItem(XLAT("leaderboards/achievements"), '3'); #endif #endif #if CAP_ANDROIDSHARE dialog::addItem("SHARE", 's'-96); #endif if(!canmove) q = "review the scene"; else if(turncount > 0) q = "continue game"; else q = "play the game!"; dialog::addItem(XLAT(q), ' '); dialog::display(); keyhandler = [] (int sym, int uni) { dialog::handleNavigation(sym, uni); if(sym == SDLK_F1 || sym == 'h') gotoHelp("@"); else if(sym == 'c' && cheater) pushScreen(showCheatMenu); else if(sym == 'b') pushScreen(showBasicConfig); else if(sym == 'g') pushScreen(showGraphConfig); else if(sym == 'd') pushScreen(showDisplayMode); else if(sym == 'm') pushScreen(showChangeMode); else if(uni == 'R') popScreenAll(), pushScreen(showStartMenu); #if CAP_SAVE else if(sym == 't') scores::load(); #endif else if(uni == 'r' || sym == SDLK_F5) { restartGame(); } else if(sym == 'q' || sym == SDLK_F10) { #if ISMOBILE extern void openURL(); openURL(); #else quitmainloop = true; #endif } else if(sym == 'o') { clearMessages(); setAppropriateOverview(); } #if CAP_INV else if(sym == 'i') { clearMessages(); pushScreen(inv::show); } #endif else if(sym == SDLK_ESCAPE) showMissionScreen(); #if ISMOBILE==1 #ifdef HAVE_ACHIEVEMENTS else if(sym == '3') { achievement_final(false); pushScreen(leader::showMenu); } #endif #endif #if CAP_ROGUEVIZ else if(uni == 'u') pushScreen(rogueviz::showMenu); #endif else if(doexiton(sym, uni)) { popScreenAll(); msgs.clear(); } }; } // -- display modes -- void editScale() { dialog::editNumber(vid.scale, .001, 1000, .1, 1, XLAT("scale factor"), XLAT("Scale the displayed model.")); dialog::scaleLog(); } void showDisplayMode() { cmode = sm::SIDE; gamescreen(0); dialog::init(XLAT("special display modes")); const char *wdmodes[6] = {"ASCII", "black", "plain", "Escher", "plain/3D", "Escher/3D"}; const char *mdmodes[6] = {"ASCII", "items only", "items and monsters", "high contrast", "3D", "high contrast/3D"}; dialog::addBoolItem(XLAT("orthogonal projection"), vid.alpha >= 500, '1'); dialog::addBoolItem(XLAT("small Poincaré model"), vid.alpha == 1 && vid.scale < 1, '2'); dialog::addBoolItem(XLAT("big Poincaré model"), vid.alpha == 1 && vid.scale >= 1, '3'); dialog::addBoolItem(XLAT("Klein-Beltrami model"), vid.alpha == 0, '4'); dialog::addSelItem(XLAT("wall display mode"), XLAT(wdmodes[vid.wallmode]), '5'); if(getcstat == '5') mouseovers = XLAT("also hold Alt during the game to toggle high contrast"); dialog::addBoolItem(XLAT("draw the grid"), (vid.grid), '6'); dialog::addBoolItem(XLAT("mark heptagons"), (vid.darkhepta), '7'); dialog::addSelItem(XLAT("3D configuration"), "", '9'); dialog::addSelItem(XLAT("scale factor"), fts(vid.scale), 'z'); dialog::addSelItem(XLAT("monster display mode"), XLAT(mdmodes[vid.monmode]), 'm'); dialog::addBreak(50); #if CAP_EDIT dialog::addBoolItem(XLAT("vector graphics editor"), (false), 'g'); #endif #if CAP_TEXTURE dialog::addBoolItem(XLAT("texture mode"), texture::config.tstate == texture::tsActive, 't'); #endif // display modes #if CAP_RUG dialog::addBoolItem(XLAT("hypersian rug mode"), (rug::rugged), 'u'); #endif #if CAP_MODEL dialog::addBoolItem(XLAT("paper model creator"), (false), 'n'); #endif dialog::addBoolItem(XLAT("models of hyperbolic geometry"), pmodel, 'a'); dialog::addBoolItem(XLAT("history mode"), (conformal::on || conformal::includeHistory), 'A'); // dialog::addBoolItem(XLAT("expansion"), viewdists, 'x'); showAllConfig(); dialog::display(); keyhandler = [] (int sym, int uni) { dialog::handleNavigation(sym, uni); char xuni = uni; // if((xuni >= 'A' && xuni <= 'Z') || (xuni >= 1 && xuni <= 26)) xuni |= 32; if(xuni == 'p') projectionDialog(); if(xuni == 'z') editScale(); #if CAP_TEXTURE if(xuni == 't') pushScreen(texture::showMenu); #endif if(xuni == 'm') { vid.monmode += 60 + (shiftmul > 0 ? 1 : -1); vid.monmode %= 6; } if(xuni == '9') pushScreen(show3D); #if CAP_EDIT else if(xuni == 'g') { pushScreen(mapeditor::showDrawEditor); mapeditor::initdraw(cwt.c); } #endif else if(xuni == 'x') { viewdists = !viewdists; } #if CAP_RUG else if(xuni == 'u') { //if(sphere) projectionDialog(); //else rug::select(); } #endif else if(uni == 'a') pushScreen(conformal::model_menu); else if(uni == 'A') pushScreen(conformal::history_menu); #if CAP_MODEL else if(xuni == 'n') netgen::run(); #endif else gmodekeys(sym, uni); handleAllConfig(sym, xuni); }; } // -- game modes -- void switchHardcore() { if(hardcore && !canmove) { restartGame(); hardcore = false; } else if(hardcore && canmove) { hardcore = false; } else { hardcore = true; canmove = true; hardcoreAt = turncount; } if(hardcore) addMessage("One wrong move, and it is game over!"); else addMessage("Not so hardcore?"); if(pureHardcore()) popScreenAll(); } void help_nochaos() { gotoHelp( "In the Chaos mode, lands change very often, and " "there are no walls between them. " "Some lands are incompatible with this." "\n\nYou need to reach Crossroads IV to unlock the Chaos mode." ); } void showChangeMode() { gamescreen(3); dialog::init(XLAT("special game modes")); // gameplay modes #if CAP_TOUR dialog::addBoolItem(XLAT("Tutorial"), tour::on, 'T'); #endif dialog::addBoolItem(XLAT("experiment with geometry"), geometry || nonbitrunc || viewdists, 'e'); dialog::addBoolItem(XLAT(SHMUPTITLE), (shmup::on || multi::players > 1), 's'); if(!shmup::on) dialog::addSelItem(XLAT("hardcore mode"), hardcore && !pureHardcore() ? XLAT("PARTIAL") : ONOFF(hardcore), 'h'); if(getcstat == 'h') mouseovers = XLAT("One wrong move and it is game over!"); multi::cpid = 0; dialog::addBoolItem(XLAT("Chaos mode"), (chaosmode), 'C'); dialog::addBoolItem(XLAT("peaceful mode"), peace::on, 'p'); dialog::addBoolItem(XLAT("Orb Strategy mode"), (inv::on), 'i'); dialog::addBoolItem(XLAT("pure tactics mode"), (tactic::on), 't'); dialog::addBoolItem(XLAT("Yendor Challenge"), (yendor::on), 'y'); dialog::addBoolItem(XLAT("%1 Challenge", moPrincess), (princess::challenge), 'P'); dialog::addBoolItem(XLAT("random pattern mode"), (randomPatternsMode), 'r'); dialog::addBreak(50); // cheating and map editor dialog::addBoolItem(XLAT("cheat mode"), (cheater), 'c'); #if CAP_EDIT dialog::addBoolItem(XLAT("map editor"), (false), 'm'); #endif dialog::addBreak(50); dialog::addItem(XLAT("return to the game"), 'v'); dialog::display(); keyhandler = [] (int sym, int uni) { dialog::handleNavigation(sym, uni); char xuni = uni; if(xuni == 'v' || sym == SDLK_ESCAPE) popScreen(); else if(uni == 'c') { if(tactic::on && gold()) { addMessage(XLAT("Not available in the pure tactics mode!")); } else if(!cheater) { cheater++; addMessage(XLAT("You activate your demonic powers!")); #if ISMOBILE==0 addMessage(XLAT("Shift+F, Shift+O, Shift+T, Shift+L, Shift+U, etc.")); #endif popScreen(); } else { popScreen(); firstland = princess::challenge ? laPalace : laIce; restartGame(); } } else if(xuni == 'e') runGeometryExperiments(); else if(xuni == 't') { clearMessages(); pushScreen(tactic::showMenu); } else if(xuni == 'y') { clearMessages(); if(yendor::everwon || autocheat) pushScreen(yendor::showMenu); else gotoHelp(yendor::chelp); } else if(xuni == 'p') pushScreen(peace::showMenu); else if(xuni == 'i') { restartGame('i'); } #if CAP_TOUR else if(uni == 'T') { tour::start(); } #endif else if(uni == 'C') { if(chaosUnlocked) restartGame('C'); if(chaosmode) help_nochaos(); } else if(xuni == 'P') { if(!princess::everSaved) addMessage(XLAT("Save %the1 first to unlock this challenge!", moPrincess)); else restartGame('p'); } #if CAP_EDIT else if(xuni == 'm') { if(tactic::on) addMessage(XLAT("Not available in the pure tactics mode!")); else { cheater++; pushScreen(mapeditor::showMapEditor); lastexplore = turncount; addMessage(XLAT("You activate your terraforming powers!")); } } #endif else if(xuni == 's') { #if ISMOBILE==1 restartGame('s'); #else multi::shmupcfg = shmup::on; pushScreen(shmup::showShmupConfig); #endif } else if(xuni == 'h' && !shmup::on) switchHardcore(); else if(xuni == 'r') { firstland = laIce; restartGame('r'); } else if(doexiton(sym, uni)) popScreen(); }; } bool showstartmenu; bool showHalloween() { time_t t = time(NULL); struct tm tm = *localtime(&t); int month = tm.tm_mon + 1; int day = tm.tm_mday; if(month == 10 && day >= 24) return true; if(month == 11 && day <= 7) return true; return false; } int daily; void showStartMenu() { if(!daily) { daily = hrand(10) + 1; if(showHalloween()) daily = 20; } gamescreen(2); getcstat = ' '; dialog::init(); dialog::addInfo(XLAT("Welcome to HyperRogue!")); dialog::addBreak(100); dialog::addBigItem(XLAT("HyperRogue classic"), 'c'); dialog::addInfo(XLAT("explore the world, collect treasures")); dialog::addInfo(XLAT("do not get checkmated")); #if CAP_INV dialog::addBreak(100); dialog::addBigItem(XLAT("Orb Strategy mode"), 'i'); dialog::addInfo(XLAT("use your Orbs in tough situations")); #endif #if CAP_TOUR dialog::addBreak(100); dialog::addBigItem(XLAT("tutorial"), 't'); dialog::addInfo(XLAT("learn about hyperbolic geometry!")); #endif switch(daily) { case 1: #if CAP_SHMUP_GOOD dialog::addBreak(100); dialog::addBigItem(XLAT("shoot'em up mode"), 's'); dialog::addInfo(XLAT("continuous spacetime")); #if CAP_ACHIEVE dialog::addInfo(XLAT("(most achievements are not available)")); #endif #endif break; case 2: dialog::addBreak(100); dialog::addBigItem(XLAT("heptagonal mode"), '7'); dialog::addInfo(XLAT("more curvature")); dialog::addInfo(XLAT("(most achievements are not available)")); break; case 3: dialog::addBreak(100); dialog::addBigItem(XLAT("geometry experiments"), 'g'); dialog::addInfo(XLAT("(most achievements are not available)")); break; case 4: if(chaosUnlocked) { dialog::addBreak(100); dialog::addBigItem(XLAT("Chaos mode"), 'C'); dialog::addInfo(XLAT("(most achievements are not available)")); } break; case 5: dialog::addBreak(100); dialog::addBigItem(XLAT("hypersian rug mode"), 'M'); dialog::addInfo(XLAT("see the true form")); break; case 6: dialog::addBreak(100); dialog::addBigItem(XLAT("texture mode"), 'T'); dialog::addInfo(XLAT("paint pictures")); break; case 20: dialog::addBreak(100); dialog::addBigItem(XLAT1("Halloween"), 'z'); dialog::addInfo(XLAT("Halloween mini-game")); break; } #if CAP_ROGUEVIZ && CAP_TOUR dialog::addBreak(100); dialog::addBigItem(XLAT("RogueViz"), 'r'); dialog::addInfo(XLAT("see the visualizations")); #endif if(have_current_settings()) { dialog::addBreak(100); dialog::addBigItem(XLAT1("use current/saved settings"), SDLK_ESCAPE); } dialog::addBreak(100); dialog::addBigItem(XLAT("main menu"), 'm'); dialog::addInfo(XLAT("more options")); dialog::display(); clearMessages(); timerstart = time(NULL); /* initquickqueue(); int siz = min(vid.xres, vid.yres) / 8; drawMonsterType(moPrincess, NULL, atscreenpos(siz,siz,siz) * spin(-M_PI/4), 0, 0); drawMonsterType(moKnight, NULL, atscreenpos(vid.xres-siz,siz,siz) * spin(-3*M_PI/4), 0, 0); drawItemType(itOrbYendor, NULL, atscreenpos(siz,vid.yres-siz,siz) * spin(M_PI/4), iinf[itOrbYendor].color, 0, false); drawItemType(itKey, NULL, atscreenpos(siz,vid.yres-siz,siz) * spin(M_PI/4), iinf[itKey].color, 0, false); drawItemType(itHyperstone, NULL, atscreenpos(vid.xres-siz,vid.yres-siz,siz) * spin(3*M_PI/4), iinf[itHyperstone].color, 0, false); quickqueue(); */ keyhandler = [] (int sym, int uni) { dialog::handleNavigation(sym, uni); if(uni == 'o') uni = 'i'; else if(uni == 'M') { rug::init(); popScreenAll(); resetModes('c'); clearMessages(); welcomeMessage(); vid.wallmode = 3; vid.monmode = 2; rug::model_distance *= 2; rug::init(); } else if(uni == 'T') { popScreenAll(); resetModes('c'); clearMessages(); welcomeMessage(); using namespace texture; if(config.data.whitetexture() && config.data.loadTextureGL()) { config.tstate = config.tstate_max = tsActive; config.perform_mapping(); config.finish_mapping(); mapeditor::initdraw(cwt.c); pushScreen(showMenu); pushScreen(mapeditor::showDrawEditor); } } else if(uni == 'g') { popScreenAll(); resetModes('c'); clearMessages(); welcomeMessage(); pushScreen(showEuclideanMenu); ewhichscreen = 2; } else if(uni == 'c' || uni == 'i' || uni == 's' || uni == 'C' || uni == '7') { if(uni == 'C' && !chaosUnlocked) { help_nochaos(); return; } popScreenAll(); resetModes(uni); clearMessages(); welcomeMessage(); stampbase = ticks; if(uni == 's') { multi::shmupcfg = shmup::on; pushScreen(multi::showShmupConfig); } } else if(uni == 'z') { popScreenAll(); resetModes('g'); stampbase = ticks; if(!sphere) { specialland = laHalloween; targetgeometry = gSphere; restartGame('g'); vid.alpha = 999; vid.scale = 998; } } #if CAP_TOUR else if(uni == 't') { popScreenAll(); resetModes('c'); tour::start(); } #endif #if CAP_ROGUEVIZ && CAP_TOUR else if(uni == 'r') { tour::slides = rogueviz::rvtour::rvslides; popScreenAll(); tour::start(); } #endif else if(sym == 'm') { popScreen(); pushScreen(showMainMenu); } else if(sym == SDLK_F10) quitmainloop = true; else if(sym == SDLK_F1) gotoHelp("@"); else if(sym == SDLK_ESCAPE || sym == ' ') { popScreen(); timerstart = time(NULL); stampbase = ticks; clearMessages(); welcomeMessage(); } }; } // -- overview -- void setAppropriateOverview() { clearMessages(); if(viewdists) runGeometryExperiments(); else if(tactic::on) pushScreen(tactic::showMenu); else if(yendor::on) pushScreen(yendor::showMenu); else if(peace::on) pushScreen(peace::showMenu); else if(geometry != gNormal && !chaosmode && !(geometry == gEuclid && isCrossroads(specialland)) && !(weirdhyperbolic && specialland == laCrossroads4)) { runGeometryExperiments(); } else { dialog::infix = ""; pushScreen(showOverview); } } int messagelogpos; int timeformat; int stampbase; string gettimestamp(msginfo& m) { char buf[128]; switch(timeformat) { case 0: return its(m.turnstamp); case 1: strftime(buf, 128, "%H:%M:%S", localtime(&m.rtstamp)); return buf; case 2: snprintf(buf, 128, "%2d:%02d", m.gtstamp/60, m.gtstamp % 60); return buf; case 3: { int t = m.stamp - stampbase; bool sign = t < 0; if(sign) t = -t; snprintf(buf, 128, "%2d:%02d.%03d", t/60000, (t/1000) % 60, t % 1000); string s = buf; if(sign) s = "-"+s; return s; } } return ""; } void showMessageLog() { DEBB(DF_GRAPH, (debugfile,"show message log\n")); int lines = vid.yres / vid.fsize - 2; int maxpos = size(gamelog) - lines; messagelogpos = min(messagelogpos, maxpos); messagelogpos = max(messagelogpos, 0); for(int y=0; y