Add setting and info screen

- setting screen is only implemented, without features
- info screen is completed currently
This commit is contained in:
Hanse-14 2023-12-17 02:42:10 +01:00
parent c10481021f
commit a7714ec15f
4 changed files with 286 additions and 60 deletions

View File

@ -7,7 +7,9 @@
#define MAX_COLOURS 10 #define MAX_COLOURS 10
#define MAX_LINES_TFF 20 //max lines (of ttf) for the whole project #define MAX_LINES_TFF 20 //max lines (of ttf) for the whole project
#define MAX_LINES_STARTSCREEN 5 #define MAX_LINES_STARTSCREEN 6
#define MAX_LINES_SETTINGS 7
#define MAX_LINE_INFOSCREEN 17
#define TIME_BETWEEN_UPDATE_MENU 1000 // [ms] #define TIME_BETWEEN_UPDATE_MENU 1000 // [ms]
// Enum that defines the active menu // Enum that defines the active menu
@ -16,26 +18,29 @@ typedef enum menus_t
NONE = 0, NONE = 0,
START, START,
SETTINGS, SETTINGS,
INFOSCREEN,
LEADERBOARD, LEADERBOARD,
PAUSE PAUSE
} menus_t; } menus_t;
// Struct that include pointer for ttl-functions in menu.c and render.c // Struct that include pointer for ttl-functions in menu.c and render.c
typedef struct tllData_t typedef struct tllData_t
{ {
TTF_Font *ptrFont_20;
TTF_Font *ptrFont_30; // used by menu.c and render.c TTF_Font *ptrFont_30; // used by menu.c and render.c
TTF_Font *ptrFont_200; // used by menu.c and render.c TTF_Font *ptrFont_200; // used by menu.c and render.c
SDL_Surface *textSurface; // used by menu.c and render.c SDL_Surface *textSurface; // used by menu.c and render.c
SDL_Texture *textTexture; // used by menu.c and render.c SDL_Texture *textTexture; // used by menu.c and render.c
SDL_Texture *textTextures[MAX_LINES_TFF]; SDL_Texture *textTextures[MAX_LINES_TFF];
SDL_Color textColour[MAX_COLOURS]; // colour in which the text is printed SDL_Color textColour[MAX_COLOURS]; // colour in which the text is printed
const int fontSize_20;
const int fontSize_30; const int fontSize_30;
const int fontSize_200; const int fontSize_200;
int64_t lastTimeStep; int64_t lastTimeStep;
int lineHeight; // to print text in middle int lineHeight; // to print text in middle
int totalHeight; // to print text in middle int totalHeight; // to print text in middle
int textPrintPosition; // where first line is printed int textPrintPosition; // where first line is printed
time_t cycleDuration; const time_t cycleDuration;
bool showEnter; bool showEnter;
} ttlData_t; } ttlData_t;
@ -61,6 +66,12 @@ void showLeaderboard();
void showSettings(); //optional void showSettings(); //optional
//startet Settungs-Menü //startet Settungs-Menü
// show info screen
void showInfoScreen();
void menuHandleInput(SDL_Event event); //als Übergabeparameter: int(?) event -> welcher Datentyp hängt von SDL ab void menuHandleInput(SDL_Event event); //als Übergabeparameter: int(?) event -> welcher Datentyp hängt von SDL ab
//switch case für welcher Modus //switch case für welcher Modus
//switch case für welche Taste gedrückt wurde //switch case für welche Taste gedrückt wurde

View File

@ -12,4 +12,8 @@ int CreateSDLWindow();
void DestroySDLWindow(); void DestroySDLWindow();
void renderStartMenu(); void renderStartMenu();
void renderSettings();
void renderInfoScreen();

View File

@ -10,19 +10,19 @@
//define global struct for tllStorage values //define global struct for tllStorage values
//default values defined here: (note: gets modified by render.c or menu.c) //default values defined here: (note: gets modified by render.c or menu.c)
ttlData_t ttlStorage = ttlData_t ttlStorage =
{ {
.fontSize_30 = 30, .fontSize_20 = 20, // size of font
.fontSize_200 = 200, .fontSize_30 = 30, // size of font
.lastTimeStep = 0, .fontSize_200 = 200, // size of font
.cycleDuration = 500, .cycleDuration = 500, // time between show and not showing ENTER in start screen
.showEnter = false .showEnter = false
}; };
// Default // Default
menus_t activeMenu = START; menus_t activeMenu = START;
// is called by main function // is called by main function if game.gameState == MENU
// choose between the selected menu-rendering functions // choose between the selected menu functions
void manageMenu() void manageMenu()
{ {
switch(activeMenu) switch(activeMenu)
@ -30,29 +30,45 @@ void manageMenu()
case START: case START:
showStartScreen(); showStartScreen();
break; break;
case SETTINGS:
showSettings();
break;
case INFOSCREEN:
showInfoScreen();
break;
case LEADERBOARD:
showLeaderboard();
break;
case PAUSE:
showPauseScreen();
break;
} }
} }
// shows start screen with blinking ENTER
void showStartScreen() void showStartScreen()
{ {
LOGI("menu: showing start-screen\n"); LOGI("menu: showing start-screen\n");
time_t now = GET_TIME_MS(); time_t now = GET_TIME_MS();
// is used to make ENTER blink
if(now > (ttlStorage.lastTimeStep + ttlStorage.cycleDuration)) if(now > (ttlStorage.lastTimeStep + ttlStorage.cycleDuration))
{ {
ttlStorage.showEnter = !ttlStorage.showEnter; // is used to make ENTER blink ttlStorage.showEnter = !ttlStorage.showEnter;
renderStartMenu(); renderStartMenu();
} }
//------------------------------------------------------------------------------------
return; return;
} }
void showLeaderboard()
{
void showLeaderboard(){
LOGI("menu: showing leaderboard\n"); LOGI("menu: showing leaderboard\n");
//--- play crash sound --- //--- play crash sound ---
@ -66,17 +82,50 @@ void showLeaderboard(){
return; return;
} }
void showPauseScreen(){ void showPauseScreen()
{
LOGI("menu: showing leaderboard\n"); LOGI("menu: showing leaderboard\n");
game.gameState = PAUSED; game.gameState = PAUSED;
return; return;
} }
void showSettings(){
void showSettings()
{
LOGI("menu: showing settings\n"); LOGI("menu: showing settings\n");
time_t now = GET_TIME_MS();
// is used to make ENTER blink
if(now > (ttlStorage.lastTimeStep + ttlStorage.cycleDuration))
{
ttlStorage.showEnter = !ttlStorage.showEnter;
renderSettings();
}
return; return;
} }
void showInfoScreen()
{
LOGI("menu: showing info-screen\n");
time_t now = GET_TIME_MS();
// is used to make ENTER blink
if(now > (ttlStorage.lastTimeStep + ttlStorage.cycleDuration))
{
ttlStorage.showEnter = !ttlStorage.showEnter;
renderInfoScreen();
}
return;
}
// handle input over keyboard
// delete text at the and of one menu section
void menuHandleInput(SDL_Event event){ void menuHandleInput(SDL_Event event){
//compare 'handleInput_runningState(SDL_Event event)' in input.c //compare 'handleInput_runningState(SDL_Event event)' in input.c
switch(activeMenu) switch(activeMenu)
@ -89,10 +138,10 @@ void menuHandleInput(SDL_Event event){
break; break;
case SDLK_RETURN: // Enter key case SDLK_RETURN: // Enter key
game.gameState = RUNNING; activeMenu = SETTINGS;
activeMenu = NONE; ttlStorage.lastTimeStep = 0;
// delete text // delete text
for (int i = 0; i < 5; ++i) for (int i = 0; i < MAX_LINES_TFF; ++i)
{ {
SDL_DestroyTexture(ttlStorage.textTextures[i]); SDL_DestroyTexture(ttlStorage.textTextures[i]);
} }
@ -101,9 +150,55 @@ void menuHandleInput(SDL_Event event){
case SETTINGS: case SETTINGS:
switch(event.key.keysym.sym)
{
case SDLK_q: // q: quit
game.gameState = EXIT;
break;
case SDLK_F1: // go to info screen
activeMenu = INFOSCREEN;
ttlStorage.lastTimeStep = 0;
// delete text
for (int i = 0; i < MAX_LINES_TFF; ++i)
{
SDL_DestroyTexture(ttlStorage.textTextures[i]);
}
break;
case SDLK_RETURN: //start game
activeMenu = NONE;
game.gameState = RUNNING;
// delete text
for (int i = 0; i < MAX_LINES_TFF; ++i)
{
SDL_DestroyTexture(ttlStorage.textTextures[i]);
}
break;
}
break;
case INFOSCREEN:
switch(event.key.keysym.sym)
{
case SDLK_q: // q: quit
game.gameState = EXIT;
break;
case SDLK_RETURN: // go return to settings
activeMenu = SETTINGS;
ttlStorage.lastTimeStep = 0;
// delete text
for (int i = 0; i < MAX_LINES_TFF; ++i)
{
SDL_DestroyTexture(ttlStorage.textTextures[i]);
}
}
break; break;
} }
return; return;
} }

View File

@ -120,15 +120,9 @@ void renderGame(){
//-------------------------------------------------------------- //--------------------------------------------------------------
void renderStartMenu() void renderStartMenu()
{ {
//=========== only first loop ================
//--------------------
// only first loop
//--------------------
if(ttlStorage.lastTimeStep == 0) if(ttlStorage.lastTimeStep == 0)
{ {
ttlStorage.ptrFont_200 = TTF_OpenFont("../fonts/Quirkus.ttf", ttlStorage.fontSize_200); ttlStorage.ptrFont_200 = TTF_OpenFont("../fonts/Quirkus.ttf", ttlStorage.fontSize_200);
ttlStorage.ptrFont_30 = TTF_OpenFont("../fonts/Quirkus.ttf", ttlStorage.fontSize_30); ttlStorage.ptrFont_30 = TTF_OpenFont("../fonts/Quirkus.ttf", ttlStorage.fontSize_30);
@ -140,16 +134,12 @@ void renderStartMenu()
SDL_Color textColor5 = {0, 255, 255}; // türkiser Text SDL_Color textColor5 = {0, 255, 255}; // türkiser Text
SDL_Color textColor6 = {255, 255, 255}; // weißer Text SDL_Color textColor6 = {255, 255, 255}; // weißer Text
ttlStorage.textColour[0] = textColor1; ttlStorage.textColour[0] = textColor1; // rosa
ttlStorage.textColour[1] = textColor2; ttlStorage.textColour[1] = textColor2; // orange
ttlStorage.textColour[2] = textColor3; ttlStorage.textColour[2] = textColor3; // grün
ttlStorage.textColour[3] = textColor4; ttlStorage.textColour[3] = textColor4; // rot
ttlStorage.textColour[4] = textColor5; ttlStorage.textColour[4] = textColor5; // türkis
ttlStorage.textColour[5] = textColor6; ttlStorage.textColour[5] = textColor6; // weiß
// text in start screen // text in start screen
const char* textLines[] = { const char* textLines[] = {
@ -168,7 +158,7 @@ void renderStartMenu()
SDL_FreeSurface(ttlStorage.textSurface); SDL_FreeSurface(ttlStorage.textSurface);
// render poem // render poem
for (int i = 1; i < (MAX_LINES_STARTSCREEN); ++i) for (int i = 1; i < (MAX_LINES_STARTSCREEN - 1); ++i)
{ {
ttlStorage.textSurface = TTF_RenderText_Solid(ttlStorage.ptrFont_30, textLines[i], ttlStorage.textColour[i]); ttlStorage.textSurface = TTF_RenderText_Solid(ttlStorage.ptrFont_30, textLines[i], ttlStorage.textColour[i]);
ttlStorage.textTextures[i] = SDL_CreateTextureFromSurface(game.renderer, ttlStorage.textSurface); ttlStorage.textTextures[i] = SDL_CreateTextureFromSurface(game.renderer, ttlStorage.textSurface);
@ -176,22 +166,17 @@ void renderStartMenu()
} }
// render ENTER // render ENTER
ttlStorage.textSurface = TTF_RenderText_Solid(ttlStorage.ptrFont_30, textLines[MAX_LINES_STARTSCREEN-1], ttlStorage.textColour[5]);
ttlStorage.textTextures[MAX_LINES_STARTSCREEN-1] = SDL_CreateTextureFromSurface(game.renderer, ttlStorage.textSurface);
ttlStorage.textSurface = TTF_RenderText_Solid(ttlStorage.ptrFont_30, textLines[MAX_LINES_STARTSCREEN], ttlStorage.textColour[5]);
ttlStorage.textTextures[MAX_LINES_STARTSCREEN] = SDL_CreateTextureFromSurface(game.renderer, ttlStorage.textSurface);
SDL_FreeSurface(ttlStorage.textSurface); SDL_FreeSurface(ttlStorage.textSurface);
} }
//------------------------------ //=========== is always performerd ================
// is always performed
//-----------------------------
SDL_RenderClear(game.renderer); SDL_RenderClear(game.renderer);
int textWidth, textHeight; int textWidth, textHeight;
// print game name // print game name
ttlStorage.textPrintPosition = (config.windowSize) / 9; // print position for game name (SNAKE++) ttlStorage.textPrintPosition = (config.windowSize) / 9; // print position for game name (SNAKE++)
SDL_QueryTexture(ttlStorage.textTextures[0], NULL, NULL, &textWidth, &textHeight); SDL_QueryTexture(ttlStorage.textTextures[0], NULL, NULL, &textWidth, &textHeight);
@ -200,35 +185,165 @@ void renderStartMenu()
// print poem // print poem
ttlStorage.textPrintPosition = (config.windowSize) / 2.7; // change print position for poem ttlStorage.textPrintPosition = (config.windowSize) / 2.7; // change print position for poem
for (int i = 1; i < (MAX_LINES_STARTSCREEN); ++i) { for (int i = 1; i < (MAX_LINES_STARTSCREEN-1); ++i) {
int textWidth, textHeight;
SDL_QueryTexture(ttlStorage.textTextures[i], NULL, NULL, &textWidth, &textHeight); SDL_QueryTexture(ttlStorage.textTextures[i], NULL, NULL, &textWidth, &textHeight);
SDL_Rect dstRect = { (config.windowSize - textWidth) / 2, ttlStorage.textPrintPosition, textWidth, textHeight }; SDL_Rect dstRect = { (config.windowSize - textWidth) / 2, ttlStorage.textPrintPosition, textWidth, textHeight };
SDL_RenderCopy(game.renderer, ttlStorage.textTextures[i], NULL, &dstRect); SDL_RenderCopy(game.renderer, ttlStorage.textTextures[i], NULL, &dstRect);
ttlStorage.textPrintPosition += textHeight; // increase print position ttlStorage.textPrintPosition += textHeight; // increase print position
} }
//print ENTER every second cycle //print ENTER every second cycle
if(ttlStorage.showEnter) if(ttlStorage.showEnter)
{ {
ttlStorage.textPrintPosition = (config.windowSize / 1.5); // print position for ENTER ttlStorage.textPrintPosition = (config.windowSize / 1.5); // print position for ENTER
SDL_QueryTexture(ttlStorage.textTextures[MAX_LINES_STARTSCREEN], NULL, NULL, &textWidth, &textHeight); SDL_QueryTexture(ttlStorage.textTextures[MAX_LINES_STARTSCREEN-1], NULL, NULL, &textWidth, &textHeight);
SDL_Rect dstRect1 = { (config.windowSize - textWidth) / 2, ttlStorage.textPrintPosition, textWidth, textHeight }; SDL_Rect dstRect1 = { (config.windowSize - textWidth) / 2, ttlStorage.textPrintPosition, textWidth, textHeight };
SDL_RenderCopy(game.renderer, ttlStorage.textTextures[MAX_LINES_STARTSCREEN], NULL, &dstRect1); SDL_RenderCopy(game.renderer, ttlStorage.textTextures[MAX_LINES_STARTSCREEN-1], NULL, &dstRect1);
} }
// update screen
SDL_RenderPresent(game.renderer); SDL_RenderPresent(game.renderer);
ttlStorage.lastTimeStep = GET_TIME_MS(); ttlStorage.lastTimeStep = GET_TIME_MS();
// show inital menu frame
return;
}
//--------------------------------------------------------------
//-------------------SETTINGS-----------------------------------
//--------------------------------------------------------------
void renderSettings()
{
//=========== only first loop ================
if(ttlStorage.lastTimeStep == 0)
{
ttlStorage.ptrFont_20 = TTF_OpenFont("../fonts/Prototype.ttf", ttlStorage.fontSize_20);
// text in start screen
const char* textLines[] = {
"Fuer Infos zum Gameplay, sowie einigen Shortcuts druecken Sie bitte F1",
" ",
//"Bitte geben Sie ihren Spielnamen ein: ",
"test",
"test",
"test",
"test",
"-- ENTER --"
};
// render setting screen
for (int i = 0; i < MAX_LINES_SETTINGS; ++i)
{
ttlStorage.textSurface = TTF_RenderText_Solid(ttlStorage.ptrFont_20, textLines[i], ttlStorage.textColour[5]);
ttlStorage.textTextures[i] = SDL_CreateTextureFromSurface(game.renderer, ttlStorage.textSurface);
SDL_FreeSurface(ttlStorage.textSurface);
}
}
//=========== is always performerd ================
SDL_RenderClear(game.renderer);
int textWidth, textHeight;
// print settings
ttlStorage.textPrintPosition = 0; // first print position
for (int i = 0; i < (MAX_LINES_SETTINGS - 1); ++i) {
SDL_QueryTexture(ttlStorage.textTextures[i], NULL, NULL, &textWidth, &textHeight);
SDL_Rect dstRect = { 1, ttlStorage.textPrintPosition, textWidth, textHeight };
SDL_RenderCopy(game.renderer, ttlStorage.textTextures[i], NULL, &dstRect);
ttlStorage.textPrintPosition += textHeight; // increase print position
}
//print ENTER every second cycle
if(ttlStorage.showEnter)
{
ttlStorage.textPrintPosition = (config.windowSize / 1.5); // print position for ENTER
SDL_QueryTexture(ttlStorage.textTextures[MAX_LINES_SETTINGS-1], NULL, NULL, &textWidth, &textHeight);
SDL_Rect dstRect1 = { (config.windowSize - textWidth) / 2, ttlStorage.textPrintPosition, textWidth, textHeight };
SDL_RenderCopy(game.renderer, ttlStorage.textTextures[MAX_LINES_SETTINGS-1], NULL, &dstRect1);
}
// update screen
SDL_RenderPresent(game.renderer);
ttlStorage.lastTimeStep = GET_TIME_MS();
return; return;
}
//--------------------------------------------------------------
//-------------------INFO SCREEN--------------------------------
//--------------------------------------------------------------
void renderInfoScreen()
{
//=========== only first loop ================
if(ttlStorage.lastTimeStep == 0)
{
// text in start screen
const char* textLines[] = {
" Steuerung: W (nach oben)",
" A (nach links)",
" S (nach unten)",
" D (nach rechts) ",
" oder: Pfeiltasten ",
" ",
" ",
" Pause: p",
" oder: Leertaste",
" ",
" ",
"Spiel verlassen: q ",
" ",
" ",
" ",
"By Jonas Schoenberger, Johannes Graf und Julia Steinberger",
"-- ENTER --"
};
// render setting screen
for (int i = 0; i < MAX_LINE_INFOSCREEN; ++i)
{
ttlStorage.textSurface = TTF_RenderText_Solid(ttlStorage.ptrFont_20, textLines[i], ttlStorage.textColour[5]);
ttlStorage.textTextures[i] = SDL_CreateTextureFromSurface(game.renderer, ttlStorage.textSurface);
SDL_FreeSurface(ttlStorage.textSurface);
}
}
//=========== is always performerd ================
SDL_RenderClear(game.renderer);
int textWidth, textHeight;
// print settings
ttlStorage.textPrintPosition = 0; // first print position
for (int i = 0; i < (MAX_LINE_INFOSCREEN - 1); ++i) {
SDL_QueryTexture(ttlStorage.textTextures[i], NULL, NULL, &textWidth, &textHeight);
SDL_Rect dstRect = { 1, ttlStorage.textPrintPosition, textWidth, textHeight };
SDL_RenderCopy(game.renderer, ttlStorage.textTextures[i], NULL, &dstRect);
ttlStorage.textPrintPosition += textHeight; // increase print position
}
//print ENTER every second cycle
if(ttlStorage.showEnter)
{
ttlStorage.textPrintPosition = (config.windowSize / 1.5); // print position for ENTER
SDL_QueryTexture(ttlStorage.textTextures[MAX_LINE_INFOSCREEN-1], NULL, NULL, &textWidth, &textHeight);
SDL_Rect dstRect1 = { (config.windowSize - textWidth) / 2, ttlStorage.textPrintPosition, textWidth, textHeight };
SDL_RenderCopy(game.renderer, ttlStorage.textTextures[MAX_LINE_INFOSCREEN-1], NULL, &dstRect1);
}
// update screen
SDL_RenderPresent(game.renderer);
ttlStorage.lastTimeStep = GET_TIME_MS();
return;
} }
@ -252,6 +367,7 @@ int CreateSDLWindow(){
void DestroySDLWindow(){ void DestroySDLWindow(){
// Zerstöre das Fenster und beende SDL // Zerstöre das Fenster und beende SDL
TTF_CloseFont(ttlStorage.ptrFont_20);
TTF_CloseFont(ttlStorage.ptrFont_30); TTF_CloseFont(ttlStorage.ptrFont_30);
TTF_CloseFont(ttlStorage.ptrFont_200); TTF_CloseFont(ttlStorage.ptrFont_200);