mirror of
https://github.com/TerryCavanagh/VVVVVV.git
synced 2025-01-10 19:09:45 +01:00
Implement music and sound volume sliders
This adds music and volume sliders to the audio options. To use the sliders, you navigate to the given option, then press ACTION, and your selection will be transferred to the slider. Pressing left or right will move the slider accordingly. Then you can press ACTION to confirm the volume is what you want and deselect it, or you can press Esc to cancel the volume change, and it will revert to the previous volume; both actions will write your settings to disk. Most of this commit is just adding infrastructure to support having sliders in menus (without copy-pasting code), which is a totally completely new user interface that has never been used before in this game. If we're going to be adding something new, I want to make sure that it at least is done the RIGHT way. Closes #706.
This commit is contained in:
parent
27874e1dc6
commit
e8316c7e9a
6 changed files with 166 additions and 11 deletions
|
@ -383,6 +383,7 @@ void Game::init(void)
|
||||||
ingame_editormode = false;
|
ingame_editormode = false;
|
||||||
#endif
|
#endif
|
||||||
kludge_ingametemp = Menu::mainmenu;
|
kludge_ingametemp = Menu::mainmenu;
|
||||||
|
slidermode = SLIDER_NONE;
|
||||||
|
|
||||||
disablepause = false;
|
disablepause = false;
|
||||||
inputdelay = false;
|
inputdelay = false;
|
||||||
|
|
|
@ -84,6 +84,13 @@ namespace Menu
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
enum SLIDERMODE
|
||||||
|
{
|
||||||
|
SLIDER_NONE,
|
||||||
|
SLIDER_MUSICVOLUME,
|
||||||
|
SLIDER_SOUNDVOLUME
|
||||||
|
};
|
||||||
|
|
||||||
struct MenuStackFrame
|
struct MenuStackFrame
|
||||||
{
|
{
|
||||||
int option;
|
int option;
|
||||||
|
@ -259,6 +266,7 @@ public:
|
||||||
int currentmenuoption ;
|
int currentmenuoption ;
|
||||||
enum Menu::MenuName currentmenuname;
|
enum Menu::MenuName currentmenuname;
|
||||||
enum Menu::MenuName kludge_ingametemp;
|
enum Menu::MenuName kludge_ingametemp;
|
||||||
|
enum SLIDERMODE slidermode;
|
||||||
int current_credits_list_index;
|
int current_credits_list_index;
|
||||||
int menuxoff, menuyoff;
|
int menuxoff, menuyoff;
|
||||||
int menuspacing;
|
int menuspacing;
|
||||||
|
|
|
@ -1475,7 +1475,7 @@ void Graphics::drawmenu( int cr, int cg, int cb, bool levelmenu /*= false*/ )
|
||||||
SDL_strlcpy(tempstring, opt.text, sizeof(tempstring));
|
SDL_strlcpy(tempstring, opt.text, sizeof(tempstring));
|
||||||
|
|
||||||
char buffer[MENU_TEXT_BYTES];
|
char buffer[MENU_TEXT_BYTES];
|
||||||
if ((int) i == game.currentmenuoption)
|
if ((int) i == game.currentmenuoption && game.slidermode == SLIDER_NONE)
|
||||||
{
|
{
|
||||||
if (opt.active)
|
if (opt.active)
|
||||||
{
|
{
|
||||||
|
|
|
@ -200,6 +200,56 @@ static void startmode(const int mode)
|
||||||
fadetomodedelay = 16;
|
fadetomodedelay = 16;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int* user_changing_volume = NULL;
|
||||||
|
static int previous_volume = 0;
|
||||||
|
|
||||||
|
static void initvolumeslider(const int menuoption)
|
||||||
|
{
|
||||||
|
switch (menuoption)
|
||||||
|
{
|
||||||
|
case 0:
|
||||||
|
game.slidermode = SLIDER_MUSICVOLUME;
|
||||||
|
user_changing_volume = &music.user_music_volume;
|
||||||
|
break;
|
||||||
|
case 1:
|
||||||
|
game.slidermode = SLIDER_SOUNDVOLUME;
|
||||||
|
user_changing_volume = &music.user_sound_volume;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
SDL_assert(0 && "Unhandled volume slider option!");
|
||||||
|
game.slidermode = SLIDER_NONE;
|
||||||
|
user_changing_volume = NULL;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
previous_volume = *user_changing_volume;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void deinitvolumeslider(void)
|
||||||
|
{
|
||||||
|
user_changing_volume = NULL;
|
||||||
|
game.savestatsandsettings_menu();
|
||||||
|
game.slidermode = SLIDER_NONE;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void slidermodeinput(void)
|
||||||
|
{
|
||||||
|
if (user_changing_volume == NULL)
|
||||||
|
{
|
||||||
|
SDL_assert(0 && "user_changing_volume is NULL!");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (game.press_left)
|
||||||
|
{
|
||||||
|
*user_changing_volume -= USER_VOLUME_STEP;
|
||||||
|
}
|
||||||
|
else if (game.press_right)
|
||||||
|
{
|
||||||
|
*user_changing_volume += USER_VOLUME_STEP;
|
||||||
|
}
|
||||||
|
*user_changing_volume = clamp(*user_changing_volume, 0, USER_VOLUME_MAX);
|
||||||
|
}
|
||||||
|
|
||||||
static void menuactionpress(void)
|
static void menuactionpress(void)
|
||||||
{
|
{
|
||||||
switch (game.currentmenuname)
|
switch (game.currentmenuname)
|
||||||
|
@ -786,10 +836,16 @@ static void menuactionpress(void)
|
||||||
switch (game.currentmenuoption)
|
switch (game.currentmenuoption)
|
||||||
{
|
{
|
||||||
case 0:
|
case 0:
|
||||||
/* Not implemented */
|
|
||||||
break;
|
|
||||||
case 1:
|
case 1:
|
||||||
/* Not implemented */
|
music.playef(11);
|
||||||
|
if (game.slidermode == SLIDER_NONE)
|
||||||
|
{
|
||||||
|
initvolumeslider(game.currentmenuoption);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
deinitvolumeslider();
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
case 2:
|
case 2:
|
||||||
if (!music.mmmmmm)
|
if (!music.mmmmmm)
|
||||||
|
@ -1641,7 +1697,27 @@ void titleinput(void)
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
if (game.ingame_titlemode
|
if (game.slidermode != SLIDER_NONE)
|
||||||
|
{
|
||||||
|
switch (game.slidermode)
|
||||||
|
{
|
||||||
|
/* Cancel volume change. */
|
||||||
|
case SLIDER_MUSICVOLUME:
|
||||||
|
case SLIDER_SOUNDVOLUME:
|
||||||
|
if (user_changing_volume == NULL)
|
||||||
|
{
|
||||||
|
SDL_assert(0 && "user_changing_volume is NULL!");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
*user_changing_volume = previous_volume;
|
||||||
|
deinitvolumeslider();
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
SDL_assert(0 && "Unhandled slider mode!");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (game.ingame_titlemode
|
||||||
&& game.currentmenuname == Menu::options)
|
&& game.currentmenuname == Menu::options)
|
||||||
{
|
{
|
||||||
game.returntoingame();
|
game.returntoingame();
|
||||||
|
@ -1655,6 +1731,8 @@ void titleinput(void)
|
||||||
}
|
}
|
||||||
|
|
||||||
if(game.menustart)
|
if(game.menustart)
|
||||||
|
{
|
||||||
|
if (game.slidermode == SLIDER_NONE)
|
||||||
{
|
{
|
||||||
if (game.press_left)
|
if (game.press_left)
|
||||||
{
|
{
|
||||||
|
@ -1665,6 +1743,11 @@ void titleinput(void)
|
||||||
game.currentmenuoption++;
|
game.currentmenuoption++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
slidermodeinput();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (game.currentmenuoption < 0) game.currentmenuoption = game.menuoptions.size()-1;
|
if (game.currentmenuoption < 0) game.currentmenuoption = game.menuoptions.size()-1;
|
||||||
if (game.currentmenuoption >= (int) game.menuoptions.size() ) game.currentmenuoption = 0;
|
if (game.currentmenuoption >= (int) game.menuoptions.size() ) game.currentmenuoption = 0;
|
||||||
|
|
|
@ -11,6 +11,9 @@
|
||||||
/* The amount of "space" for the scale of the user-set volume. */
|
/* The amount of "space" for the scale of the user-set volume. */
|
||||||
#define USER_VOLUME_MAX 256
|
#define USER_VOLUME_MAX 256
|
||||||
|
|
||||||
|
/* It is advised that USER_VOLUME_MAX be divisible by this. */
|
||||||
|
#define USER_VOLUME_STEP 32
|
||||||
|
|
||||||
class musicclass
|
class musicclass
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
|
|
|
@ -46,6 +46,62 @@ static inline void drawslowdowntext(void)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void volumesliderrender(void)
|
||||||
|
{
|
||||||
|
char buffer[40 + 1];
|
||||||
|
|
||||||
|
char slider[20 + 1];
|
||||||
|
int slider_length;
|
||||||
|
|
||||||
|
const char symbol[] = "[]";
|
||||||
|
int symbol_length;
|
||||||
|
|
||||||
|
int offset;
|
||||||
|
int num_positions;
|
||||||
|
|
||||||
|
const int* volume_ptr;
|
||||||
|
|
||||||
|
switch (game.currentmenuoption)
|
||||||
|
{
|
||||||
|
case 0:
|
||||||
|
volume_ptr = &music.user_music_volume;
|
||||||
|
break;
|
||||||
|
case 1:
|
||||||
|
volume_ptr = &music.user_sound_volume;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
SDL_assert(0 && "Unhandled volume slider menu option!");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
VVV_fillstring(slider, sizeof(slider), '.');
|
||||||
|
slider_length = sizeof(slider) - 1;
|
||||||
|
|
||||||
|
symbol_length = sizeof(symbol) - 1;
|
||||||
|
|
||||||
|
num_positions = slider_length - symbol_length + 1;
|
||||||
|
|
||||||
|
offset = num_positions * (*volume_ptr) / USER_VOLUME_MAX;
|
||||||
|
offset = clamp(offset, 0, slider_length - symbol_length);
|
||||||
|
|
||||||
|
/* SDL_strlcpy null-terminates, which would end the string in the middle of
|
||||||
|
* it, which we don't want!
|
||||||
|
*/
|
||||||
|
SDL_memcpy(&slider[offset], symbol, symbol_length);
|
||||||
|
|
||||||
|
if (game.slidermode == SLIDER_NONE)
|
||||||
|
{
|
||||||
|
SDL_strlcpy(buffer, slider, sizeof(buffer));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* Draw selection brackets. */
|
||||||
|
SDL_snprintf(buffer, sizeof(buffer), "[ %s ]", slider);
|
||||||
|
}
|
||||||
|
|
||||||
|
graphics.Print(-1, 85, buffer, tr, tg, tb, true);
|
||||||
|
}
|
||||||
|
|
||||||
static void menurender(void)
|
static void menurender(void)
|
||||||
{
|
{
|
||||||
int temp = 50;
|
int temp = 50;
|
||||||
|
@ -281,10 +337,14 @@ static void menurender(void)
|
||||||
switch (game.currentmenuoption)
|
switch (game.currentmenuoption)
|
||||||
{
|
{
|
||||||
case 0:
|
case 0:
|
||||||
/* Not implemented */
|
graphics.bigprint(-1, 30, "Music Volume", tr, tg, tb, true);
|
||||||
|
graphics.Print(-1, 65, "Change the volume of the music.", tr, tg, tb, true);
|
||||||
|
volumesliderrender();
|
||||||
break;
|
break;
|
||||||
case 1:
|
case 1:
|
||||||
/* Not implemented */
|
graphics.bigprint(-1, 30, "Sound Volume", tr, tg, tb, true);
|
||||||
|
graphics.Print(-1, 65, "Change the volume of sound effects.", tr, tg, tb, true);
|
||||||
|
volumesliderrender();
|
||||||
break;
|
break;
|
||||||
case 2:
|
case 2:
|
||||||
if (!music.mmmmmm)
|
if (!music.mmmmmm)
|
||||||
|
|
Loading…
Reference in a new issue