// Hyperbolic Rogue -- Heads-Up Display // Copyright (C) 2011-2019 Zeno Rogue, see 'hyper.cpp' for details /** \file hud.cpp * \brief Heads-Up display: items collected, monsters killed, radar, etc. */ #include "hyper.h" namespace hr { EX purehookset hooks_stats; EX int monsterclass(eMonster m) { if(isFriendly(m) || m == moTortoise) return 1; else if(isMonsterPart(m)) return 2; else return 0; } EX int glyphclass(int i) { if(i < ittypes) { eItem it = eItem(i); return itemclass(it) == IC_TREASURE ? 0 : 1; } else { eMonster m = eMonster(i-ittypes); return monsterclass(m) == 0 ? 2 : 3; } } EX int subclass(int i) { if(i < ittypes) return itemclass(eItem(i)); else return monsterclass(eMonster(i-ittypes)); } #define GLYPH_MARKTODO 1 #define GLYPH_MARKOVER 2 #define GLYPH_LOCAL 4 #define GLYPH_IMPORTANT 8 #define GLYPH_NONUMBER 16 #define GLYPH_DEMON 32 #define GLYPH_RUNOUT 64 #define GLYPH_INPORTRAIT 128 #define GLYPH_LOCAL2 256 #define GLYPH_TARGET 512 #define GLYPH_INSQUARE 1024 #define GLYPH_INLANDSCAPE 2048 #define GLYPH_ACTIVE 4096 #if HDR enum eGlyphsortorder { gsoFirstTop, gsoFirstBottom, gsoLastTop, gsoLastBottom, gsoLand, gsoValue, gsoMAX }; #endif EX eGlyphsortorder glyphsortorder; int zero = 0; int& ikmerge(int i) { if(i < ittypes) return items[i]; else if(i == ittypes) return zero; else return kills[i-ittypes]; } bool ikappear(int i) { if(i == itInventory && inv::on) return true; if(i == itCrossbow && bow::crossbow_mode()) return true; return ikmerge(i); } const int glyphs = ittypes + motypes; int gfirsttime[glyphs], glasttime[glyphs], gcopy[glyphs], ikland[glyphs]; int glyphorder[glyphs]; int glyphphase[glyphs]; int glyph_lastticks; void updatesort() { for(int i=0; i= ittypes) || i == itTerra)) { int a = gp; gp += (ticks - glyph_lastticks); if(a/500 != gp/500) gp = gp/500*500; } } glyph_lastticks = ticks; } EX void preparesort() { for(int i=0; i gfirsttime[j]; if(glyphsortorder == gsoLastTop) return glasttime[i] > glasttime[j]; if(glyphsortorder == gsoLastBottom) return glasttime[i] < glasttime[j]; if(glyphsortorder == gsoValue) return ikmerge(i) > ikmerge(j); if(glyphsortorder == gsoLand) return ikland[i] < ikland[j]; return 0; } int glyphflags(int gid) { int f = 0; if(gid < ittypes) { eItem i = eItem(gid); if(itemclass(i) == IC_NAI && i != itFatigue && i != itCrossbow) f |= GLYPH_NONUMBER; if(i == itCrossbow) { if(items[i] == 0) f |= GLYPH_NONUMBER; if(bow::fire_mode) f |= GLYPH_ACTIVE; } if(isElementalShard(i)) { f |= GLYPH_LOCAL | GLYPH_INSQUARE; if(i == localshardof(cwt.at->land)) f |= GLYPH_LOCAL2; } if(i == treasureType(cwt.at->land) || daily::on) f |= GLYPH_LOCAL | GLYPH_LOCAL2 | GLYPH_IMPORTANT | GLYPH_INSQUARE | GLYPH_INPORTRAIT | GLYPH_INLANDSCAPE; if(i == itHolyGrail) { if(items[i] >= 3 && !inv::on) f |= GLYPH_MARKOVER; } else if(itemclass(i) == IC_TREASURE) { if(items[i] >= 25 && items[i] < 100 && !inv::on) f |= GLYPH_MARKOVER; else if(items[i] < 10) f |= GLYPH_MARKTODO; } else { f |= GLYPH_IMPORTANT | GLYPH_INSQUARE | GLYPH_INPORTRAIT | GLYPH_INLANDSCAPE; if(itemclass(i) == IC_ORB && items[i] < 10) f |= GLYPH_RUNOUT; } if(i == orbToTarget) f |= GLYPH_TARGET; if(!less_in_portrait) f |= GLYPH_INPORTRAIT; if(!less_in_landscape) f |= GLYPH_INLANDSCAPE; } else { eMonster m = eMonster(gid-ittypes); if(m == moLesser) f |= GLYPH_IMPORTANT | GLYPH_DEMON | GLYPH_INPORTRAIT | GLYPH_INSQUARE | GLYPH_INLANDSCAPE; int isnat = isNative(cwt.at->land, m); if(isnat) f |= GLYPH_LOCAL | GLYPH_IMPORTANT | GLYPH_INPORTRAIT | GLYPH_INSQUARE | GLYPH_INLANDSCAPE; if(isnat == 2) f |= GLYPH_LOCAL2; if(m == monsterToSummon) f |= GLYPH_TARGET; if(!less_in_landscape) f |= GLYPH_INLANDSCAPE; } return f; } EX bool graphglyph(bool isMonster) { // if(GDIM == 3) return false; if(vrhr::active()) return false; return vid.graphglyph == 2 || (vid.graphglyph == 1 && (isMonster ? mmmon : mmitem)); } bool displayglyph(int cx, int cy, int buttonsize, char glyph, color_t color, int qty, int flags, int id) { bool b = mousex >= cx && mousex < cx+buttonsize && mousey >= cy-buttonsize/2 && mousey <= cy-buttonsize/2+buttonsize; int glsize = buttonsize; if(glyph == '%' || glyph == 'M' || glyph == 'W') glsize = glsize*4/5; int d = ticks - glasttime[id]; double zoom = (d <= 250 && d >= 0) ? 1.25 - .001 * d : 1; glsize = int(glsize * zoom); bool isMonster = (id >= ittypes); if(graphglyph(isMonster)) { initquickqueue(); poly_outline = OUTLINE_DEFAULT; if(isMonster) { eMonster m = eMonster(id - ittypes); double bsize = buttonsize * 2/3; if(m == moKrakenH) bsize /= 3; if(m == moKrakenT || m == moDragonTail) bsize /= 2; if(m == moSlime) bsize = (2*bsize+1)/3; transmatrix V = atscreenpos(cx+buttonsize/2, cy, bsize*zoom); if(isWorm(m) && cgi.wormscale != 1) for(int i=0; i> 2; drawMonsterType(m, NULL, shiftless(V), mcol, glyphphase[id]/500.0, NOCOLOR); } else { eItem it = eItem(id); double bsize = buttonsize / 2; if(glyph == '*') bsize *= 2; if(glyph == '%') bsize *= 2; if(glyph == '!') bsize *= 2; if(glyph == '?') bsize *= 2; if(glyph == ';') bsize *= 2; if(glyph == '$') bsize = (bsize*5+2)/3; if(glyph == 'o') bsize = (bsize*3+1)/2; if(glyph == 'c') bsize = (bsize*3+1)/2; if(glyph == 't') bsize = bsize*5/2; if(glyph == '(') bsize = bsize*2.5; if(glyph == '/') bsize = bsize*2; if(glyph == ']') bsize = bsize*1.5; if(glyph == 'x') bsize = bsize*1.5; if(it == itWarning) bsize *= 2; if(it == itCrossbow) bsize *= 2; if(it == itBombEgg || it == itTrollEgg || it == itDodeca) bsize = bsize*3/2; int icol = color; icol -= (color & 0xFCFCFC) >> 2; int ic = itemclass(it); bsize = bsize * zoom; transmatrix V = atscreenpos(cx+buttonsize/2, cy, bsize); double t = (ic == IC_ORB || ic == IC_NAI) ? ticks*2 : ((glyph == 't' && qty%5) || it == itOrbYendor) ? ticks/2 : it == itTerra ? glyphphase[id] * 3 * M_PI + 900 * M_PI: glyphphase[id] * 2; drawItemType(it, NULL, shiftless(V), icol, t, false); int c1 = max(color_diff(color, backcolor), color_diff(color, bordcolor)); if(c1 < 0x80) { color = orb_auxiliary_color(it); } } int c1 = max(color_diff(color, backcolor), color_diff(color, bordcolor)); if(c1 < 0x80) color = gradient(color, forecolor, 0, .4, 1); sortquickqueue(); quickqueue(); } else if(zh_ascii) { const char* zh; if(isMonster) { eMonster m = eMonster(id - ittypes); zh = XLAT1_acc(minf[m].name, 8); } else { eItem it = eItem(id); zh = XLAT1_acc(iinf[it].name, 8); } if(!zh) goto non_zh; dynamicval df(cfont, cfont_chinese); displaystr(cx + buttonsize/2, cy, 0, glsize, zh, darkenedby(color, b?0:1), 0); } else non_zh: if(glyph == '*') displaychr(cx + buttonsize/2, cy+buttonsize/4, 0, glsize*3/2, glyph, darkenedby(color, b?0:1)); else displaychr(cx + buttonsize/2, cy, 0, glsize, glyph, darkenedby(color, b?0:1)); string fl = ""; string str = its(qty); if(flags & GLYPH_TARGET) fl += "!"; if(flags & GLYPH_LOCAL2) fl += "+"; else if(flags & GLYPH_LOCAL) fl += "-"; if(flags & GLYPH_DEMON) fl += "X"; if(flags & GLYPH_ACTIVE) fl += "A"; if(flags & GLYPH_MARKOVER) str += "!"; if(fl != "") displaystr(cx + buttonsize, cy-buttonsize/2 + buttonsize/4, 0, buttonsize/2, fl, darkenedby(color, 0), 16); if(flags & GLYPH_NONUMBER) str = ""; int bsize = (qty < 10 && (flags & (GLYPH_MARKTODO | GLYPH_RUNOUT))) ? buttonsize*3/4 : qty < 100 ? buttonsize / 2 : buttonsize / 3; if(id == moMutant + ittypes && clearing::imputed.nonzero()) { bignum bn = clearing::imputed + qty; str = short_form(bn); bsize = buttonsize / 4; } if(str != "") { if(textwidth(bsize, str) < buttonsize) displayfr(cx + buttonsize, cy + buttonsize/2 - bsize/2, 1, bsize, str, color, 16); else displayfr(cx, cy + buttonsize/2 - bsize/2, 1, bsize, str, color, 0); } return b; } void displayglyph2(int cx, int cy, int buttonsize, int i) { char glyph = i < ittypes ? iinf[i].glyph : minf[i-ittypes].glyph; color_t color = i < ittypes ? iinf[i].color : minf[i-ittypes].color; int imp = glyphflags(i); if(displayglyph(cx, cy, buttonsize, glyph, color, ikmerge(i), imp, i)) { instat = true; getcstat = SDLK_F1; if(i < ittypes) { eItem it = eItem(i); int t = itemclass(it); if(t == IC_TREASURE) mouseovers = XLAT("treasure collected: %1", it); if(t == IC_OTHER) mouseovers = XLAT("objects found: %1", it); if(t == IC_NAI) mouseovers = XLATT1(it); if(t == IC_ORB) mouseovers = XLAT("orb power: %1", eItem(i)); if(it == itGreenStone) { mouseovers += XLAT(" (click to drop)"); getcstat = 'g'; } if(it == itStrongWind) { mouseovers += XLAT(" (click to use)"); getcstat = 't'; } if(it == itInventory) { mouseovers += XLAT(" (click to use)"); getcstat = 'i'; } if(it == itCrossbow) { if(items[it]) mouseovers += XLAT(" (turns to reload: %1)", its(items[it])); else if(items[it] && bow::fire_mode) mouseovers += XLAT(" (fire mode on / turns to reload: %1)", its(items[it])); else if(bow::fire_mode) mouseovers += XLAT(" (fire mode on)"); else mouseovers += XLAT(" (click to fire)"); getcstat = 'f'; } if(imp & GLYPH_LOCAL) mouseovers += XLAT(" (local treasure)"); help = generateHelpForItem(it); } else { eMonster m = eMonster(i-ittypes); if(isMonsterPart(m)) mouseovers = s0 + XLAT("parts destroyed: %1", m); else if(isFriendly(m) && isNonliving(m)) mouseovers = s0 + XLAT("friends destroyed: %1", m); else if(isFriendly(m)) mouseovers = s0 + XLAT("friends killed: %1", m); else if(isNonliving(m)) mouseovers = s0 + XLAT("monsters destroyed: %1", m); else if(m == moTortoise) mouseovers = s0 + XLAT("animals killed: %1", m); else mouseovers = s0 + XLAT("monsters killed: %1", m); if(imp & GLYPH_LOCAL2) mouseovers += XLAT(" (killing increases treasure spawn)"); else if(imp & GLYPH_LOCAL) mouseovers += XLAT(" (appears here)"); help = generateHelpForMonster(m); } } } EX bool nohud, nomenukey, nomsg; EX hookset hooks_prestats; #if CAP_SHAPES void drawMobileArrow(int i) { int dir = i; cell *c = cwt.at->move(i); if(!c) return; transmatrix T = currentmap->adj(cwt.at, i); // color_t col = getcs().uicolor; // col -= (col & 0xFF) >> 1; bool invalid = !legalmoves[dir]; color_t col = cellcolor(c); if(col == OUTLINE_NONE) col = 0xC0C0C0FF; col -= (col & 0xFF) >> 1; if(invalid) col -= (col & 0xFF) >> 1; if(invalid) col -= (col & 0xFF) >> 1; poly_outline = OUTLINE_DEFAULT; // transmatrix m2 = Id; ld scale = vid.mobilecompasssize * (sphere ? 7 : euclid ? 6 : 5); // m2[0][0] = scale; m2[1][1] = scale; m2[2][2] = 1; shiftmatrix U = ggmatrix(cwt.at); shiftpoint H = tC0(U); shiftmatrix Centered = rgpushxto0(H); hyperpoint P = inverse_shift(Centered, U * T * C0); double alpha = atan2(P[1], P[0]); using namespace shmupballs; double dx = xmove + rad*(1+SKIPFAC-.2)/2 * cos(alpha); double dy = yb + rad*(1+SKIPFAC-.2)/2 * sin(alpha); queuepolyat(shiftless(atscreenpos(dx, dy, scale) * spin(-alpha)), cgi.shArrow, col, PPR::MOBILE_ARROW); } #endif EX bool nofps = false; EX color_t crosshair_color = 0xFFFFFFC0; EX ld crosshair_size = 0; EX bool long_kills; /** HUD margin in pixels. In mobile devices we need this margin because the corners are hidden. side==0: top, side==1: bottom */ EX int hud_margin(int side) { if(ISIOS) return 24; if(ISANDROID) return 8; if(ISFAKEMOBILE) return 8; return 0; } EX void draw_crosshair() { auto& cd = current_display; auto xc = cd->xcenter; auto yc = cd->ycenter; flat_model_enabler fme; if(crosshair_color && crosshair_size > 0) { initquickqueue(); vid.linewidth = 1; queueline(shiftless(tC0(atscreenpos(xc - crosshair_size, yc, 1))), shiftless(tC0(atscreenpos(xc + crosshair_size, yc, 1))), crosshair_color).prio = PPR::SUPERLINE; queueline(shiftless(tC0(atscreenpos(xc, yc - crosshair_size, 1))), shiftless(tC0(atscreenpos(xc, yc + crosshair_size, 1))), crosshair_color).prio = PPR::SUPERLINE; quickqueue(); } return; } EX bool less_in_portrait, less_in_landscape; EX string mode_description() { string md; if(autocheat) md += " god"; if(casual) md += " casual"; else if(cheater) md += " cheat"; if(inv::on) md += " inv"; if(tour::on) md += " tour"; if(shmup::on) md += " shmup"; if(multi::players > 1) md += " P" + its(multi::players); if(pureHardcore()) md += " hardcore"; else if(hardcore) md += " partial hardcore"; if(daily::on) md += " strange"; if(bow::crossbow_mode()) md += " b/" + bow::bowName[bow::style]; if(land_structure != default_land_structure()) md += " " + land_structure_name(true); if(randomPatternsMode) md += " RPM"; if(use_custom_land_list) md += " custom"; if(geometry != gNormal || !BITRUNCATED) md = md + " " + full_geometry_name(); return md; } EX string mode_description1() { string md = mode_description(); if(md == "") return "standard"; return md.substr(1); } EX bool radar_drawn; EX void drawStats() { if(vid.stereo_mode == sLR) return; draw_crosshair(); if(nohud) return; if(callhandlers(false, hooks_prestats)) return; if(viewdists && show_distance_lists) get_expansion().view_distances_dialog(); if(current_display->sidescreen) return; first_cell_to_draw = true; bool h = hide_player(); instat = false; bool cornermode = (vid.xres > vid.yres * 85/100 && vid.yres > vid.xres * 85/100); bool hyb = mhybrid; #if MAXMDIM >= 4 if(mhybrid) hybrid::draw_underlying(!cornermode); #endif { radar_drawn = false; if(vid.radarsize > 0 && h) #if CAP_RACING if(!racing::on) #endif if(!peace::on) if(!(cmode & sm::MISSION)) radar_drawn = true; if(radar_drawn) draw_radar(cornermode); flat_model_enabler fme; if(haveMobileCompass()) { initquickqueue(); using namespace shmupballs; calc(); #if CAP_QUEUE queuecircle(xmove, yb, rad, 0xFF0000FF); queuecircle(xmove, yb, rad*SKIPFAC, legalmoves[cwt.at->type] ? 0xFF0000FF : 0xFF000080 ); #endif #if CAP_SHAPES for(int i=0; itype; i++) drawMobileArrow(i); #endif if(hypot(mousex-xmove, mousey-yb) <= rad) getcstat = '-'; quickqueue(); } if(racing::on) #if CAP_RACING racing::drawStats(); #else {} #endif else if(!hr_hud_enabled) {} else if(cornermode) { int bycorner[4]; for(int u=0; u<4; u++) bycorner[u] = 0; for(int i=0; i rad) { spots++; } if(spots >= bycorner[cor] && spots >= 3) { int next = 0; vector glyphstoshow; for(int i=0; i rad) { if(next >= isize(glyphstoshow)) break; int cx = u; int cy = v + s/2; if(cor&1) cx = vid.xres-1-s-cx; if(cor&2) cy = vid.yres-1-cy; displayglyph2(cx, cy, s, glyphstoshow[next++]); } break; } } } } else { auto& cd = current_display; bool portrait = vid.xres < vid.yres; int colspace = portrait ? (cd->ycenter - cd->scrsize - 3 * vid.fsize) : (vid.xres - vid.yres - 16) / 2; int radar_size = 0; if(radar_drawn) radar_size = 2 * vid.radarsize + 10 + 3.5 * vid.fsize; if(hyb && hybrid::underlying_scale) radar_size = max(radar_size, min(cd->xsize, cd->ysize) * hybrid::underlying_scale + 10 + 3.5 * vid.fsize); int rowspace = portrait ? vid.xres - 16 : vid.yres - max(radar_size, vid.fsize * (vid.msgleft ? 9 : 4)); int colid[4], rowid[4]; int maxbyclass[4]; for(int z=0; z<4; z++) maxbyclass[z] = 0; flagtype flag = portrait ? GLYPH_INPORTRAIT : GLYPH_INLANDSCAPE; for(int i=0; i columns) { vid.killreduction++; continue; } coltaken = 0; } colid[z] = coltaken, rowid[z] = 0, coltaken += (maxbyclass[z] + rows-1) / rows; } if(coltaken > columns) { vid.killreduction++; continue; } break; } if(buttonsize <= vid.fsize*3/4) { imponly = true; buttonsize = minsize; rows = rowspace / buttonsize; if(!rows) return; colid[0] = 0; colid[2] = portrait ? 1 : 0; } updatesort(); stable_sort(glyphorder, glyphorder+glyphs, glyphsort); for(int i0=0; i0= rows) rowid[z] = 0, colid[z]++; displayglyph2(cx, cy, buttonsize, i); } } } glflush(); calcparam(); int top_y = vid.fsize + hud_margin(0); string s0; if(racing::on) { #if CAP_RACING using namespace racing; color_t col; if(ticks >= race_start_tick) col = 0x00FF00; else if(ticks >= race_start_tick - 2000) col = 0xFFFF00; else col = 0xFF0000; for(int i=0; i x(vid.fsize, vid.fsize*2); if(displayButtonS(vid.xres - 8, top_y, racetimeformat(ticks - race_start_tick), col, 16, vid.fsize)) getcstat = 'o'; for(int i=0; i> 8), 16, vid.fsize)) getcstat = 'o'; } else { int comp = get_percentage(i); if(displayButtonS(vid.xres - 8, top_y + vid.fsize * (2+2*i), its(comp) + "%", (getcs().uicolor >> 8), 16, vid.fsize)) getcstat = 'o'; } if(displayButtonS(vid.xres - 8, top_y + vid.fsize * (3+2*i), fts_fixed(shmup::pc[i]->vel * SCALE * 1000/600, 2), (getcs().uicolor >> 8), 16, vid.fsize)) getcstat = 'o'; } #endif } else if(!peace::on) { string scoreline = XLAT("score: %1", its(gold())); if(displayButtonS(vid.xres - 8, top_y, scoreline, forecolor, 16, vid.fsize)) { mouseovers = XLAT("Your total wealth"), instat = true, getcstat = SDLK_F1, help = helptitle(XLAT("Your total wealth"), 0xFFD500) + XLAT( "The total value of the treasure you have collected.\n\n" "Every world type contains a specific type of treasure, worth 1 $$$; " "your goal is to collect as much treasure as possible, but every treasure you find " "causes more enemies to hunt you in its native land.\n\n" "Orbs of Yendor are worth 50 $$$ each.\n\n" ); } string s = XLAT("kills: %1", its(tkills())); long_kills = false; int siz = vid.fsize; if(cwt.at->land == laClearing && clearing::imputed.approx_ld() >= 100000) { long_kills = true; s = XLAT("leaves cut: %1", (bignum(kills[moMutant]) + clearing::imputed).get_str(200)); if(mouseovers == standard_help()) mouseovers = " "; while(siz > 4 && textwidth(siz, s) > vid.xres - textwidth(vid.fsize, scoreline)) siz--; } if(displayButtonS(8, top_y, s, forecolor, 0, siz)) { instat = true; getcstat = SDLK_F1; if(long_kills) { mouseovers = " "; help = generateHelpForMonster(moMutant); } else { mouseovers = XLAT("Your total kills")+": " + its(tkills()), help = helptitle(XLAT("Your total kills") + ": " + its(tkills()), 0x404040) + XLAT( "In most lands, more treasures are generated with each enemy native to this land you kill. " "Moreover, 100 kills is a requirement to enter the Graveyard and the Hive.\n\n" "Friendly creatures and parts of monsters (such as the Ivy) do appear in the list, " "but are not counted in the total kill count."); } } } string vers = VER; if(true) { if(modename.count(current_modecode)) vers += " '" + modename[current_modecode] + "' "; vers += mode_description(); if(peace::on) vers += " peace"; if(racing::on) vers += " racing"; if(princess::challenge) vers += " Princess"; if(yendor::on) vers += " Yendor"; if(tactic::on) vers += " PTM"; } if(!nofps) vers += XLAT(" fps: ") + its(calcfps()); #if CAP_MEMORY_RESERVE if(reserve_limit && reserve_count < reserve_limit) { vers += " " + its(reserve_count) + "/" + its(reserve_limit) + " MB"; if(displayButtonS(4, vid.yres - 4 - vid.fsize/2 - hud_margin(1), vers, 0xFF2020, 0, vid.fsize/2)) getcstat = PSEUDOKEY_MEMORY, instat = true; } else #endif if(displayButtonS(4, vid.yres - 4 - vid.fsize/2 - hud_margin(1), vers, 0x202020, 0, vid.fsize/2)) { mouseovers = XLAT("frames per second"), getcstat = SDLK_F1, instat = true, help = helptitle(XLAT("frames per second"), 0xFF4040) + XLAT( "The higher the number, the smoother the animations in the game. " "If you find that animations are not smooth enough, you can try " "to change the options " ) + #if ISIOS XLAT( "(in the MENU). You can reduce the sight range, this should make " "the animations smoother."); #else XLAT( "(press v) and change the wall/monster mode to ASCII, or change " "the resolution."); #endif } glflush(); achievement_display(); callhooks(hooks_stats); } }