From b5e85de24cc19e5aef8338bf72ef04c82dc98385 Mon Sep 17 00:00:00 2001 From: Hanse-14 Date: Mon, 18 Dec 2023 04:47:44 +0100 Subject: [PATCH] Add module 'files' - write score after death - read a number of scores --- CMakeLists.txt | 1 + include/files.h | 26 +++++++++ include/game.h | 13 +---- src/files.c | 147 ++++++++++++++++++++++++++++++++++++++++++++++++ src/game.c | 41 ++------------ src/menu.c | 1 + 6 files changed, 183 insertions(+), 46 deletions(-) create mode 100644 include/files.h create mode 100644 src/files.c diff --git a/CMakeLists.txt b/CMakeLists.txt index 9f9cc51..e776c8b 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -53,6 +53,7 @@ set(SOURCES src/common.c src/sound.c src/difficulty.c + src/files.c ) diff --git a/include/files.h b/include/files.h new file mode 100644 index 0000000..0c3f0c0 --- /dev/null +++ b/include/files.h @@ -0,0 +1,26 @@ +#pragma once +#include "config.h" + +#define MAX_PRINTED_SCORES 5 + + +// struct that store player score at the end of the game +typedef struct playerScore_t +{ + int score; + char playerName[30]; + int difficulty; + char map[30]; +} playerScore_t; + +// global struct for storing the 10 best players after reading it from the file (defined in files.c) +extern playerScore_t topScores[MAX_PRINTED_SCORES]; + +// function which saves score in a .bin file +void savePlayerScore(const char *filename); + +// function which reads the 10 best finisher from the .bin file +void readTopScores(const char *filename); + +// counts records in file +int countRecordsInFile(const char *filename); \ No newline at end of file diff --git a/include/game.h b/include/game.h index f30aff7..9a971d8 100644 --- a/include/game.h +++ b/include/game.h @@ -33,19 +33,14 @@ typedef struct gameData_t } gameData_t; -// struct that store player score at the end of the game -typedef struct playerScore_t -{ - int score; - char playerName[30]; - int difficulty; - char map[30]; -} playerScore_t; + // global struct for storing all game data (defined in game.c) extern gameData_t game; + + // run once at game start and does the following: // - init snake // - load map @@ -63,5 +58,3 @@ void handlePortals(); //(ran in gameCycle) // - triggers frame update (render.c) void runGameCycle(); -// function which saves score in a .csv file - // void savePlayerScore() \ No newline at end of file diff --git a/src/files.c b/src/files.c new file mode 100644 index 0000000..615140b --- /dev/null +++ b/src/files.c @@ -0,0 +1,147 @@ +#include "game.h" +#include "menu.h" +#include "files.h" + + + +//global struct for storing all data of the 10 best players +playerScore_t topScores[]; + + + +//========================== +//==== savePlayerScores ==== +//========================== + void savePlayerScore(const char *filename/*int score, int difficulty, const char *playerName, const char *map*/) + { + playerScore_t playerScore; + + // copy data into struct + playerScore.score = game.snake.length - config.snakeDefaultLength; + playerScore.difficulty = config.difficulty; + strcpy(playerScore.playerName, ttlStorage.userName); + strcpy(playerScore.map, "testmap"); + + + FILE *file; + // open file + file = fopen(filename, "ab"); + + // write data in file + if (file != NULL) + { + fwrite(&playerScore, sizeof(playerScore_t), 1, file); + fclose(file); + + LOGI("Spielergebnis wurde erfolgreich in die Binaerdatei gespeichert.\n"); + } + else + { + LOGI("Fehler beim Öffnen der Datei!\n"); + } +} + + +//========================== +//==== readTopScores ====== +//========================== +// number of reads depends on 'MAX_PRINTED_SCORES' +void readTopScores(const char *filename) +{ + FILE *filePtr; + playerScore_t tempPlayerScore; + int recordsInFile; + int highestPlayerScore; + int count = 0; // increase up to 'MAX_PRINTED_SCORES' + + // determine the number of contents in the file + recordsInFile = countRecordsInFile(filename); + // if failure + if(recordsInFile == -1) + { + game.gameState = EXIT; + return; + } + + filePtr = fopen(filename, "rb"); + + // fail with file opening + if (filePtr == NULL) + { + LOGI("Datei: Fehler beim Öffnen der Datei für die besten 10 Ergebnisse!\n"); + game.gameState = EXIT; + return; + } + + + LOGI("Datensaetze in Datei: %d\n", recordsInFile); + + + //---- search for the highest score------ + for (int i = 0; i < recordsInFile; i++) + { + fread(&tempPlayerScore, sizeof(playerScore_t), 1, filePtr); + if(tempPlayerScore.score > highestPlayerScore) + { + highestPlayerScore == tempPlayerScore.score; + } + } + + //--- decrease highest score ----- + while((count < MAX_PRINTED_SCORES) && (count < MAX_PRINTED_SCORES)) + { + // set file pointer to start of the file + rewind(filePtr); + + // search for the highest score and then save it in topScores + for (int i = 0; i < recordsInFile; i++) + { + fread(&tempPlayerScore, sizeof(playerScore_t), 1, filePtr); + + // current highscore found + if(tempPlayerScore.score == highestPlayerScore) + { + topScores[count] = tempPlayerScore; + LOGI("score: %d name: %s schwierigkeit: %d map: %s\n", topScores[count].score, topScores[count].playerName, topScores[count].difficulty, topScores[count].map); + count++; + } + + // leave if MAX_PRINTED_SCORES is reached + if(count >= MAX_PRINTED_SCORES) + { + break; + } + } + highestPlayerScore--; + } + + fclose(filePtr); +} + + + +//========================== +//==== readTop10Scores ===== +//========================== +// return the number of various scores of the file +int countRecordsInFile(const char *filename) +{ + FILE *file; + file = fopen(filename, "rb"); + + // failure + if (file == NULL) + { + LOGI("Datei: Fehler beim Öffnen der Datei!\n"); + return -1; + } + + fseek(file, 0, SEEK_END); // Gehe zum Dateiende + long fileSize = ftell(file); // Hole die Größe der Datei in Bytes + fclose(file); + + int recordSize = sizeof(playerScore_t); + int numberOfRecords = fileSize / recordSize; // Berechne die Anzahl der Datensätze + + return numberOfRecords; +} diff --git a/src/game.c b/src/game.c index 10c11b4..d416fb2 100644 --- a/src/game.c +++ b/src/game.c @@ -5,6 +5,7 @@ #include "food.h" #include "render.h" #include "sound.h" +#include "files.h" // global struct for storing all game data @@ -111,8 +112,10 @@ void runGameCycle() // TODO consider game.lifesRemaining and reset if still good? LOGI("game: collided with wall or self! => show leaderboard\n"); //game.gameState = MENU; //TODO add config.collisionEnabled option? - //savePlayerScore(/*(game.snake.length - config.snakeDefaultLength), ttlStorage.userName, config.difficulty, *storedMaps[ttlStorage.userSelectedMap - 1]*/); - showLeaderboard(); + savePlayerScore("../build/player_scores.bin"/*(game.snake.length - config.snakeDefaultLength), ttlStorage.userName, config.difficulty, *storedMaps[ttlStorage.userSelectedMap - 1]*/); + readTopScores("../build/player_scores.bin"); + game.gameState = MENU; + activeMenu = LEADERBOARD; return; } @@ -136,37 +139,3 @@ void runGameCycle() #endif return; } - - -//========================== -//==== savePlayerScores ==== -//========================== -// void savePlayerScore(/*int score, int difficulty, const char *playerName, const char *map*/) -// { -// // playerScore_t playerScore; - -// // // copy data into struct -// // playerScore.score = game.snake.length - config.snakeDefaultLength; -// // playerScore.difficulty = config.difficulty; -// // strcpy(playerScore.playerName, ttlStorage.userName); -// // strcpy(playerScore.map, "testmap"); - - -// // FILE *file; -// // // open file -// // file = fopen("../player_scores.bin", "ab"); - -// // // write data in file -// // if (file != NULL) -// // { -// // fwrite(&playerScore, sizeof(playerScore_t), 1, file); -// // fclose(file); - -// // LOGI("Spielergebnis wurde erfolgreich in die Binärdatei gespeichert.\n"); -// // } -// // else -// // { -// // LOGI("Fehler beim Öffnen der Datei!\n"); -// // } -// } - diff --git a/src/menu.c b/src/menu.c index 81ba624..4f3e66a 100644 --- a/src/menu.c +++ b/src/menu.c @@ -77,6 +77,7 @@ void showLeaderboard() { LOGD("menu: showing leaderboard\n"); + //--- play crash sound --- //play audio file, wait until playback is finished //note: when displaying actual leaderboard, the second parameter should be 'false' to not block the program