1
0
mirror of https://github.com/zenorogue/hyperrogue.git synced 2025-01-12 10:20:32 +00:00

racing:: achievements/leaderboards

This commit is contained in:
Zeno Rogue 2019-01-11 02:19:28 +01:00
parent ada9a99a5d
commit e4939ad8d4
3 changed files with 70 additions and 3 deletions

View File

@ -71,6 +71,7 @@ const char* leadernames[NUMLEADER] = {
#define LB_STATISTICS 62 #define LB_STATISTICS 62
#define LB_HALLOWEEN 63 #define LB_HALLOWEEN 63
#define LB_RACING 81
void upload_score(int id, int v); void upload_score(int id, int v);
@ -87,7 +88,8 @@ bool wrongMode(char flags) {
if(geometry != gNormal) return true; if(geometry != gNormal) return true;
} }
if(shmup::on != (flags == rg::shmup)) return true; if(shmup::on != (flags == rg::shmup || flags == rg::racing)) return true;
if(racing::on != (flags == rg::racing)) return true;
#if CAP_DAILY #if CAP_DAILY
if(daily::on != (flags == rg::daily)) return true; if(daily::on != (flags == rg::daily)) return true;
#endif #endif
@ -521,10 +523,11 @@ void achievement_score(int cat, int number) {
else if(geometry) return; else if(geometry) return;
if(CHANGED_VARIATION) return; if(CHANGED_VARIATION) return;
if(randomPatternsMode) return; if(randomPatternsMode) return;
if(shmup::on && cat != LB_PURE_TACTICS_SHMUP && cat != LB_PURE_TACTICS_COOP) return; if(shmup::on && cat != LB_PURE_TACTICS_SHMUP && cat != LB_PURE_TACTICS_COOP && cat != LB_RACING) return;
if(yendor::on && cat != LB_YENDOR_CHALLENGE) return; if(yendor::on && cat != LB_YENDOR_CHALLENGE) return;
if(tactic::on && cat != LB_PURE_TACTICS && cat != LB_PURE_TACTICS_SHMUP && cat != LB_PURE_TACTICS_COOP) if(tactic::on && cat != LB_PURE_TACTICS && cat != LB_PURE_TACTICS_SHMUP && cat != LB_PURE_TACTICS_COOP)
return; return;
if(racing::on && cat != LB_RACING) return;
upload_score(cat, number); upload_score(cat, number);
#endif #endif
} }

View File

@ -162,6 +162,7 @@ void addMessage(string s, char spamtype = 0);
#define NUMLEADER 82 #define NUMLEADER 82
#define LB_PURE_TACTICS_SHMUP 49 #define LB_PURE_TACTICS_SHMUP 49
#define LB_PURE_TACTICS_COOP 50 #define LB_PURE_TACTICS_COOP 50
#define LB_RACING 81
#if ISMOBILE || ISWEB || ISPANDORA || 1 #if ISMOBILE || ISWEB || ISPANDORA || 1
typedef double ld; typedef double ld;
@ -4699,6 +4700,7 @@ namespace racing {
void apply_seed(); void apply_seed();
string racetimeformat(int t); string racetimeformat(int t);
void add_debug(cell *c); void add_debug(cell *c);
void displayScore(eLand l);
} }
bool subscreen_split(reaction_t for_each_subscreen); bool subscreen_split(reaction_t for_each_subscreen);

View File

@ -15,6 +15,8 @@ bool on;
bool player_relative = false; bool player_relative = false;
bool track_ready; bool track_ready;
bool official_race = false;
int TWIDTH; int TWIDTH;
ld race_advance = 0; ld race_advance = 0;
@ -75,7 +77,7 @@ struct ghost {
vector<ghostmoment> history; vector<ghostmoment> history;
}; };
using raceset = map<eLand, vector<ghost>>; typedef map<eLand, vector<ghost>> raceset;
map<pair<string, int>, raceset> race_ghosts; map<pair<string, int>, raceset> race_ghosts;
map<pair<string, int>, raceset> official_race_ghosts; map<pair<string, int>, raceset> official_race_ghosts;
@ -83,6 +85,14 @@ map<pair<string, int>, raceset> official_race_ghosts;
raceset& ghostset() { return race_ghosts[make_pair(track_code, modecode())]; } raceset& ghostset() { return race_ghosts[make_pair(track_code, modecode())]; }
raceset& oghostset() { return official_race_ghosts[make_pair(track_code, modecode())]; } raceset& oghostset() { return official_race_ghosts[make_pair(track_code, modecode())]; }
int get_score_in_land(eLand l) {
auto& gh = ghostset();
if(!gh.count(l)) return 0;
auto& v = gh[l];
if(!isize(v)) return 0;
return v[0].result;
}
array<vector<ghostmoment>, MAXPLAYER> current_history; array<vector<ghostmoment>, MAXPLAYER> current_history;
string ghost_prefix = "default"; string ghost_prefix = "default";
@ -172,6 +182,10 @@ bool read_ghosts(string seed, int mcode) {
for(auto gh: v) for(auto gh: v)
println(hlog, " ", racetimeformat(gh.result), " : ", format("%08X", gh.checksum), " = ", minf[gh.cs.uicolor].name); println(hlog, " ", racetimeformat(gh.result), " : ", format("%08X", gh.checksum), " = ", minf[gh.cs.uicolor].name);
} }
fhstream f("officials.data", "wb");
hwrite(f, (const int&) VERNUM_HEX);
hwrite(f, ghostset());
} }
string fname = ghost_filename(seed, mcode); string fname = ghost_filename(seed, mcode);
@ -659,6 +673,12 @@ void generate_track() {
race_start_tick = 0; race_start_tick = 0;
for(int i=0; i<MAXPLAYER; i++) race_finish_tick[i] = 0; for(int i=0; i<MAXPLAYER; i++) race_finish_tick[i] = 0;
official_race = (track_code == "OFFICIAL" && modecode() == 2);
if(official_race && race_checksum != oghostset() [specialland] [0].checksum) {
official_race = false;
addMessage(XLAT("Race did not generate correctly for some reason -- not ranked"));
}
} }
bool inrec = false; bool inrec = false;
@ -841,6 +861,8 @@ string racetimeformat(int t) {
void track_chooser(string new_track) { void track_chooser(string new_track) {
dialog::init(XLAT("Racing")); dialog::init(XLAT("Racing"));
map<char, eLand> landmap;
char let = 'a'; char let = 'a';
for(eLand l: race_lands) { for(eLand l: race_lands) {
auto& gh = race_ghosts[make_pair(new_track, modecode())] [l]; auto& gh = race_ghosts[make_pair(new_track, modecode())] [l];
@ -848,6 +870,7 @@ void track_chooser(string new_track) {
int best = LOST; int best = LOST;
for(auto& gc: gh) best = min(best, gc.result); for(auto& gc: gh) best = min(best, gc.result);
string s = (best == LOST) ? "" : racetimeformat(best); string s = (best == LOST) ? "" : racetimeformat(best);
landmap[let] = l;
dialog::addSelItem(XLAT1(linf[l].name), s, let++); dialog::addSelItem(XLAT1(linf[l].name), s, let++);
dialog::add_action([l, new_track] () { dialog::add_action([l, new_track] () {
stop_game(); stop_game();
@ -1051,6 +1074,40 @@ void prepare_subscreens() {
} }
} }
map<string, map<eLand, int> > scoreboard;
void uploadScore() {
int tscore = 0;
for(eLand l: race_lands) {
int i = get_score_in_land(l);
if(!i) continue;
int score = 60000000 / i; // 1000 points for minute, 2000 points for 30 sec
tscore += score;
}
achievement_score(LB_RACING, tscore);
}
void displayScore(eLand l) {
int vf = min((vid.yres-64) / 70, vid.xres/80);
int x = vid.xres / 4;
if(get_sync_status() == 1) {
displayfr(x, 56, 1, vf, "(syncing)", 0xC0C0C0, 0);
}
else {
vector<pair<int, string> > scores;
for(auto p: scoreboard) if(p.second.count(l)) scores.emplace_back(p.second[l], p.first);
sort(scores.begin(), scores.end());
int i = 0;
for(auto& sc: scores) {
int i0 = 56 + (i++) * vf;
displayfr(x, i0, 1, vf, racetimeformat(sc.first), 0xC0C0C0, 16);
displayfr(x+8, i0, 1, vf, sc.second, 0xC0C0C0, 0);
}
}
}
void race_won() { void race_won() {
if(!race_finish_tick[current_player]) { if(!race_finish_tick[current_player]) {
int result = ticks - race_start_tick; int result = ticks - race_start_tick;
@ -1064,6 +1121,9 @@ void race_won() {
if(place == 2) trophy[current_player] = 0xFFFFC0FF; if(place == 2) trophy[current_player] = 0xFFFFC0FF;
if(place == 3) trophy[current_player] = 0x967444FF; if(place == 3) trophy[current_player] = 0x967444FF;
if(place == 1 && losers && official_race)
achievement_gain("RACEWON", rg::racing);
race_finish_tick[current_player] = ticks; race_finish_tick[current_player] = ticks;
charstyle gcs = getcs(); charstyle gcs = getcs();
for(color_t *x: {&gcs.skincolor, &gcs.haircolor, &gcs.dresscolor, &gcs.swordcolor, &gcs.dresscolor2}) { for(color_t *x: {&gcs.skincolor, &gcs.haircolor, &gcs.dresscolor, &gcs.swordcolor, &gcs.dresscolor2}) {
@ -1080,6 +1140,8 @@ void race_won() {
subtrack.resize(ghosts_to_save); subtrack.resize(ghosts_to_save);
if(ghosts_to_save > 0) if(ghosts_to_save > 0)
write_ghosts(track_code, modecode()); write_ghosts(track_code, modecode());
if(official_race) uploadScore();
} }
} }