taisei/src/menu/replayview.c
Andrei Alexeyev fda8556a39
src: deprecate strdup(), use mem_strdup() explicitly
This fixes a nasty bug that manifests on windows when building without
precompiled headers. util/stringops.h used to silently replace strdup
with a macro that's compatible with mem_free(). This header would
typically be included everywhere due to PCH, but without it the strdup
from libc would sometimes be in scope. On most platforms mem_free() is
equivalent to free(), but not on windows, because we have to use
_aligned_free() there. Attempting to mem_free() the result of a libc
strdup() would segfault in such a configuration.

Avoid the footgun by banning strdup() entirely. Maybe redefining libc
names isn't such a great idea, who knew?
2024-09-10 14:31:55 +02:00

474 lines
12 KiB
C

/*
* This software is licensed under the terms of the MIT License.
* See COPYING for further information.
* ---
* Copyright (c) 2011-2024, Lukas Weber <laochailan@web.de>.
* Copyright (c) 2012-2024, Andrei Alexeyev <akari@taisei-project.org>.
*/
#include "replayview.h"
#include "common.h"
#include "menu.h"
#include "options.h"
#include "audio/audio.h"
#include "plrmodes.h"
#include "replay/struct.h"
#include "resource/font.h"
#include "stageinfo.h"
#include "video.h"
// Type of MenuData.context
typedef struct ReplayviewContext {
MenuData *submenu;
MenuData *next_submenu;
double sub_fade;
} ReplayviewContext;
// Type of MenuEntry.arg (which should be renamed to context, probably...)
typedef struct ReplayviewItemContext {
Replay *replay;
char *replayname;
} ReplayviewItemContext;
static MenuData *replayview_sub_messagebox(MenuData *parent, const char *message);
static void replayview_set_submenu(MenuData *parent, MenuData *submenu) {
ReplayviewContext *ctx = parent->context;
if(ctx->submenu) {
ctx->submenu->state = MS_Dead;
ctx->next_submenu = submenu;
} else {
ctx->submenu = submenu;
}
if(submenu != NULL) {
// submenu->context = ctx;
assert(submenu->context == ctx);
}
}
typedef struct {
Replay *rpy;
int stgnum;
} startrpy_arg_t;
static void on_replay_finished(CallChainResult ccr) {
replay_destroy_events(ccr.ctx);
audio_bgm_play(res_bgm("menu"), true, 0, 0);
}
static void really_start_replay(CallChainResult ccr) {
startrpy_arg_t *argp = ccr.ctx;
auto arg = *argp;
mem_free(argp);
if(!TRANSITION_RESULT_CANCELED(ccr)) {
replay_play(arg.rpy, arg.stgnum, false, CALLCHAIN(on_replay_finished, arg.rpy));
}
}
static void start_replay(MenuData *menu, void *arg) {
ReplayviewItemContext *ictx = arg;
ReplayviewContext *mctx = menu->context;
int stagenum = 0;
if(mctx->submenu) {
stagenum = mctx->submenu->cursor;
}
Replay *rpy = ictx->replay;
if(!replay_load(rpy, ictx->replayname, REPLAY_READ_EVENTS)) {
replayview_set_submenu(menu, replayview_sub_messagebox(menu, "Failed to load replay events"));
return;
}
ReplayStage *stg = dynarray_get_ptr(&rpy->stages, stagenum);
char buf[64];
if(!stageinfo_get_by_id(stg->stage)) {
replay_destroy_events(rpy);
snprintf(buf, sizeof(buf), "Can't replay this stage: unknown stage ID %X", stg->stage);
replayview_set_submenu(menu, replayview_sub_messagebox(menu, buf));
return;
}
if(!plrmode_find(stg->plr_char, stg->plr_shot)) {
replay_destroy_events(rpy);
snprintf(buf, sizeof(buf), "Can't replay this stage: unknown player character/mode %X/%X", stg->plr_char, stg->plr_shot);
replayview_set_submenu(menu, replayview_sub_messagebox(menu, buf));
return;
}
set_transition(TransFadeBlack, FADE_TIME, FADE_TIME,
CALLCHAIN(really_start_replay, ALLOC(startrpy_arg_t, {
.rpy = ictx->replay,
.stgnum = stagenum
}))
);
}
static void replayview_draw_stagemenu(MenuData*);
static void replayview_draw_messagebox(MenuData*);
static MenuData *replayview_sub_stageselect(MenuData *parent, ReplayviewItemContext *ictx) {
MenuData *m = alloc_menu();
Replay *rpy = ictx->replay;
m->draw = replayview_draw_stagemenu;
m->flags = MF_Transient | MF_Abortable;
m->transition = NULL;
m->context = parent->context;
dynarray_foreach_elem(&rpy->stages, ReplayStage *rstg, {
uint16_t stage_id = rstg->stage;
StageInfo *stg = stageinfo_get_by_id(stage_id);
if(LIKELY(stg != NULL)) {
add_menu_entry(m, stg->title, start_replay, ictx);
} else {
char label[64];
snprintf(label, sizeof(label), "Unknown stage %X", stage_id);
add_menu_entry(m, label, menu_action_close, NULL);
}
});
return m;
}
static MenuData *replayview_sub_messagebox(MenuData *parent, const char *message) {
MenuData *m = alloc_menu();
m->draw = replayview_draw_messagebox;
m->flags = MF_Transient | MF_Abortable;
m->transition = NULL;
m->context = parent->context;
add_menu_entry(m, message, menu_action_close, NULL);
return m;
}
static void replayview_run(MenuData *menu, void *arg) {
ReplayviewItemContext *ctx = arg;
Replay *rpy = ctx->replay;
if(rpy->stages.num_elements > 1) {
replayview_set_submenu(menu, replayview_sub_stageselect(menu, ctx));
} else {
start_replay(menu, ctx);
}
}
static void replayview_freearg(void *a) {
ReplayviewItemContext *ctx = a;
replay_reset(ctx->replay);
mem_free(ctx->replay);
mem_free(ctx->replayname);
mem_free(ctx);
}
static void replayview_draw_submenu_bg(float width, float height, float alpha) {
r_state_push();
r_mat_mv_push();
r_mat_mv_translate(SCREEN_W*0.5, SCREEN_H*0.5, 0);
r_mat_mv_scale(width, height, 1);
alpha *= 0.7;
r_color4(0.1 * alpha, 0.1 * alpha, 0.1 * alpha, alpha);
r_shader_standard_notex();
r_draw_quad();
r_mat_mv_pop();
r_state_pop();
}
static void replayview_draw_messagebox(MenuData* m) {
// this context is shared with the parent menu
ReplayviewContext *ctx = m->context;
MenuEntry *e = dynarray_get_ptr(&m->entries, 0);
float alpha = 1 - ctx->sub_fade;
float height = font_get_lineskip(res_font("standard")) * 2;
float width = text_width(res_font("standard"), e->name, 0) + 64;
replayview_draw_submenu_bg(width, height, alpha);
text_draw(e->name, &(TextParams) {
.align = ALIGN_CENTER,
.color = RGBA_MUL_ALPHA(0.9, 0.6, 0.2, alpha),
.pos = { SCREEN_W*0.5, SCREEN_H*0.5 },
.shader = "text_default",
});
}
static void replayview_draw_stagemenu(MenuData *m) {
// this context is shared with the parent menu
ReplayviewContext *ctx = m->context;
float alpha = 1 - ctx->sub_fade;
float height = (1 + m->entries.num_elements) * 20;
float width = 180;
replayview_draw_submenu_bg(width, height, alpha);
r_mat_mv_push();
r_mat_mv_translate(SCREEN_W*0.5, (SCREEN_H - (m->entries.num_elements - 1) * 20) * 0.5, 0);
dynarray_foreach(&m->entries, int i, MenuEntry *e, {
float a = e->drawdata;
Color clr;
if(e->action == NULL) {
clr = *RGBA_MUL_ALPHA(0.5, 0.5, 0.5, 0.5 * alpha);
} else {
float ia = 1-a;
clr = *RGBA_MUL_ALPHA(0.9 + ia * 0.1, 0.6 + ia * 0.4, 0.2 + ia * 0.8, (0.7 + 0.3 * a) * alpha);
}
text_draw(e->name, &(TextParams) {
.align = ALIGN_CENTER,
.pos = { 0, 20*i },
.color = &clr,
.shader = "text_default",
});
});
r_mat_mv_pop();
}
static void replayview_drawitem(MenuEntry *e, int item, int cnt, void *ctx) {
ReplayviewItemContext *ictx = e->arg;
if(!ictx) {
return;
}
Replay *rpy = ictx->replay;
float sizes[] = { 1.2, 2.2, 0.5, 0.55, 0.55 };
int columns = sizeof(sizes)/sizeof(float), i, j;
float base_size = (SCREEN_W - 110.0) / columns;
ReplayStage *first_stage = dynarray_get_ptr(&rpy->stages, 0);
time_t t = first_stage->start_time;
struct tm *timeinfo = localtime(&t);
for(i = 0; i < columns; ++i) {
char tmp[128];
float csize = sizes[i] * base_size;
float o = 0;
for(j = 0; j < i; ++j)
o += sizes[j] * base_size;
Alignment a = ALIGN_CENTER;
switch(i) {
case 0:
a = ALIGN_LEFT;
strftime(tmp, sizeof(tmp), "%Y-%m-%d %H:%M", timeinfo);
break;
case 1:
a = ALIGN_LEFT;
strlcpy(tmp, rpy->playername, sizeof(tmp));
break;
case 2: {
a = ALIGN_RIGHT;
PlayerMode *plrmode = plrmode_find(first_stage->plr_char, first_stage->plr_shot);
if(plrmode == NULL) {
strlcpy(tmp, "?????", sizeof(tmp));
} else {
plrmode_repr(tmp, sizeof(tmp), plrmode, false);
}
break;
}
case 3:
a = ALIGN_CENTER;
snprintf(tmp, sizeof(tmp), "%s", difficulty_name(first_stage->diff));
break;
case 4:
a = ALIGN_LEFT;
if(rpy->stages.num_elements == 1) {
StageInfo *stg = stageinfo_get_by_id(first_stage->stage);
if(stg) {
snprintf(tmp, sizeof(tmp), "%s", stg->title);
} else {
snprintf(tmp, sizeof(tmp), "?????");
}
} else {
snprintf(tmp, sizeof(tmp), "%i stages", rpy->stages.num_elements);
}
break;
}
switch(a) {
case ALIGN_CENTER: o += csize * 0.5 - text_width(res_font("standard"), tmp, 0) * 0.5; break;
case ALIGN_RIGHT: o += csize - text_width(res_font("standard"), tmp, 0); break;
default: break;
}
text_draw(tmp, &(TextParams) {
.pos = { o + 10, 20 * item },
.shader = "text_default",
.max_width = csize,
});
}
}
static void replayview_logic(MenuData *m) {
ReplayviewContext *ctx = m->context;
if(ctx->submenu) {
MenuData *sm = ctx->submenu;
dynarray_foreach(&sm->entries, int i, MenuEntry *e, {
e->drawdata += 0.2 * ((i == sm->cursor) - e->drawdata);
});
if(sm->state == MS_Dead) {
if(ctx->sub_fade == 1.0) {
free_menu(sm);
ctx->submenu = ctx->next_submenu;
ctx->next_submenu = NULL;
return;
}
ctx->sub_fade = approach(ctx->sub_fade, 1.0, 1.0/FADE_TIME);
} else {
ctx->sub_fade = approach(ctx->sub_fade, 0.0, 1.0/FADE_TIME);
}
}
m->drawdata[0] = SCREEN_W/2 - 50;
m->drawdata[1] = (SCREEN_W - 100)*1.75;
m->drawdata[2] += (20*m->cursor - m->drawdata[2])/10.0;
animate_menu_list_entries(m);
}
static void replayview_draw(MenuData *m) {
ReplayviewContext *ctx = m->context;
draw_options_menu_bg(m);
draw_menu_title(m, "Replays");
draw_menu_list(m, 50, 100, replayview_drawitem, SCREEN_H, NULL);
if(ctx->submenu) {
ctx->submenu->draw(ctx->submenu);
}
r_shader_standard();
}
static int replayview_cmp(const void *a, const void *b) {
ReplayviewItemContext *actx = ((MenuEntry*)a)->arg;
ReplayviewItemContext *bctx = ((MenuEntry*)b)->arg;
Replay *arpy = actx->replay;
Replay *brpy = bctx->replay;
return dynarray_get(&brpy->stages, 0).start_time - dynarray_get(&arpy->stages, 0).start_time;
}
static int fill_replayview_menu(MenuData *m) {
VFSDir *dir = vfs_dir_open("storage/replays");
const char *filename;
int rpys = 0;
if(!dir) {
log_warn("VFS error: %s", vfs_get_error());
return -1;
}
char ext[5];
snprintf(ext, 5, ".%s", REPLAY_EXTENSION);
while((filename = vfs_dir_read(dir))) {
if(!strendswith(filename, ext))
continue;
auto rpy = ALLOC(Replay);
if(!replay_load(rpy, filename, REPLAY_READ_META)) {
mem_free(rpy);
continue;
}
auto ictx = ALLOC(ReplayviewItemContext, {
.replay = rpy,
.replayname = mem_strdup(filename),
});
add_menu_entry(m, " ", replayview_run, ictx)->transition = /*rpy->numstages < 2 ? TransFadeBlack :*/ NULL;
++rpys;
}
vfs_dir_close(dir);
dynarray_qsort(&m->entries, replayview_cmp);
return rpys;
}
static void replayview_menu_input(MenuData *m) {
ReplayviewContext *ctx = (ReplayviewContext*)m->context;
MenuData *sub = ctx->submenu;
if(transition.state == TRANS_FADE_IN) {
menu_no_input(m);
} else {
if(sub && sub->state != MS_Dead) {
sub->input(sub);
} else {
menu_input(m);
}
}
}
static void replayview_free(MenuData *m) {
if(m->context) {
ReplayviewContext *ctx = m->context;
free_menu(ctx->next_submenu);
free_menu(ctx->submenu);
mem_free(m->context);
m->context = NULL;
}
dynarray_foreach_elem(&m->entries, MenuEntry *e, {
if(e->action == replayview_run) {
replayview_freearg(e->arg);
}
});
}
MenuData *create_replayview_menu(void) {
MenuData *m = alloc_menu();
m->logic = replayview_logic;
m->input = replayview_menu_input;
m->draw = replayview_draw;
m->end = replayview_free;
m->transition = TransFadeBlack;
m->context = ALLOC(ReplayviewContext, {
.sub_fade = 1.0,
});
m->flags = MF_Abortable;
int r = fill_replayview_menu(m);
if(!r) {
add_menu_entry(m, "No replays available. Play the game and record some!", menu_action_close, NULL);
} else if(r < 0) {
add_menu_entry(m, "There was a problem getting the replay list :(", menu_action_close, NULL);
} else {
add_menu_separator(m);
add_menu_entry(m, "Back", menu_action_close, NULL);
}
return m;
}