taisei/src/menu/musicroom.c
Andrei Alexeyev 8480d41b7b
Audio rewrite (#236)
This replaces SDL_mixer with an internal streaming and mixing system,
relying only on basic SDL audio support. It's also a partial refactor of
the audio API, most notably BGM-related. The BGM metadata resource type
is gone, as well as the `.bgm` files. The metadata is now stored inside
the `.opus` files directly, using standard Opus tags.
2020-06-22 17:41:03 +03:00

287 lines
7.5 KiB
C

/*
* This software is licensed under the terms of the MIT License.
* See COPYING for further information.
* ---
* Copyright (c) 2011-2019, Lukas Weber <laochailan@web.de>.
* Copyright (c) 2012-2019, Andrei Alexeyev <akari@taisei-project.org>.
*/
#include "taisei.h"
#include "musicroom.h"
#include "resource/resource.h"
#include "resource/font.h"
#include "audio/audio.h"
#include "progress.h"
#include "common.h"
#include "options.h"
#include "renderer/api.h"
#include "video.h"
enum {
MSTATE_UNLOCKED = 1,
MSTATE_CONFIRM = 2,
MSTATE_PLAYING = 4,
MSTATE_COMMENT_BG_VISIBLE = MSTATE_UNLOCKED | MSTATE_CONFIRM | MSTATE_PLAYING,
MSTATE_COMMENT_VISIBLE = MSTATE_UNLOCKED | MSTATE_PLAYING,
MSTATE_TITLE_VISIBLE = MSTATE_UNLOCKED | MSTATE_PLAYING,
};
typedef struct MusicEntryParam {
BGM *bgm;
ShaderProgram *text_shader;
uint8_t state;
} MusicEntryParam;
static void musicroom_logic(MenuData *m) {
float prev_selector_x = m->drawdata[0];
float prev_selector_w = m->drawdata[1];
animate_menu_list(m);
MenuEntry *cursor_entry = dynarray_get_ptr(&m->entries, m->cursor);
if(cursor_entry->arg) {
MusicEntryParam *p = cursor_entry->arg;
float selector_w = SCREEN_W - 200;
m->drawdata[0] = prev_selector_x;
m->drawdata[1] = prev_selector_w;
fapproach_asymptotic_p(&m->drawdata[0], 0.5 * selector_w, 0.1, 1e-5);
fapproach_asymptotic_p(&m->drawdata[1], 2 * selector_w, 0.1, 1e-5);
fapproach_asymptotic_p(&m->drawdata[3], (p->state & MSTATE_COMMENT_BG_VISIBLE) != 0, 0.2, 1e-5);
} else {
fapproach_asymptotic_p(&m->drawdata[3], 0, 0.2, 1e-5);
}
BGM *current_bgm = audio_bgm_current();
dynarray_foreach(&m->entries, int i, MenuEntry *e, {
MusicEntryParam *p = e->arg;
if(p) {
if(i != m->cursor && e->drawdata / 10.0 <= 0.02) {
p->state &= ~MSTATE_CONFIRM;
}
if(current_bgm && current_bgm == p->bgm) {
p->state |= MSTATE_PLAYING;
} else {
p->state &= ~MSTATE_PLAYING;
}
}
});
}
static void musicroom_draw_item(MenuEntry *e, int i, int cnt) {
if(!e->name) {
return;
}
MusicEntryParam *p = e->arg;
if(!p) {
text_draw(e->name, &(TextParams) {
.pos = { 20 - e->drawdata, 20 * i },
.shader = "text_default",
});
return;
}
char buf[16];
const char *title = p->state & MSTATE_TITLE_VISIBLE ? e->name : "???????";
Color clr = *r_color_current();
TextParams tparams = {
.pos = { 20 - e->drawdata, 20 * i },
.shader_ptr = p->text_shader,
.font_ptr = res_font("standard"),
.color = &clr,
};
bool kerning_saved = font_get_kerning_enabled(tparams.font_ptr);
tparams.pos.x += text_draw("", &tparams);
font_set_kerning_enabled(tparams.font_ptr, false);
snprintf(buf, sizeof(buf), " %i ", i + 1);
tparams.pos.x += text_width(tparams.font_ptr, " 99 ", 0);
tparams.align = ALIGN_RIGHT;
tparams.pos.x += text_draw(buf, &tparams);
tparams.align = ALIGN_LEFT;
font_set_kerning_enabled(tparams.font_ptr, kerning_saved);
if(!(p->state & MSTATE_TITLE_VISIBLE)) {
clr.r *= 0.5;
clr.g *= 0.5;
clr.b *= 0.5;
}
tparams.pos.x += text_draw(title, &tparams);
if(p->state & MSTATE_PLAYING) {
color_mul(&clr, RGBA(0.1, 0.6, 0.8, 0.8));
text_draw("Now playing", &(TextParams) {
.pos = { SCREEN_W - 200, 20 * i },
.shader_ptr = p->text_shader,
.align = ALIGN_RIGHT,
.color = &clr,
});
}
}
static void musicroom_draw(MenuData *m) {
r_state_push();
draw_options_menu_bg(m);
draw_menu_title(m, "Music Room");
draw_menu_list(m, 100, 100, musicroom_draw_item, SCREEN_H / GOLDEN_RATIO);
float comment_height = SCREEN_H * (1 - 1 / GOLDEN_RATIO);
float comment_alpha = (1 - menu_fade(m)) * m->drawdata[3];
float comment_offset = smoothstep(0, 1, (1 - comment_alpha)) * comment_height;
r_shader_standard_notex();
r_mat_mv_push();
r_mat_mv_translate(SCREEN_W * 0.5, SCREEN_H - comment_height * 0.5 + comment_offset, 0);
r_mat_mv_scale(SCREEN_W, comment_height, 1);
r_color4(0, 0, 0, 0.6 * comment_alpha);
r_draw_quad();
r_mat_mv_pop();
r_state_pop();
Font *const text_font = res_font("standard");
ShaderProgram *const text_shader = res_shader("text_default");
const float text_x = 50;
const float text_y = SCREEN_H - comment_height + font_get_lineskip(text_font) * 1.5 + comment_offset;
dynarray_foreach_elem(&m->entries, MenuEntry *e, {
float a = e->drawdata / 10.0 * comment_alpha;
if(a < 0.05 || !e->arg) {
continue;
}
const char *comment;
MusicEntryParam *p = e->arg;
Color *clr = RGBA(a, a, a, a);
if(p->state & MSTATE_CONFIRM) {
comment = (
"\nYou have not unlocked this track yet!\n\n"
"If you wish to hear it anyway, please select it again to confirm."
);
clr->g *= 0.3;
clr->b *= 0.2;
} else if(!(p->state & MSTATE_COMMENT_VISIBLE)) {
continue;
} else if(!(comment = bgm_get_comment(p->bgm))) {
comment = "\nNo comment available";
}
text_draw_wrapped(comment, SCREEN_W - text_x * 2, &(TextParams) {
.pos = { text_x + 0.5*(SCREEN_W - text_x * 2), text_y },
.font_ptr = text_font,
.shader_ptr = text_shader,
.color = clr,
.align = ALIGN_CENTER,
});
if(p->state & MSTATE_COMMENT_VISIBLE) {
const char *artist = bgm_get_artist(p->bgm);
if(artist) {
const char *prefix = "";
char buf[strlen(prefix) + strlen(artist) + 1];
strcpy(buf, prefix);
strcat(buf, artist);
text_draw(buf, &(TextParams) {
.pos = { SCREEN_W - text_x, SCREEN_H + comment_offset - font_get_lineskip(text_font) },
.font_ptr = text_font,
.shader_ptr = text_shader,
.color = RGBA(a, a, a, a),
.align = ALIGN_RIGHT,
});
}
}
});
}
static void action_play_bgm(MenuData *m, void *arg) {
MusicEntryParam *p = arg;
assume(p != NULL);
if(p->state & (MSTATE_CONFIRM | MSTATE_UNLOCKED)) {
p->state &= ~MSTATE_CONFIRM;
audio_bgm_play(p->bgm, true, 0, 0);
} else if (!(p->state & MSTATE_PLAYING)) {
p->state |= MSTATE_CONFIRM;
}
}
static void add_bgm(MenuData *m, const char *bgm_name, bool preload) {
if(preload) {
preload_resource(RES_BGM, bgm_name, RESF_OPTIONAL);
return;
}
BGM *bgm = res_bgm(bgm_name);
const char *title = bgm ? bgm_get_title(bgm) : NULL;
if(!title) {
title = "Unknown track";
}
MusicEntryParam *p = calloc(1, sizeof(*p));
p->bgm = bgm;
p->text_shader = res_shader("text_default");
if(progress_is_bgm_unlocked(bgm_name)) {
p->state |= MSTATE_UNLOCKED;
}
MenuEntry *e = add_menu_entry(m, title, action_play_bgm, p);
e->transition = NULL;
}
static void musicroom_free(MenuData *m) {
dynarray_foreach_elem(&m->entries, MenuEntry *e, {
free(e->arg);
});
}
MenuData* create_musicroom_menu(void) {
MenuData *m = alloc_menu();
m->logic = musicroom_logic;
m->draw = musicroom_draw;
m->end = musicroom_free;
m->transition = TransFadeBlack;
m->flags = MF_Abortable;
for(int preload = 1; preload >= 0; --preload) {
add_bgm(m, "menu", preload);
add_bgm(m, "stage1", preload);
add_bgm(m, "stage1boss", preload);
add_bgm(m, "stage2", preload);
add_bgm(m, "stage2boss", preload);
add_bgm(m, "stage3", preload);
add_bgm(m, "stage3boss", preload);
add_bgm(m, "scuttle", preload);
add_bgm(m, "stage4", preload);
add_bgm(m, "stage4boss", preload);
add_bgm(m, "stage5", preload);
add_bgm(m, "stage5boss", preload);
add_bgm(m, "bonus0", preload);
add_bgm(m, "stage6", preload);
add_bgm(m, "stage6boss_phase1", preload);
add_bgm(m, "stage6boss_phase2", preload);
add_bgm(m, "stage6boss_phase3", preload);
add_bgm(m, "ending", preload);
add_bgm(m, "credits", preload);
}
add_menu_separator(m);
add_menu_entry(m, "Back", menu_action_close, NULL);
while(!dynarray_get(&m->entries, m->cursor).action) {
++m->cursor;
}
return m;
}