From d75d3ca8b56356e38eefc93b6c3b0f3e5f0bb6ec Mon Sep 17 00:00:00 2001 From: rillk500 Date: Sun, 5 Apr 2020 19:47:21 +0600 Subject: [PATCH] Snake Game with comments --- lesson#14/.gitignore | 15 +++ lesson#14/dub.json | 13 +++ lesson#14/dub.selections.json | 6 + lesson#14/main.d | 3 - lesson#14/source/app.d | 200 ++++++++++++++++++++++++++++++++++ 5 files changed, 234 insertions(+), 3 deletions(-) create mode 100644 lesson#14/.gitignore create mode 100644 lesson#14/dub.json create mode 100644 lesson#14/dub.selections.json delete mode 100644 lesson#14/main.d create mode 100644 lesson#14/source/app.d diff --git a/lesson#14/.gitignore b/lesson#14/.gitignore new file mode 100644 index 0000000..76e70f2 --- /dev/null +++ b/lesson#14/.gitignore @@ -0,0 +1,15 @@ +.dub +docs.json +__dummy.html +docs/ +/lesson14 +lesson14.so +lesson14.dylib +lesson14.dll +lesson14.a +lesson14.lib +lesson14-test-* +*.exe +*.o +*.obj +*.lst diff --git a/lesson#14/dub.json b/lesson#14/dub.json new file mode 100644 index 0000000..f042717 --- /dev/null +++ b/lesson#14/dub.json @@ -0,0 +1,13 @@ +{ + "authors": [ + "rillk500" + ], + "copyright": "no copyright", + "dependencies": { + "raylib-d": "~>2.5.0" + }, + "libs": [ "raylib" ], + "description": "2D Snake Game", + "license": "no license", + "name": "lesson14" +} \ No newline at end of file diff --git a/lesson#14/dub.selections.json b/lesson#14/dub.selections.json new file mode 100644 index 0000000..9d49892 --- /dev/null +++ b/lesson#14/dub.selections.json @@ -0,0 +1,6 @@ +{ + "fileVersion": 1, + "versions": { + "raylib-d": "2.5.0" + } +} diff --git a/lesson#14/main.d b/lesson#14/main.d deleted file mode 100644 index e5773f3..0000000 --- a/lesson#14/main.d +++ /dev/null @@ -1,3 +0,0 @@ -import std.stdio; - -void main() {} diff --git a/lesson#14/source/app.d b/lesson#14/source/app.d new file mode 100644 index 0000000..5ea4d8c --- /dev/null +++ b/lesson#14/source/app.d @@ -0,0 +1,200 @@ +import raylib; + +import std.string: toStringz; // for converting D string into C string +import std.random: uniform; // random number generator + +immutable int windowSize = 320; // window width and height +immutable int blockSize = 16; // entity size + +enum GameStates { play, exit } // game states +enum Direction { up, down, left, right } // direction + +// Fruit struct -> generates a fruit at random location +struct Fruit { + Rectangle rect; // rectangle + + bool eaten = true; // if eaten == false, generate new fruit + + void generateFruit() { + // randomize fruit's position + rect = Rectangle(uniform(0, windowSize/blockSize)*blockSize, + uniform(0, windowSize/blockSize)*blockSize, blockSize, blockSize); + } + + // update the fruit + void update() { + if(eaten) { + generateFruit(); + eaten = false; + } + } + + // draw the fruit, if eaten, skip + void render() { + if(eaten) { + return; + } + + DrawRectangleRec(rect, GREEN); + } +} + +// Snake struct -> takes care of snake entity (updating snake logic, growing, movement, etc...) +struct Snake { + Rectangle[100] rect = Rectangle(-blockSize, -blockSize, blockSize, blockSize); // the snake: array of maximum snake length + Direction dir = Direction.up; // direction + + int length = 5; // snake length (max is windowSize^2 = 400 entities (16x16 size)) + + // init + this(int posX, int posY) { + rect[0] = Rectangle(posX, posY, blockSize, blockSize); + } + + // game logic + void update() { + // moving the tail + for(int i = length; i > 0; i--) { + rect[i].x = rect[i-1].x; + rect[i].y = rect[i-1].y; + } + + // moving the snake head + if(dir == Direction.up) { + rect[0].y -= blockSize; + } else if(dir == Direction.down) { + rect[0].y += blockSize; + } else if(dir == Direction.left) { + rect[0].x -= blockSize; + } else if(dir == Direction.right) { + rect[0].x += blockSize; + } + + // keeping the snake within the bounds of our window + if(rect[0].x < 0) { + rect[0].x = windowSize-blockSize; + } else if (rect[0].x > windowSize) { + rect[0].x = 0; + } else if(rect[0].y < 0) { + rect[0].y = windowSize-blockSize; + } else if (rect[0].y > windowSize) { + rect[0].y = 0; + } + } + + // drawing + void render() { + for(int i = 0; i < length; i++) { + DrawRectangleRec(rect[i], RED); + } + } + + // grow tail + void grow() { + length++; + } +} + +// Game struct -> manages gameplay logic, events, drawing +struct Game { + GameStates gstate = GameStates.play; + + Snake snake = Snake(windowSize/2, windowSize/2); + Fruit fruit = Fruit(); + + bool gameOver = false; + double time = 0; + + // initializing the game + this(string title) { + InitWindow(windowSize, windowSize, title.toStringz); + SetTargetFPS(30); + } + + // freing resources in the end of the game + ~this() { + CloseWindow(); + } + + // processing game events + void processEvents() { + if(WindowShouldClose()) { + gstate = GameStates.exit; + } else if(IsKeyPressed(KeyboardKey.KEY_UP)) { + snake.dir = Direction.up; + } else if(IsKeyPressed(KeyboardKey.KEY_DOWN)) { + snake.dir = Direction.down; + } else if(IsKeyPressed(KeyboardKey.KEY_LEFT)) { + snake.dir = Direction.left; + } else if(IsKeyPressed(KeyboardKey.KEY_RIGHT)) { + snake.dir = Direction.right; + } + } + + // updating game logic + void update() { + if(!gameOver) { + // update the snake + time += GetFrameTime(); + if(time > 0.12) { + time = 0; + + // check whether the head of the snake has bumbed into its tail + for(int i = 1; i < snake.length; i++) { + if(snake.rect[0].x == snake.rect[i].x && snake.rect[0].y == snake.rect[i].y) { + gameOver = true; + } + } + + // check if the snake has eaten the fruit + if(snake.rect[0].x == fruit.rect.x && snake.rect[0].y == fruit.rect.y){ + fruit.eaten = true; + + snake.grow(); + } + + // update the fruit + fruit.update(); + + // update the snake + snake.update(); + } + } + } + + // drawing + void render() { + BeginDrawing(); + ClearBackground(WHITE); + + // drawing the grid + for(int i = 1; i < windowSize/blockSize; i++) { + DrawLineEx(Vector2(blockSize*i, 0), Vector2(blockSize*i, windowSize), 1, BLACK); + DrawLineEx(Vector2(0, blockSize*i), Vector2(windowSize, blockSize*i), 1, BLACK); + } + + fruit.render(); + snake.render(); + + if(gameOver) { + DrawRectangleRec(Rectangle(0, 0, windowSize, windowSize), Color(0, 0, 0, 150)); + DrawText("Game Over!", 30, windowSize/2-48, 48, WHITE); + } + + DrawFPS(10, 10); + EndDrawing(); + } +} + + +void main() { + // init + Game game = Game("Dlang Snake Game"); + + // game loop + while(game.gstate != GameStates.exit) { + game.processEvents(); + game.update(); + game.render(); + } +}