Implement game.c, food.c and map.c
- Implemented the functions in the above files - game and map are partially tested - food is extensively tested using the created test-function - Also added DELAY(ms) macro to common.c
This commit is contained in:
parent
71c054f092
commit
061b4431c7
@ -10,15 +10,34 @@
|
||||
//conditional logging when DEBUG_OUTPUT_ENABLED is defined in config.h
|
||||
//example: LOGD("game: %d", count)
|
||||
#ifdef DEBUG_OUTPUT_ENABLED
|
||||
#define LOGD(format, ...) printf(format, ##__VA_ARGS__)
|
||||
#define LOGD(format, ...) printf("[D] " format, ##__VA_ARGS__)
|
||||
#else
|
||||
#define LOGD(format, ...) do {} while (0)
|
||||
#endif
|
||||
|
||||
//conditional logging when INFO_OUTPUT_ENABLED is defined in config.h
|
||||
//example: LOGD("game: %d", count)
|
||||
//example: LOGI("game: %d", count)
|
||||
#ifdef INFO_OUTPUT_ENABLED
|
||||
#define LOGI(format, ...) printf(format, ##__VA_ARGS__)
|
||||
#define LOGI(format, ...) printf("[I] " format, ##__VA_ARGS__)
|
||||
#else
|
||||
#define LOGI(format, ...) do {} while (0)
|
||||
#endif
|
||||
|
||||
|
||||
//===========================
|
||||
//========== DELAY ==========
|
||||
//===========================
|
||||
//macro for DELAY(ms) function that works on Windows and Linux
|
||||
#ifdef _WIN32
|
||||
#include <windows.h>
|
||||
#else
|
||||
#include <unistd.h>
|
||||
#endif
|
||||
|
||||
#ifdef _WIN32
|
||||
#include <windows.h>
|
||||
#define DELAY(ms) Sleep(ms)
|
||||
#else
|
||||
#include <unistd.h>
|
||||
#define DELAY(ms) usleep((ms) * 1000)
|
||||
#endif
|
||||
|
@ -1,12 +1,15 @@
|
||||
#pragma once
|
||||
|
||||
// global configuration macros
|
||||
#define MAX_MAP_SIZE 10
|
||||
#define MAX_MAP_SIZE 20
|
||||
#define MAX_MAP_FIELDS (MAX_MAP_SIZE*MAX_MAP_SIZE)
|
||||
|
||||
// logging settings
|
||||
#define DEBUG_OUTPUT_ENABLED
|
||||
#define INFO_OUTPUT_ENABLED
|
||||
|
||||
|
||||
|
||||
// struct for storing game configuration
|
||||
typedef struct config_t
|
||||
{
|
||||
|
@ -1,10 +1,19 @@
|
||||
#pragma once
|
||||
|
||||
#include <stdbool.h>
|
||||
|
||||
//function that spawns food respecting the following rules:
|
||||
// - not at Collision, Snake-tail, Snake-head, portal-in, portal-out position (objects)
|
||||
// - not closer to an object than minDist (if possible)
|
||||
// - not further from an object than maxDist (if possible)
|
||||
// maxDist and minDist are currently defined in food.c
|
||||
void placeFood();
|
||||
// platziert zufällig (mit bestimmtem Algorithmus) Fressen auf dem Spielfeld
|
||||
// darf nicht auf der Schlange oder auf Wänden sein
|
||||
|
||||
|
||||
//function that returns true when snake head is at current food position
|
||||
bool checkEaten();
|
||||
// Überprüft, ob Snake gefressen hat -> true wenn gefressen
|
||||
// Vergleich mit gameData_t foodX, foodY
|
||||
|
||||
|
||||
// indefinitely spawn food and print the map to console until the program is killed
|
||||
// for testing and adjusting the food placement algorithm
|
||||
void startFoodPlacementTest();
|
@ -5,6 +5,7 @@
|
||||
#include "config.h"
|
||||
#include "map.h"
|
||||
|
||||
|
||||
// Enum that defines the current game state
|
||||
typedef enum gameState_t
|
||||
{
|
||||
@ -18,32 +19,32 @@ typedef enum gameState_t
|
||||
// Struct that stores all data of the running game (all game-related functions access it globally)
|
||||
typedef struct gameData_t
|
||||
{
|
||||
snake_t snake;
|
||||
map_t map; // definition der geladenen karte
|
||||
bool mapIsLoaded; // true when config.map is valid
|
||||
int foodX, foodY; // Positon des Futters (es gibt immer nur 1 Futter)
|
||||
int lifesRemaining; // implementieren wir nicht!!
|
||||
int timestampLastCycle;
|
||||
gameState_t gameState;
|
||||
snake_t snake; // data describing snake
|
||||
map_t map; // loaded map
|
||||
bool mapIsLoaded; // true when game.map is valid
|
||||
int foodX, foodY; // current position of food
|
||||
int lifesRemaining; // not implemented
|
||||
int timestampLastCycle; // time last game cycle started
|
||||
gameState_t gameState; // state the game is in
|
||||
} gameData_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
|
||||
// - place initial food
|
||||
void gameInit();
|
||||
// berechnet BlockSizePx: windowSize/mapWidth
|
||||
// ruft snakeInit auf
|
||||
// ruft placeFood auf
|
||||
// platziert Wände
|
||||
|
||||
void handlePortals(); //(local)
|
||||
// Prüft, ob Snake sich auf einem Portal befindet
|
||||
//if true: snakeSetHeadPos auf
|
||||
|
||||
void runGameCycle();
|
||||
// checkCollision() auf
|
||||
// ruft placeFood() auf
|
||||
// ruft checkEaten() auf
|
||||
// if checkEaten then snakeGrow()
|
||||
// Snakemove(), TickTimerReset
|
||||
//ruft am Ende vom gameCycle renderGame() auf
|
||||
// when snake head is on a portal-input, sets snake head to portal-target
|
||||
void handlePortals(); //(ran in gameCycle)
|
||||
|
||||
|
||||
// function that is repeatedly run at every game tick
|
||||
// - moves snake to next position
|
||||
// - handles collision, portals, food
|
||||
// - triggers frame update (render.c)
|
||||
void runGameCycle();
|
@ -1,6 +1,8 @@
|
||||
#pragma once
|
||||
|
||||
#include <stdbool.h>
|
||||
#include "config.h"
|
||||
#include "snake.h"
|
||||
|
||||
|
||||
// Struct that stores all information needed for one Portal on the map
|
||||
@ -11,32 +13,57 @@ typedef struct portal_t
|
||||
char *color;
|
||||
} portal_t;
|
||||
|
||||
|
||||
// Struct that stores all information needed for one Collision box on the map
|
||||
typedef struct collisionBox_t
|
||||
{
|
||||
int posX, posY;
|
||||
} collisionBox_t;
|
||||
|
||||
|
||||
// Struct that describes an entire map
|
||||
typedef struct map_t {
|
||||
int width;
|
||||
int height;
|
||||
const char *name[128];
|
||||
char name[128];
|
||||
collisionBox_t collisions[MAX_MAP_FIELDS];
|
||||
int collisionCount;
|
||||
portal_t portals[MAX_MAP_FIELDS];
|
||||
int portalCount;
|
||||
} map_t;
|
||||
|
||||
//return true when provided coordinate matches a collision box
|
||||
bool checkCollides(int x, int y);
|
||||
|
||||
//generate random map based on difficulty level
|
||||
map_t generateMap(int difficulty);
|
||||
|
||||
//search and load map by name (if not found loads default map)
|
||||
// search and load map by name in storedMaps[] (map.c)
|
||||
// stops program when map not found!
|
||||
void loadMapByName(char *name);
|
||||
|
||||
|
||||
//load map by passed definition
|
||||
void loadMap(map_t map);
|
||||
|
||||
|
||||
//return true when provided coordinate matches the position of a collision box
|
||||
bool checkCollides(map_t map, int x, int y);
|
||||
|
||||
|
||||
// generate random map based on difficulty level
|
||||
// NOT IMPLEMENTED
|
||||
map_t generateMap(int difficulty, int sizeX, int sizeY);
|
||||
|
||||
|
||||
void printMap(map_t map);
|
||||
// function that prints a map to console (stdout)
|
||||
// note: currently also prints snake and food which may be bugged/unintended
|
||||
|
||||
|
||||
// function that renders all current game objects to one 2d int array
|
||||
// NOTE: passed Array has to be zero-initialized! (int arr[][] = {{0}})
|
||||
// useful for rendering game to console or sdl
|
||||
// 1=collision, 2=portalIn, 3=portalOut, 4=snakeHead, 5=snakeTail
|
||||
void renderGameToArray(int mapFrame[MAX_MAP_SIZE][MAX_MAP_SIZE], map_t map, snake_t snake);
|
||||
|
||||
|
||||
// stored map presets can be globally accessed (maybe needed by menu.c)
|
||||
// not: maps defined in map.c end of file
|
||||
extern const map_t * storedMaps[16];
|
||||
extern const int storedMapsCount;
|
146
src/food.c
146
src/food.c
@ -1,14 +1,150 @@
|
||||
#include "food.h"
|
||||
#include <stdbool.h>
|
||||
#include <math.h>
|
||||
#include <stdlib.h>
|
||||
#include "food.h"
|
||||
#include "common.h"
|
||||
#include "map.h"
|
||||
#include "game.h"
|
||||
|
||||
// platziert zufällig (mit bestimmtem Algorithmus) Fressen auf dem Spielfeld
|
||||
void placeFood(int count)
|
||||
|
||||
|
||||
//--------------------------------------------
|
||||
//----- getRandomPositionWithMinDistance -----
|
||||
//--------------------------------------------
|
||||
// local function used in placeFood that returns random coordinates that have
|
||||
// at least in_minDist blocks distance to every Collision, portal and snake block on the current map.
|
||||
void getRandomPositionWithMinDistance(int *outX, int *outY, float *out_minDist, int *out_triesNeeded, float in_minDist)
|
||||
{
|
||||
//--- config ---
|
||||
static const int maxTries = 50; // each maxTries the limits above get loosened
|
||||
|
||||
//--- get game frame ---
|
||||
// get 2d array containing position of all objects of current game state
|
||||
int gameFrame[MAX_MAP_SIZE][MAX_MAP_SIZE] = {{0}};
|
||||
renderGameToArray(gameFrame, game.map, game.snake);
|
||||
int foodX, foodY;
|
||||
|
||||
//--- search random location ---
|
||||
// search random location for food that is within the defined min distance
|
||||
int tries = 0;
|
||||
//stores distance to closest block for return value
|
||||
float minActualDistance = MAX_MAP_SIZE;
|
||||
newValues:
|
||||
while (1)
|
||||
{
|
||||
// generate random coodinates within map range
|
||||
foodX = rand() % game.map.width;
|
||||
foodY = rand() % game.map.height;
|
||||
tries++;
|
||||
// loop through all coordinates of current game/map
|
||||
for (int y = 0; y < game.map.height; y++)
|
||||
{
|
||||
for (int x = 0; x < game.map.width; x++)
|
||||
{
|
||||
if (gameFrame[y][x] > 0) // any object is present
|
||||
{
|
||||
// calculate from random-coordinate to current block
|
||||
float dist = sqrt(pow(foodX - x, 2) + pow(foodY - y, 2));
|
||||
// save minimum distance
|
||||
if (dist < minActualDistance) minActualDistance = dist;
|
||||
// verify minimum distance is kept
|
||||
if (dist < in_minDist) //too close
|
||||
{
|
||||
LOGD("food: distance: min=%.1f now=%.1f => placed too close => reroll...\n", in_minDist, dist);
|
||||
// prevent deadlock if no suitable position exists - loosen limits every k*maxTries
|
||||
if (tries % maxTries == 0)
|
||||
{
|
||||
//decrease min distance but not below 1
|
||||
if ((in_minDist -= 0.1) < 1) in_minDist = 1;
|
||||
LOGI("[WARN] food: too much tries achieving min dist -> loosen limit to %.1f\n", in_minDist);
|
||||
}
|
||||
//reset stored distance and reroll coordinates
|
||||
minActualDistance = MAX_MAP_SIZE;
|
||||
goto newValues;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
//success: no block closer than minDist to randomly generated coordinates -> break loop
|
||||
break;
|
||||
}
|
||||
// return variables
|
||||
*out_minDist = minActualDistance;
|
||||
*outX = foodX;
|
||||
*outY = foodY;
|
||||
*out_triesNeeded = tries;
|
||||
}
|
||||
|
||||
|
||||
|
||||
//=========================
|
||||
//======= placeFood =======
|
||||
//=========================
|
||||
//function that spawns food respecting the following rules:
|
||||
// - not at Collision, Snake-tail, Snake-head, portal-in, portal-out position (objects)
|
||||
// - not closer to an object than minDist (if possible)
|
||||
// - not further from an object than maxDist (if possible)
|
||||
void placeFood()
|
||||
{
|
||||
//--- config ---
|
||||
static const float minDist = 3; // new food has to be at least minDist blocks away from any object
|
||||
float maxDist = 5; // new food has to be closer than maxDist to an object
|
||||
static const int maxTries = 25; // each maxTries the limit maxDist get loosened (prevents deadlock)
|
||||
// TODO calculate this range using configured difficulty level
|
||||
// e.g. in hard difficulty the maxDist could be <2 so it is always placed close to a collision
|
||||
|
||||
//--- variables ---
|
||||
int foodX, foodY, triesMax = 0, triesMin;
|
||||
float currentMinDist;
|
||||
|
||||
//--- generate random food position within min/max range ---
|
||||
LOGD("food: generating random position + verifying min/max distance...\n");
|
||||
do
|
||||
{
|
||||
// prevent deadlock when position respecting maxDist not found
|
||||
triesMax++;
|
||||
if (triesMax % maxTries == 0)
|
||||
{
|
||||
maxDist += 0.1;
|
||||
LOGI("[WARN] food: too many tries for MAX_DIST -> loosen limits to max=%.1f\n", maxDist);
|
||||
}
|
||||
// generate random coordinates respecting minimum distance to objects
|
||||
getRandomPositionWithMinDistance(&foodX, &foodY, ¤tMinDist, &triesMin, minDist);
|
||||
//restart when max distance limit exceeded
|
||||
} while (currentMinDist > maxDist);
|
||||
|
||||
//--- update position ---
|
||||
LOGI("food: placed food at x=%d, y=%d (took %d = %d*%d tries)\n", foodX, foodY, triesMax * triesMin, triesMax, triesMin);
|
||||
game.foodX = foodX;
|
||||
game.foodY = foodY;
|
||||
return;
|
||||
}
|
||||
|
||||
// Überprüft, ob Snake gefressen hat
|
||||
|
||||
|
||||
//=============================
|
||||
//===== foodPlacementTest =====
|
||||
//=============================
|
||||
// indefinitely spawn food and print the map to console until the program is killed
|
||||
// for testing and adjusting the food placement algorithm
|
||||
void startFoodPlacementTest()
|
||||
{
|
||||
while (1)
|
||||
{
|
||||
loadMapByName("default");
|
||||
placeFood();
|
||||
printMap(game.map);
|
||||
DELAY(100);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
||||
//==========================
|
||||
//======= checkEaten =======
|
||||
//==========================
|
||||
//returns true when snake head is at current food position
|
||||
bool checkEaten()
|
||||
{
|
||||
return 0;
|
||||
return (game.snake.headX == game.foodX && game.snake.headY == game.foodY);
|
||||
}
|
||||
|
90
src/game.c
90
src/game.c
@ -1,49 +1,113 @@
|
||||
#include "game.h"
|
||||
#include "map.h"
|
||||
#include "common.h"
|
||||
#include "menu.h"
|
||||
#include "food.h"
|
||||
#include "render.h"
|
||||
|
||||
|
||||
// global struct for storing all game data
|
||||
gameData_t game;
|
||||
// default values where needed:
|
||||
gameData_t game = {
|
||||
.snake.length = 0,
|
||||
.foodX = 0,
|
||||
.foodY = 0,
|
||||
.mapIsLoaded = false,
|
||||
.lifesRemaining = 1,
|
||||
.timestampLastCycle = 0,
|
||||
.gameState = MENU
|
||||
};
|
||||
|
||||
|
||||
|
||||
//========================
|
||||
//======= gameInit =======
|
||||
//========================
|
||||
// run once at game start and does the following:
|
||||
// - init snake
|
||||
// - load map
|
||||
// - place initial food
|
||||
void gameInit()
|
||||
{
|
||||
LOGI("game: initializing game...\n");
|
||||
//----- snake -----
|
||||
// defines initial values of game.snake
|
||||
// snakeInit(); FIXME: uncomment when implemented
|
||||
snakeInit(); //TODO assign return value to game.snake?
|
||||
|
||||
//----- load map -----
|
||||
//load default map if no map loaded yet
|
||||
if (!game.mapIsLoaded){
|
||||
char * defaultName = "default";
|
||||
loadMapByName("default");
|
||||
//loadMapByName("intermediate");
|
||||
}
|
||||
// place initial food
|
||||
//placeFood(); FIXME uncomment when implemented
|
||||
|
||||
//----- initialize variables -----
|
||||
game.lifesRemaining = 1;
|
||||
// game.lifesRemaining = config.maxLifes; TODO: add maxLifes to config
|
||||
// game.gameState = RUNNING; ??
|
||||
|
||||
game.timestampLastCycle = -config.cycleDurationMs; // next cycle starts immediately
|
||||
|
||||
//--- place initial food ---
|
||||
placeFood();
|
||||
LOGI("game: placed initial food at x=%d, y=%d\n", game.foodX, game.foodY);
|
||||
}
|
||||
|
||||
|
||||
|
||||
//=========================
|
||||
//===== handlePortals =====
|
||||
//=========================
|
||||
// when snake head is on a portal-input, sets snake head to portal-target
|
||||
void handlePortals()
|
||||
{
|
||||
LOGD("game: handling portals...\n");
|
||||
// loop through all existin portals in current map (game.map)
|
||||
for (int i=0; i < game.map.portalCount; i++){
|
||||
portal_t p = game.map.portals[i]; //copy curren portal (code more readable)
|
||||
// is at portal
|
||||
if (game.snake.headX == p.posX && game.snake.headY == p.posY){
|
||||
snakeSetHeadPos(p.posX, p.posY);
|
||||
LOGI("game: entered portal i=%d at x=%d, y=%d -> set head to x=%d y=%d\n", i, p.posX, p.posY, p.targetX, p.targetY);
|
||||
return;
|
||||
}
|
||||
}
|
||||
// snake not on any portal
|
||||
return;
|
||||
}
|
||||
|
||||
|
||||
|
||||
//==========================
|
||||
//====== runGameCycle ======
|
||||
//==========================
|
||||
// function that is repeatedly run at every game tick
|
||||
// - moves snake to next position
|
||||
// - handles collision, portals, food
|
||||
// - triggers frame update (render.c)
|
||||
void runGameCycle()
|
||||
{
|
||||
if (checkCollides(game.snake.headX, game.snake.headY))
|
||||
LOGD("game: starting GameCycle %d\n", game.timestampLastCycle);
|
||||
|
||||
//--- move snake ---
|
||||
// move snake to next position
|
||||
snakeMove();
|
||||
|
||||
//--- handle collision ---
|
||||
//collision with map object or snake tail
|
||||
if (checkCollides(game.map, game.snake.headX, game.snake.headY) || !snakeIsAlive()){
|
||||
// show leaderboard when collided
|
||||
// TODO consider game.lifesRemaining and reset if still good?
|
||||
LOGI("game: collided with wall or self! => show leaderboard\n");
|
||||
game.gameState = MENU;
|
||||
showLeaderboard();
|
||||
return;
|
||||
}
|
||||
|
||||
//--- handle portals ---
|
||||
handlePortals();
|
||||
|
||||
//--- handle food ---
|
||||
if (checkEaten()) {
|
||||
LOGI("game: picked up food at x=%d y=%d -> growing, placing food\n", game.foodX, game.foodY);
|
||||
snakeGrow();
|
||||
placeFood();
|
||||
}
|
||||
|
||||
//--- update frame ---
|
||||
renderGame();
|
||||
return;
|
||||
}
|
179
src/map.c
179
src/map.c
@ -1,43 +1,202 @@
|
||||
#include <string.h>
|
||||
#include "map.h"
|
||||
#include "game.h"
|
||||
#include "common.h"
|
||||
|
||||
//===========================
|
||||
//==== renderGameToArray ====
|
||||
//===========================
|
||||
// function that renders all current game objects to one 2d int array
|
||||
// NOTE: passed Array has to be zero-initialized! (int arr[][] = {{0}})
|
||||
// useful for rendering game to console or sdl
|
||||
// 1=collision, 2=portalIn, 3=portalOut, 4=snakeHead, 5=snakeTail
|
||||
void renderGameToArray(int mapFrame[MAX_MAP_SIZE][MAX_MAP_SIZE], map_t map, snake_t snake)
|
||||
{
|
||||
// copy collisions
|
||||
for (int i = 0; i < map.collisionCount; i++)
|
||||
{
|
||||
mapFrame[map.collisions[i].posY][map.collisions[i].posX] = 1;
|
||||
}
|
||||
// copy portals
|
||||
for (int i = 0; i < map.portalCount; i++)
|
||||
{
|
||||
mapFrame[map.portals[i].posY][map.portals[i].posX] = 2;
|
||||
mapFrame[map.portals[i].targetY][map.portals[i].targetX] = 3;
|
||||
}
|
||||
// copy snake head
|
||||
mapFrame[snake.headX][snake.headY] = 4;
|
||||
// copy snake tail
|
||||
for (int i = 0; i < snake.length; i++)
|
||||
{
|
||||
mapFrame[snake.tail[i][1]][snake.tail[i][0]] = 5;
|
||||
}
|
||||
// copy food
|
||||
mapFrame[game.foodY][game.foodX] = 6;
|
||||
return;
|
||||
}
|
||||
|
||||
|
||||
|
||||
//========================
|
||||
//======= printMap =======
|
||||
//========================
|
||||
// function that prints a map to console (stdout)
|
||||
// note: currently also prints snake and food which may be bugged/unintended
|
||||
void printMap(map_t map)
|
||||
{
|
||||
LOGI("map: Preview of map '%s' (%dx%d):\n", map.name, map.width, map.height);
|
||||
int mapFrame[MAX_MAP_SIZE][MAX_MAP_SIZE] = {{0}};
|
||||
renderGameToArray(mapFrame, map, game.snake);
|
||||
// --- print top line ---
|
||||
printf("+");
|
||||
for (int i = 0; i < map.width; i++) printf("-");
|
||||
printf("+\n");
|
||||
// --- print field ---
|
||||
// loop through rows (y)
|
||||
for (int row = 0; row < map.height; row++)
|
||||
{
|
||||
printf("|"); // vert line left
|
||||
// loop through line (x)
|
||||
for (int column = 0; column < map.width; column++)
|
||||
{
|
||||
switch (mapFrame[row][column])
|
||||
{
|
||||
case 1: printf("X"); // collistion
|
||||
break;
|
||||
case 2: printf("O"); // portal-in
|
||||
break;
|
||||
case 3: printf("T"); // portal-out
|
||||
break;
|
||||
case 6: printf("F"); // food
|
||||
break;
|
||||
default: printf(" "); // empty
|
||||
break;
|
||||
}
|
||||
}
|
||||
printf("|\n"); // vert line right
|
||||
}
|
||||
// --- print bot line ---
|
||||
printf("+");
|
||||
for (int i = 0; i < map.width; i++)
|
||||
printf("-");
|
||||
printf("+\n");
|
||||
}
|
||||
|
||||
|
||||
|
||||
//===========================
|
||||
//======= generateMap =======
|
||||
//===========================
|
||||
// generate random map based on difficulty level
|
||||
map_t generateMap(int difficulty)
|
||||
// NOT IMPLEMENTED
|
||||
map_t generateMap(int difficulty, int sizeX, int sizeY)
|
||||
{
|
||||
map_t newMap;
|
||||
return newMap;
|
||||
// TODO add map generator
|
||||
}
|
||||
|
||||
// search and load map by name (if not found loads default map)
|
||||
|
||||
|
||||
//===========================
|
||||
//====== loadMapByName ======
|
||||
//===========================
|
||||
// search and load map by name in storedMaps[] (map.c)
|
||||
// stops program when map not found!
|
||||
void loadMapByName(char *name)
|
||||
{
|
||||
LOGI("map: loading map %s", name);
|
||||
// loop through all stored maps
|
||||
for (int i = 0; i < storedMapsCount; i++)
|
||||
{
|
||||
// compare name
|
||||
if (strcmp(name, storedMaps[i]->name) == 0)
|
||||
{
|
||||
// load matched map
|
||||
LOGI("map: found map '%s'\n", name);
|
||||
loadMap(*storedMaps[i]);
|
||||
return;
|
||||
}
|
||||
}
|
||||
// map not found
|
||||
printf("[FATAL ERROR] map: could not find '%s' in storedMaps!\n", name);
|
||||
game.gameState = EXIT;
|
||||
return;
|
||||
// TODO add map presets
|
||||
}
|
||||
|
||||
|
||||
//===========================
|
||||
//========= loadMap =========
|
||||
//===========================
|
||||
// load map by passed definition
|
||||
void loadMap(map_t map)
|
||||
{
|
||||
LOGI("map: loading map '%s':\n", map.name);
|
||||
#ifdef DEBUG_OUTPUT_ENABLED
|
||||
printMap(map);
|
||||
#endif
|
||||
game.map = map;
|
||||
game.mapIsLoaded = true;
|
||||
return;
|
||||
}
|
||||
|
||||
// check if there is collision at certain coordinate
|
||||
bool checkCollides(int x, int y)
|
||||
|
||||
|
||||
//===========================
|
||||
//====== checkCollides ======
|
||||
//===========================
|
||||
// check if there is collision box at a certain coordinate
|
||||
bool checkCollides(map_t map, int x, int y)
|
||||
{
|
||||
// loop through all collision boxes on the map
|
||||
for (int i = 0; i < game.map.collisionCount; i++)
|
||||
for (int i = 0; i < map.collisionCount; i++)
|
||||
{
|
||||
// return true if match found
|
||||
if (game.map.collisions[i].posX == x && game.map.collisions[i].posY == y)
|
||||
// LOGD("map: checking collision i=%d at x=%d y=%d\n", i, map.collisions[i].posX, map.collisions[i].posY);
|
||||
if (map.collisions[i].posX == x && map.collisions[i].posY == y)
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
|
||||
//TODO add map presets here:
|
||||
|
||||
//===========================
|
||||
//======= MAP PRESETS =======
|
||||
//===========================
|
||||
// stored map presets
|
||||
// TODO add more maps or map generator
|
||||
static const map_t map_default = {
|
||||
.width = 10,
|
||||
.height = 10,
|
||||
.name = "default",
|
||||
.collisions = {{8, 9}, {8, 8}, {4, 5}, {0, 1}},
|
||||
.collisionCount = 4,
|
||||
.portals = {
|
||||
{.posX = 5,
|
||||
.posY = 8,
|
||||
.targetX = 7,
|
||||
.targetY = 1,
|
||||
.color = "blue"}},
|
||||
.portalCount = 1};
|
||||
|
||||
static const map_t map_intermediate = {
|
||||
.width = 15,
|
||||
.height = 15,
|
||||
.name = "intermediate",
|
||||
.collisions = {{8, 9}, {8, 8}, {4, 5}, {0, 1}, {9, 9}, {7, 5}, {4, 0}, {3, 0}, {12, 11}, {14, 13}},
|
||||
.collisionCount = 10,
|
||||
.portals = {
|
||||
{.posX = 5,
|
||||
.posY = 8,
|
||||
.targetX = 7,
|
||||
.targetY = 1,
|
||||
.color = "blue"},
|
||||
{.posX = 1,
|
||||
.posY = 2,
|
||||
.targetX = 2,
|
||||
.targetY = 8,
|
||||
.color = "red"}},
|
||||
.portalCount = 2};
|
||||
|
||||
// global variables for accessing the stored maps
|
||||
const map_t *storedMaps[16] = {&map_default, &map_intermediate};
|
||||
const int storedMapsCount = 2;
|
Loading…
x
Reference in New Issue
Block a user