Skip to content

Commit b9b7d36

Browse files
authored
Merge develop branch into Master branch - Release v1.1.0 (#5)
2 parents a261be3 + 77a4b6a commit b9b7d36

File tree

116 files changed

+57713
-406
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

116 files changed

+57713
-406
lines changed

.gitignore

-1
Original file line numberDiff line numberDiff line change
@@ -51,7 +51,6 @@ obj/
5151
*.lo
5252

5353
# Shared objects (inc. Windows DLLs)
54-
*.dll
5554
*.so
5655
*.so.*
5756
*.dylib

.vscode/tasks.json

+36-8
Original file line numberDiff line numberDiff line change
@@ -8,34 +8,62 @@
88
"& (mkdir \"${workspaceFolder}\\libs\\bin\\Release\")"
99
],
1010
"group": {"kind": "build", "isDefault": false},
11-
"label": "Create \\libs\\obj\\Release and \\libs\\bin\\Release folders",
11+
"label": "Create `\\libs\\obj\\Release` and `\\libs\\bin\\Release` folders",
1212
"type": "shell"
1313
},
1414
{
1515
"command": [
1616
"(gcc -Wall -O2 -pedantic-errors -c \"${workspaceFolder}\\libs\\utilities.c\" -o \"${workspaceFolder}\\libs\\obj\\Release\\utilities.o\")",
1717
"& (gcc -Wall -O2 -pedantic-errors -c \"${workspaceFolder}\\libs\\validators.c\" -o \"${workspaceFolder}\\libs\\obj\\Release\\validators.o\")",
18-
"& (gcc -Wall -O2 -pedantic-errors -c \"${workspaceFolder}\\libs\\patterns\\constructors.c\" -o \"${workspaceFolder}\\libs\\obj\\Release\\constructors.o\")",
19-
"& (gcc -Wall -O2 -pedantic-errors -c \"${workspaceFolder}\\libs\\patterns\\methods.c\" -o \"${workspaceFolder}\\libs\\obj\\Release\\methods.o\")",
20-
"& (ar -r -s \"${workspaceFolder}\\libs\\bin\\Release\\libs.a\" \"${workspaceFolder}\\libs\\obj\\Release\\constructors.o\" \"${workspaceFolder}\\libs\\obj\\Release\\methods.o\" \"${workspaceFolder}\\libs\\obj\\Release\\utilities.o\" \"${workspaceFolder}\\libs\\obj\\Release\\validators.o\")"
18+
"& (gcc -Wall -O2 -pedantic-errors -c \"${workspaceFolder}\\libs\\game\\methods.c\" -o \"${workspaceFolder}\\libs\\obj\\Release\\game-methods.o\")",
19+
"& (gcc -Wall -O2 -pedantic-errors -c \"${workspaceFolder}\\libs\\patterns\\constructors.c\" -o \"${workspaceFolder}\\libs\\obj\\Release\\patterns-constructors.o\")",
20+
"& (gcc -Wall -O2 -pedantic-errors -c \"${workspaceFolder}\\libs\\patterns\\methods.c\" -o \"${workspaceFolder}\\libs\\obj\\Release\\patterns-methods.o\")",
21+
"& (ar -r -s \"${workspaceFolder}\\libs\\bin\\Release\\libs.a\" \"${workspaceFolder}\\libs\\obj\\Release\\game-methods.o\" \"${workspaceFolder}\\libs\\obj\\Release\\patterns-constructors.o\" \"${workspaceFolder}\\libs\\obj\\Release\\patterns-methods.o\" \"${workspaceFolder}\\libs\\obj\\Release\\utilities.o\" \"${workspaceFolder}\\libs\\obj\\Release\\validators.o\")"
2122
],
2223
"group": {"kind": "build", "isDefault": false},
2324
"label": "Compile libs project",
2425
"type": "shell",
25-
"dependsOn": "Create \\libs\\obj\\Release and \\libs\\bin\\Release folders"
26+
"dependsOn": "Create `\\libs\\obj\\Release` and `\\libs\\bin\\Release` folders"
27+
},
28+
{
29+
"command": [
30+
"(if exist \"${workspaceFolder}\\src\\obj\" rmdir /s /q \"${workspaceFolder}\\src\\obj\")",
31+
"& (mkdir \"${workspaceFolder}\\src\\obj\\Release\")",
32+
"& (if exist \"${workspaceFolder}\\src\\bin\" rmdir /s /q \"${workspaceFolder}\\src\\bin\")",
33+
"& (mkdir \"${workspaceFolder}\\src\\bin\\Release\")"
34+
],
35+
"group": {"kind": "build", "isDefault": false},
36+
"label": "Create `\\src\\obj\\Release` and `\\src\\bin\\Release` folders",
37+
"type": "shell"
38+
},
39+
{
40+
"command": "copy \"${workspaceFolder}\\src\\SDL2.dll\" \"${workspaceFolder}\\src\\bin\\Release\\SDL2.dll\"",
41+
"dependsOn": "Create `\\src\\obj\\Release` and `\\src\\bin\\Release` folders",
42+
"group": {"kind": "build", "isDefault": false},
43+
"label": "Copy `SDL2.dll` file from `.\\src\\` into `.\\src\\bin\\Release\\",
44+
"type": "shell"
2645
},
2746
{
2847
"args": [
2948
"-fdiagnostics-color=always",
3049
"-g",
31-
"${file}",
50+
"${workspaceFolder}\\src\\*.c",
51+
"${workspaceFolder}\\src\\sdl\\*.c",
3252
"-o",
33-
"${fileDirname}\\bin\\Release\\${fileBasenameNoExtension}.exe",
53+
"${workspaceFolder}\\src\\bin\\Release\\src.exe",
54+
"-I${workspaceFolder}\\src\\sdl\\SDL2\\include\\SDL2",
55+
"-I${workspaceFolder}\\src\\sdl\\SDL2\\include",
56+
"-L${workspaceFolder}\\src\\sdl\\SDL2\\lib",
57+
"-lSDL2",
58+
"-lSDL2main",
3459
"-s",
3560
"${workspaceFolder}\\libs\\bin\\Release\\libs.a"
3661
],
3762
"command": "gcc",
38-
"dependsOn": "Compile libs project",
63+
"dependsOn": [
64+
"Compile libs project",
65+
"Copy `SDL2.dll` file from `.\\src\\` into `.\\src\\bin\\Release\\"
66+
],
3967
"group": {"kind": "build", "isDefault": true},
4068
"label": "C/C++: gcc.exe build active file",
4169
"problemMatcher": ["$gcc"],

libs/game/macros.h

+71
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,71 @@
1+
2+
#ifndef GAME_MACROS_H_INCLUDED
3+
#define GAME_MACROS_H_INCLUDED
4+
5+
/**
6+
* @def DASHBOARD_ROWS
7+
* @brief Defines the number of rows in a grid.
8+
*
9+
* This macro is used to define the number of rows in a grid.
10+
* It is typically used in conjunction with the `DASHBOARD_COLS` macro to define
11+
* the size of a grid.
12+
*
13+
* @warning The value of `DASHBOARD_ROWS` must be a positive integer.
14+
*/
15+
#define DASHBOARD_ROWS 56
16+
17+
/**
18+
* @def DASHBOARD_COLS
19+
* @brief Defines the number of columns in a grid.
20+
*
21+
* This macro is used to define the number of columns in a grid.
22+
* It is typically used in conjunction with the `DASHBOARD_ROWS` macro to define
23+
* the size of a grid.
24+
*
25+
* @warning The value of `DASHBOARD_COLS` must be a positive integer.
26+
*/
27+
#define DASHBOARD_COLS 110
28+
29+
/**
30+
* @def ALIVE_CELL
31+
* @brief Represents a live cell.
32+
*
33+
* @warning The value of `ALIVE_CELL` must be a single character.
34+
*/
35+
#define ALIVE_CELL 'o'
36+
37+
/**
38+
* @def ALIVE_CELL_NG
39+
* @brief Represents a cell that is going to be alive in the next generation.
40+
*
41+
* @warning The value of `ALIVE_CELL_NG` must be a single character and must not be equal to
42+
* `ALIVE_CELL`, `DEAD_CELL`, or `DEAD_CELL_NG` macros.
43+
*/
44+
#define ALIVE_CELL_NG '1'
45+
46+
/**
47+
* @def DEAD_CELL
48+
* @brief Represents a dead cell.
49+
*
50+
* @warning The value of `DEAD_CELL` must be a single character.
51+
*/
52+
#define DEAD_CELL ' '
53+
54+
/**
55+
* @def DEAD_CELL_NG
56+
* @brief Represents a cell that is going to be dead in the next generation.
57+
*
58+
* @warning The value of `DEAD_CELL_NG` must be a single character and must not be equal to
59+
* `ALIVE_CELL`, `ALIVE_CELL_NG`, or `DEAD_CELL` macros.
60+
*/
61+
#define DEAD_CELL_NG '0'
62+
63+
/**
64+
* @def NEIGHBORHOOD_RADIUS
65+
* @brief Defines the radius of the neighborhood.
66+
*
67+
* @warning The value of `NEIGHBORHOOD_RADIUS` must be greater or equal to 1.
68+
*/
69+
#define NEIGHBORHOOD_RADIUS 1
70+
71+
#endif // GAME_MACROS_H_INCLUDED

libs/game/main.h

+10
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
2+
#ifndef GAME_MAIN_H_INCLUDED
3+
#define GAME_MAIN_H_INCLUDED
4+
5+
// Root
6+
#include "./macros.h"
7+
#include "./methods.h"
8+
#include "./structs.h"
9+
10+
#endif // GAME_MAIN_H_INCLUDED

libs/game/methods.c

+233
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,233 @@
1+
2+
#include "./methods.h"
3+
4+
#include <limits.h>
5+
#include <stdio.h>
6+
7+
#include "../patterns/main.h"
8+
#include "./macros.h"
9+
#include "./structs.h"
10+
11+
int countAliveNeighbors(TGame* pGame, int cellRow, int cellCol, int radius) {
12+
int i;
13+
int j;
14+
15+
int startRow = cellRow - radius;
16+
int startCol = cellCol - radius;
17+
18+
int endRow = cellRow + radius + 1;
19+
int endCol = cellCol + radius + 1;
20+
21+
int aliveNeighbors = 0;
22+
23+
for (i = startRow; i < endRow; i++) {
24+
if (i > pGame->rows - 1) break;
25+
if (i < 0) continue;
26+
27+
for (j = startCol; j < endCol; j++) {
28+
if (j > pGame->cols - 1) break;
29+
if (j < 0) continue;
30+
31+
if (i == cellRow && j == cellCol) continue;
32+
33+
if (pGame->dashboard[i][j] == ALIVE_CELL || pGame->dashboard[i][j] == DEAD_CELL_NG)
34+
aliveNeighbors++;
35+
}
36+
}
37+
38+
return aliveNeighbors;
39+
}
40+
41+
void drawPattern(TGame* pGame, char* pattern) {
42+
TPattern SPattern;
43+
44+
char arr[PATTERN_ROWS][PATTERN_COLS];
45+
46+
SPattern.arr = arr;
47+
48+
fillDashboard(pGame, DEAD_CELL);
49+
50+
if (strcmpi(pattern, "glider") == 0) {
51+
newGliderPattern(&SPattern);
52+
pGame->cellsAlive = 5;
53+
} else if (strcmpi(pattern, "glider cannon") == 0) {
54+
newGliderCannonPattern(&SPattern);
55+
pGame->cellsAlive = 36;
56+
} else if (strcmpi(pattern, "press") == 0) {
57+
newPressPattern(&SPattern);
58+
pGame->cellsAlive = 48;
59+
} else if (strcmpi(pattern, "toad") == 0) {
60+
newToadPattern(&SPattern);
61+
pGame->cellsAlive = 6;
62+
} else {
63+
return;
64+
}
65+
66+
pGame->cellsDead = (pGame->cols * pGame->rows) - pGame->cellsAlive;
67+
pGame->generation = 0;
68+
69+
drawPatternInDashboard(pGame, &SPattern);
70+
}
71+
72+
void drawPatternInDashboard(TGame* pGame, TPattern* pPattern) {
73+
int i;
74+
int j;
75+
76+
int pI = 0;
77+
int pJ = 0;
78+
79+
int startRow = pGame->center[0] - pPattern->center[0];
80+
int startCol = pGame->center[1] - pPattern->center[1];
81+
82+
for (i = startRow; pI < pPattern->rows; i++) {
83+
if (i < 0) continue;
84+
if (i > pGame->rows - 1) break;
85+
86+
for (j = startCol; pJ < pPattern->cols; j++) {
87+
if (j < 0) continue;
88+
if (j > pGame->cols - 1) break;
89+
90+
pGame->dashboard[i][j] = pPattern->arr[pI][pJ];
91+
pJ++;
92+
};
93+
94+
pJ = 0;
95+
pI++;
96+
}
97+
}
98+
99+
void fillDashboard(TGame* pGame, char with) {
100+
int i;
101+
int j;
102+
103+
for (i = 0; i < pGame->rows; i++) {
104+
for (j = 0; j < pGame->cols; j++) {
105+
pGame->dashboard[i][j] = with;
106+
}
107+
}
108+
}
109+
110+
void generateNextGeneration(TGame* pGame) {
111+
int i;
112+
int j;
113+
114+
int aliveNeighbors;
115+
116+
for (i = 0; i < pGame->rows; i++) {
117+
for (j = 0; j < pGame->cols; j++) {
118+
aliveNeighbors = countAliveNeighbors(pGame, i, j, NEIGHBORHOOD_RADIUS);
119+
120+
if (pGame->dashboard[i][j] == DEAD_CELL) {
121+
if (aliveNeighbors == 3) {
122+
pGame->cellsDead--;
123+
pGame->cellsAlive++;
124+
pGame->dashboard[i][j] = ALIVE_CELL_NG;
125+
};
126+
127+
continue;
128+
}
129+
130+
if (aliveNeighbors < 2 || aliveNeighbors > 3) {
131+
pGame->cellsAlive--;
132+
pGame->cellsDead++;
133+
pGame->dashboard[i][j] = DEAD_CELL_NG;
134+
};
135+
}
136+
}
137+
138+
for (i = 0; i < pGame->rows; i++) {
139+
for (j = 0; j < pGame->cols; j++) {
140+
if (pGame->dashboard[i][j] == DEAD_CELL_NG) {
141+
pGame->dashboard[i][j] = DEAD_CELL;
142+
continue;
143+
}
144+
145+
if (pGame->dashboard[i][j] == ALIVE_CELL_NG) pGame->dashboard[i][j] = ALIVE_CELL;
146+
}
147+
}
148+
}
149+
150+
void printDashboardByConsole(TGame* pGame) {
151+
int i;
152+
int j;
153+
154+
for (i = 0; i < pGame->rows; i++) {
155+
printf("\n");
156+
157+
for (j = 0; j < pGame->cols; j++) {
158+
printf("%c", pGame->dashboard[i][j]);
159+
}
160+
}
161+
}
162+
163+
void printGameByConsole(TGame* pGame) {
164+
int i;
165+
int j;
166+
167+
// Print header
168+
for (i = 0; i < pGame->cols + 2; i++) printf("-");
169+
170+
printf("\n| Cells alive: %*d |", pGame->cols - 17 + 2, pGame->cellsAlive);
171+
printf("\n| Cells dead: %*d |", pGame->cols - 16 + 2, pGame->cellsDead);
172+
printf("\n| Generation: %*d |", pGame->cols - 16 + 2, pGame->generation);
173+
174+
if (pGame->maximumGeneration == INT_MAX) {
175+
printf("\n| Maximum generation: %*s |", pGame->cols - 25 + 3, "infinity");
176+
} else {
177+
printf("\n| Maximum generation: %*d |", pGame->cols - 25 + 3, pGame->maximumGeneration);
178+
}
179+
180+
printf("\n| Delay between generations: %*d ms |\n", pGame->cols - 35 + 3,
181+
pGame->delayBetweenGenerations);
182+
183+
// Print dashboard
184+
for (i = 0; i < pGame->cols + 2; i++) printf("-");
185+
186+
for (i = 0; i < pGame->rows; i++) {
187+
printf("\n|");
188+
189+
for (j = 0; j < pGame->cols; j++) {
190+
printf("%c", pGame->dashboard[i][j]);
191+
}
192+
193+
printf("|");
194+
}
195+
196+
printf("\n");
197+
for (i = 0; i < pGame->cols + 2; i++) printf("-");
198+
}
199+
200+
void setDashboardCenter(TGame* pGame) {
201+
int row = pGame->rows / 2;
202+
int col = pGame->cols / 2;
203+
204+
pGame->center[0] = row;
205+
pGame->center[1] = col;
206+
}
207+
208+
void startGameByConsole(TGame* pGame, int maxGeneration, int delayBetweenGenerations) {
209+
int generation = 0;
210+
int isToInfinity = maxGeneration == INT_MAX;
211+
212+
pGame->generation = 0;
213+
pGame->maximumGeneration = maxGeneration;
214+
pGame->delayBetweenGenerations = delayBetweenGenerations;
215+
216+
system("cls");
217+
printGameByConsole(pGame);
218+
if (generation == maxGeneration) return;
219+
sleep(delayBetweenGenerations);
220+
221+
while (isToInfinity || generation < maxGeneration) {
222+
generateNextGeneration(pGame);
223+
224+
if (generation != INT_MAX) {
225+
generation++;
226+
pGame->generation = generation;
227+
};
228+
229+
system("cls");
230+
printGameByConsole(pGame);
231+
if (generation != maxGeneration) sleep(delayBetweenGenerations);
232+
}
233+
}

0 commit comments

Comments
 (0)