aboutsummaryrefslogtreecommitdiff
path: root/game
diff options
context:
space:
mode:
Diffstat (limited to 'game')
-rw-r--r--game/.crepe-root0
-rw-r--r--game/.gitignore1
-rw-r--r--game/CMakeLists.txt124
-rw-r--r--game/Config.h60
-rw-r--r--game/EngineConfig.h19
-rw-r--r--game/Events.h5
-rw-r--r--game/GameScene.cpp130
-rw-r--r--game/GameScene.h11
-rw-r--r--game/MoveCameraManualyScript.cpp23
-rw-r--r--game/MoveCameraManualyScript.h11
-rw-r--r--game/PreviewScene.cpp166
-rw-r--r--game/PreviewScene.h11
-rw-r--r--game/QuitScript.cpp21
-rw-r--r--game/QuitScript.h11
-rw-r--r--game/Random.cpp29
-rw-r--r--game/Random.h10
-rw-r--r--game/StartGameScript.cpp75
-rw-r--r--game/StartGameScript.h12
-rw-r--r--game/background/AquariumScript.cpp26
-rw-r--r--game/background/AquariumScript.h12
-rw-r--r--game/background/AquariumSubScene.cpp225
-rw-r--r--game/background/AquariumSubScene.h18
-rw-r--r--game/background/BackgroundSubScene.cpp37
-rw-r--r--game/background/BackgroundSubScene.h10
-rw-r--r--game/background/CMakeLists.txt9
-rw-r--r--game/background/ForestParallaxScript.cpp55
-rw-r--r--game/background/ForestParallaxScript.h17
-rw-r--r--game/background/ForestSubScene.cpp169
-rw-r--r--game/background/ForestSubScene.h15
-rw-r--r--game/background/HallwayScript.cpp70
-rw-r--r--game/background/HallwayScript.h13
-rw-r--r--game/background/HallwaySubScene.cpp167
-rw-r--r--game/background/HallwaySubScene.h24
-rw-r--r--game/background/StartSubScene.cpp527
-rw-r--r--game/background/StartSubScene.h20
-rw-r--r--game/coins/CoinPoolSubScene.cpp13
-rw-r--r--game/coins/CoinPoolSubScene.h11
-rw-r--r--game/coins/CoinScript.cpp71
-rw-r--r--game/coins/CoinScript.h14
-rw-r--r--game/coins/CoinSubScene.cpp64
-rw-r--r--game/coins/CoinSubScene.h12
-rw-r--r--game/coins/CoinSystemScript.cpp258
-rw-r--r--game/coins/CoinSystemScript.h107
-rw-r--r--game/enemy/BattleScript.cpp61
-rw-r--r--game/enemy/BattleScript.h26
-rw-r--r--game/enemy/EnemyBulletPool.cpp11
-rw-r--r--game/enemy/EnemyBulletPool.h11
-rw-r--r--game/enemy/EnemyBulletScript.cpp40
-rw-r--r--game/enemy/EnemyBulletScript.h11
-rw-r--r--game/enemy/EnemyBulletSubScene.cpp52
-rw-r--r--game/enemy/EnemyBulletSubScene.h10
-rw-r--r--game/enemy/EnemyConfig.h8
-rw-r--r--game/enemy/EnemyPool.cpp10
-rw-r--r--game/enemy/EnemyPool.h8
-rw-r--r--game/enemy/EnemyScript.cpp206
-rw-r--r--game/enemy/EnemyScript.h37
-rw-r--r--game/enemy/EnemySubScene.cpp117
-rw-r--r--game/enemy/EnemySubScene.h10
-rw-r--r--game/hud/HudConfig.h33
-rw-r--r--game/hud/HudScript.cpp98
-rw-r--r--game/hud/HudScript.h25
-rw-r--r--game/hud/HudSubScene.cpp68
-rw-r--r--game/hud/HudSubScene.h8
-rw-r--r--game/hud/SpeedScript.cpp42
-rw-r--r--game/hud/SpeedScript.h15
-rw-r--r--game/main.cpp27
-rw-r--r--game/makefile5
-rw-r--r--game/menus/BannerSubScene.cpp49
-rw-r--r--game/menus/BannerSubScene.h20
-rw-r--r--game/menus/ButtonNextMainMenuSubScript.cpp44
-rw-r--r--game/menus/ButtonNextMainMenuSubScript.h14
-rw-r--r--game/menus/ButtonReplaySubScript.cpp43
-rw-r--r--game/menus/ButtonReplaySubScript.h21
-rw-r--r--game/menus/ButtonSetMainMenuSubScript.cpp23
-rw-r--r--game/menus/ButtonSetMainMenuSubScript.h14
-rw-r--r--game/menus/ButtonSetShopSubScript.cpp17
-rw-r--r--game/menus/ButtonSetShopSubScript.h14
-rw-r--r--game/menus/ButtonShowCreditsSubScript.cpp20
-rw-r--r--game/menus/ButtonShowCreditsSubScript.h14
-rw-r--r--game/menus/ButtonSubScene.cpp266
-rw-r--r--game/menus/ButtonSubScene.h84
-rw-r--r--game/menus/FloatingWindowSubScene.cpp220
-rw-r--r--game/menus/FloatingWindowSubScene.h17
-rw-r--r--game/menus/IButtonScript.cpp32
-rw-r--r--game/menus/IButtonScript.h10
-rw-r--r--game/menus/IFloatingWindowScript.cpp22
-rw-r--r--game/menus/IFloatingWindowScript.h15
-rw-r--r--game/menus/MenusConfig.h16
-rw-r--r--game/menus/endgame/EndGameSubScene.cpp127
-rw-r--r--game/menus/endgame/EndGameSubScene.h9
-rw-r--r--game/menus/endgame/EndGameSubScript.cpp101
-rw-r--r--game/menus/endgame/EndGameSubScript.h18
-rw-r--r--game/menus/mainmenu/ButtonTransitionPreviewSubScript.cpp23
-rw-r--r--game/menus/mainmenu/ButtonTransitionPreviewSubScript.h12
-rw-r--r--game/menus/mainmenu/CreditsSubScene.cpp132
-rw-r--r--game/menus/mainmenu/CreditsSubScene.h9
-rw-r--r--game/menus/mainmenu/CreditsSubScript.cpp58
-rw-r--r--game/menus/mainmenu/CreditsSubScript.h18
-rw-r--r--game/menus/mainmenu/ITransitionScript.cpp30
-rw-r--r--game/menus/mainmenu/ITransitionScript.h15
-rw-r--r--game/menus/mainmenu/MainMenuConfig.h19
-rw-r--r--game/menus/mainmenu/MainMenuScene.cpp137
-rw-r--r--game/menus/mainmenu/MainMenuScene.h10
-rw-r--r--game/menus/mainmenu/TransitionStartSubScript.cpp16
-rw-r--r--game/menus/mainmenu/TransitionStartSubScript.h9
-rw-r--r--game/menus/shop/ButtonBuySelectBubbleScript.cpp34
-rw-r--r--game/menus/shop/ButtonBuySelectBubbleScript.h14
-rw-r--r--game/menus/shop/ButtonBuySelectBulletScript.cpp34
-rw-r--r--game/menus/shop/ButtonBuySelectBulletScript.h14
-rw-r--r--game/menus/shop/ShopLoadScript.cpp126
-rw-r--r--game/menus/shop/ShopLoadScript.h10
-rw-r--r--game/menus/shop/ShopMenuScene.cpp326
-rw-r--r--game/menus/shop/ShopMenuScene.h12
-rw-r--r--game/menus/shop/Shopconfig.h14
-rw-r--r--game/missile/AlertScript.cpp38
-rw-r--r--game/missile/AlertScript.h11
-rw-r--r--game/missile/AlertSubScene.cpp33
-rw-r--r--game/missile/AlertSubScene.h8
-rw-r--r--game/missile/MissilePool.cpp18
-rw-r--r--game/missile/MissilePool.h11
-rw-r--r--game/missile/MissileScript.cpp106
-rw-r--r--game/missile/MissileScript.h20
-rw-r--r--game/missile/MissileSubScene.cpp101
-rw-r--r--game/missile/MissileSubScene.h12
-rw-r--r--game/missile/SpawnEvent.cpp49
-rw-r--r--game/missile/SpawnEvent.h19
-rw-r--r--game/player/PlayerAudioScript.cpp62
-rw-r--r--game/player/PlayerAudioScript.h13
-rw-r--r--game/player/PlayerBulletPool.cpp11
-rw-r--r--game/player/PlayerBulletPool.h11
-rw-r--r--game/player/PlayerBulletScript.cpp41
-rw-r--r--game/player/PlayerBulletScript.h11
-rw-r--r--game/player/PlayerBulletSubScene.cpp52
-rw-r--r--game/player/PlayerBulletSubScene.h10
-rw-r--r--game/player/PlayerEndScript.cpp104
-rw-r--r--game/player/PlayerEndScript.h14
-rw-r--r--game/player/PlayerScript.cpp214
-rw-r--r--game/player/PlayerScript.h32
-rw-r--r--game/player/PlayerSubScene.cpp222
-rw-r--r--game/player/PlayerSubScene.h10
-rw-r--r--game/prefab/CMakeLists.txt6
-rw-r--r--game/prefab/ZapperObject.cpp118
-rw-r--r--game/prefab/ZapperObject.h40
-rw-r--r--game/prefab/ZapperPoolScript.cpp70
-rw-r--r--game/prefab/ZapperPoolScript.h33
-rw-r--r--game/prefab/ZapperPoolSubScene.cpp17
-rw-r--r--game/prefab/ZapperPoolSubScene.h19
-rw-r--r--game/preview/NpcScript.cpp27
-rw-r--r--game/preview/NpcScript.h8
-rw-r--r--game/preview/NpcSubScene.cpp65
-rw-r--r--game/preview/NpcSubScene.h10
-rw-r--r--game/preview/PrevPlayerScript.cpp126
-rw-r--r--game/preview/PrevPlayerScript.h23
-rw-r--r--game/preview/PrevPlayerSubScene.cpp81
-rw-r--r--game/preview/PrevPlayerSubScene.h10
-rw-r--r--game/preview/PreviewReplaySubScript.cpp55
-rw-r--r--game/preview/PreviewReplaySubScript.h24
-rw-r--r--game/preview/PreviewStartRecSubScript.cpp20
-rw-r--r--game/preview/PreviewStartRecSubScript.h11
-rw-r--r--game/preview/PreviewStopRecSubScript.cpp20
-rw-r--r--game/preview/PreviewStopRecSubScript.h11
-rw-r--r--game/preview/SmokeSubScene.cpp37
-rw-r--r--game/preview/SmokeSubScene.h10
-rw-r--r--game/scheduler/ObjectsScheduler.cpp120
-rw-r--r--game/scheduler/ObjectsScheduler.h34
-rwxr-xr-xgame/util/scrollgen36
-rw-r--r--game/workers/CollisionScript.cpp69
-rw-r--r--game/workers/CollisionScript.h12
-rw-r--r--game/workers/PanicFromPlayerScript.cpp55
-rw-r--r--game/workers/PanicFromPlayerScript.h8
-rw-r--r--game/workers/WorkerScript.cpp145
-rw-r--r--game/workers/WorkerScript.h8
-rw-r--r--game/workers/WorkersSubScene.cpp423
-rw-r--r--game/workers/WorkersSubScene.h20
174 files changed, 8928 insertions, 0 deletions
diff --git a/game/.crepe-root b/game/.crepe-root
new file mode 100644
index 0000000..e69de29
--- /dev/null
+++ b/game/.crepe-root
diff --git a/game/.gitignore b/game/.gitignore
new file mode 100644
index 0000000..2bd69c0
--- /dev/null
+++ b/game/.gitignore
@@ -0,0 +1 @@
+asset
diff --git a/game/CMakeLists.txt b/game/CMakeLists.txt
new file mode 100644
index 0000000..a94beca
--- /dev/null
+++ b/game/CMakeLists.txt
@@ -0,0 +1,124 @@
+cmake_minimum_required(VERSION 3.28)
+
+set(CMAKE_C_STANDARD 11)
+set(CMAKE_CXX_STANDARD 20)
+set(CMAKE_EXPORT_COMPILE_COMMANDS 1)
+set(CMAKE_BUILD_TYPE Release)
+project(game C CXX)
+
+add_subdirectory(../src crepe)
+
+add_executable(main)
+
+target_sources(main PUBLIC
+ # enemy
+ enemy/BattleScript.cpp
+ enemy/EnemyPool.cpp
+ enemy/EnemyBulletScript.cpp
+ enemy/EnemyBulletSubScene.cpp
+ enemy/EnemyBulletPool.cpp
+ enemy/EnemySubScene.cpp
+ enemy/EnemyScript.cpp
+
+ #background
+ background/AquariumSubScene.cpp
+ background/AquariumScript.cpp
+ background/BackgroundSubScene.cpp
+ background/ForestParallaxScript.cpp
+ background/ForestSubScene.cpp
+ background/HallwaySubScene.cpp
+ background/StartSubScene.cpp
+ background/HallwayScript.cpp
+
+ # mainscenes
+ GameScene.cpp
+ menus/shop/ShopMenuScene.cpp
+ menus/shop/ShopLoadScript.cpp
+ menus/shop/ButtonBuySelectBubbleScript.cpp
+ menus/shop/ButtonBuySelectBulletScript.cpp
+ menus/mainmenu/MainMenuScene.cpp
+ PreviewScene.cpp
+ main.cpp
+
+ # missile
+ missile/MissilePool.cpp
+ missile/MissileScript.cpp
+ missile/MissileSubScene.cpp
+ missile/AlertSubScene.cpp
+ missile/AlertScript.cpp
+ missile/SpawnEvent.cpp
+
+ #scheduling
+ scheduler/ObjectsScheduler.cpp
+
+ # Preview
+ preview/SmokeSubScene.cpp
+ preview/NpcSubScene.cpp
+ preview/NpcScript.cpp
+ preview/PrevPlayerSubScene.cpp
+ preview/PrevPlayerScript.cpp
+ preview/PreviewStopRecSubScript.cpp
+ preview/PreviewStartRecSubScript.cpp
+ preview/PreviewReplaySubScript.cpp
+
+ # scripts
+ GameScene.cpp
+ MoveCameraManualyScript.cpp
+ StartGameScript.cpp
+ QuitScript.cpp
+
+ # player
+ player/PlayerScript.cpp
+ player/PlayerSubScene.cpp
+ player/PlayerBulletPool.cpp
+ player/PlayerBulletScript.cpp
+ player/PlayerBulletSubScene.cpp
+ player/PlayerEndScript.cpp
+ player/PlayerAudioScript.cpp
+
+ # workers
+ workers/WorkersSubScene.cpp
+ workers/WorkerScript.cpp
+ workers/PanicFromPlayerScript.cpp
+ workers/CollisionScript.cpp
+
+ # menus
+ menus/BannerSubScene.cpp
+ menus/ButtonSubScene.cpp
+ menus/IButtonScript.cpp
+ menus/ButtonSetShopSubScript.cpp
+ menus/ButtonSetMainMenuSubScript.cpp
+ menus/ButtonReplaySubScript.cpp
+ menus/ButtonNextMainMenuSubScript.cpp
+ menus/FloatingWindowSubScene.cpp
+ menus/IFloatingWindowScript.cpp
+ menus/ButtonShowCreditsSubScript.cpp
+ menus/mainmenu/ButtonTransitionPreviewSubScript.cpp
+ menus/mainmenu/ITransitionScript.cpp
+ menus/mainmenu/TransitionStartSubScript.cpp
+ menus/mainmenu/CreditsSubScene.cpp
+ menus/mainmenu/CreditsSubScript.cpp
+ menus/endgame/EndGameSubScene.cpp
+ menus/endgame/EndGameSubScript.cpp
+
+ # coins
+ coins/CoinSubScene.cpp
+ coins/CoinPoolSubScene.cpp
+ coins/CoinSystemScript.cpp
+ coins/CoinScript.cpp
+
+ # hud
+ hud/HudSubScene.cpp
+ hud/HudScript.cpp
+ hud/SpeedScript.cpp
+
+ #random
+ Random.cpp
+)
+
+add_subdirectory(background)
+add_subdirectory(prefab)
+
+target_link_libraries(main PUBLIC crepe)
+target_include_directories(main PRIVATE .)
+
diff --git a/game/Config.h b/game/Config.h
new file mode 100644
index 0000000..d2b5fc4
--- /dev/null
+++ b/game/Config.h
@@ -0,0 +1,60 @@
+#pragma once
+
+#include "types.h"
+
+static constexpr int SORT_IN_LAY_BACK_BACKGROUND = 3; // For all scenes
+static constexpr int SORT_IN_LAY_BACKGROUND = 4; // For all scenes
+static constexpr int SORT_IN_LAY_FORE_BACKGROUND = 5; // For all scenes
+static constexpr int SORT_IN_LAY_PARTICLES_BACKGROUND = 6; // For all scenes
+static constexpr int SORT_IN_LAY_COINS = 7; // Only for GameScene
+static constexpr int SORT_IN_LAY_OBSTACLES = 8; // Only for GameScene
+static constexpr int SORT_IN_LAY_WORKERS_BACK = 9; // Only for GameScene
+static constexpr int SORT_IN_LAY_PLAYER = 10; // Only for GameScene
+static constexpr int SORT_IN_LAY_WORKERS_FRONT = 12; // Only for GameScene
+static constexpr int SORT_IN_LAY_PARTICLES_FOREGROUND = 15; // Only for GameScene
+
+static constexpr int COLL_LAY_BOT_TOP = 1; // Only for GameScene
+static constexpr int COLL_LAY_BOT_LOW = 2; // Only for GameScene
+static constexpr int COLL_LAY_BOT_HIGH = 3; // Only for GameScene
+static constexpr int COLL_LAY_PLAYER = 4; // Only for GameScene
+static constexpr int COLL_LAY_WALL_FRAGS = 5; // Only for GameScene
+static constexpr int COLL_LAY_ZAPPER = 6; // Only for GameScene
+static constexpr int COLL_LAY_LASER = 7; // Only for GameScene
+static constexpr int COLL_LAY_MISSILE = 8; // Only for GameScene
+static constexpr int COLL_LAY_BULLET = 9; // Only for GameScene
+static constexpr int COLL_LAY_ENEMY = 10; // Only for GameScene
+static constexpr int COLL_LAY_PLAYER_BULLET = 11; // Only for GameScene
+
+static constexpr float GAME_HEIGHT = 800; // In game units
+static constexpr float HALLWAY_HEIGHT = 450; // In game units
+
+static constexpr float VIEWPORT_X = 1100; // In game units
+// 'GAME_HEIGHT' (below) should be replaced by '500' when game development is finished
+static constexpr float VIEWPORT_Y = 500; // In game units
+
+// Font settings
+static constexpr const char * FONT = "Jetpackia";
+static constexpr crepe::vec2 FONTOFFSET = {0, 0};
+
+// Amount of coins in game
+static constexpr const char * TOTAL_COINS_GAME = "total_coins_game";
+
+// Amount of coins in current run
+static constexpr const char * TOTAL_COINS_RUN = "total_coins_run";
+
+// Distance
+static constexpr const char * DISTANCE_GAME = "distance_game";
+static constexpr const char * DISTANCE_RUN = "distance_run";
+
+// Player config
+static constexpr const char * PLAYER_NAME = "player";
+static constexpr int PLAYER_SPEED = 7500; // In game units
+static constexpr float PLAYER_GRAVITY_SCALE = 2.2; // factor
+static constexpr float PLAYER_HELP_KICK_SCALE = 0.2; // factor
+static constexpr float PLAYER_HELP_KICK_MAX = 0.3; // factor
+
+static constexpr const char * CAMERA_NAME = "camera";
+// Jetpack particles
+static constexpr const char * JETPACK_PARTICLES = "jetpack_particles";
+
+static constexpr bool DISABLE_REPLAY = false;
diff --git a/game/EngineConfig.h b/game/EngineConfig.h
new file mode 100644
index 0000000..6a03a14
--- /dev/null
+++ b/game/EngineConfig.h
@@ -0,0 +1,19 @@
+#pragma once
+
+#include "Config.h"
+
+#include <crepe/api/Config.h>
+
+static const crepe::Config ENGINE_CONFIG {
+ .log {
+ .level = crepe::Log::Level::DEBUG,
+ },
+ .physics {
+ // this division factor is now the amount of seconds it approximately takes to naturally
+ // fall from the ceiling to floor
+ .gravity = HALLWAY_HEIGHT / 0.5,
+ },
+ .window_settings {
+ .window_title = "Jetpack joyride clone",
+ },
+};
diff --git a/game/Events.h b/game/Events.h
new file mode 100644
index 0000000..cf0be68
--- /dev/null
+++ b/game/Events.h
@@ -0,0 +1,5 @@
+#pragma once
+
+#include "api/Event.h"
+
+struct EndGameEvent : public crepe::Event {};
diff --git a/game/GameScene.cpp b/game/GameScene.cpp
new file mode 100644
index 0000000..7803c9d
--- /dev/null
+++ b/game/GameScene.cpp
@@ -0,0 +1,130 @@
+#include "GameScene.h"
+#include "Config.h"
+#include "StartGameScript.h"
+#include "coins/CoinPoolSubScene.h"
+#include "coins/CoinSystemScript.h"
+
+#include "background/BackgroundSubScene.h"
+#include "enemy/BattleScript.h"
+#include "enemy/EnemyBulletPool.h"
+#include "enemy/EnemyBulletSubScene.h"
+#include "enemy/EnemyPool.h"
+#include "enemy/EnemySubScene.h"
+#include "hud/HudScript.h"
+#include "hud/HudSubScene.h"
+#include "hud/SpeedScript.h"
+#include "menus/endgame/EndGameSubScene.h"
+#include "missile/MissilePool.h"
+#include "missile/SpawnEvent.h"
+#include "player/PlayerBulletPool.h"
+#include "player/PlayerBulletSubScene.h"
+#include "player/PlayerSubScene.h"
+#include "prefab/ZapperPoolSubScene.h"
+#include "scheduler/ObjectsScheduler.h"
+#include "workers/WorkersSubScene.h"
+
+#include <cmath>
+#include <crepe/api/AI.h>
+#include <crepe/api/Animator.h>
+#include <crepe/api/Asset.h>
+#include <crepe/api/AudioSource.h>
+#include <crepe/api/BehaviorScript.h>
+#include <crepe/api/BoxCollider.h>
+#include <crepe/api/Camera.h>
+#include <crepe/api/Color.h>
+#include <crepe/api/Event.h>
+#include <crepe/api/GameObject.h>
+#include <crepe/api/ParticleEmitter.h>
+#include <crepe/api/Rigidbody.h>
+#include <crepe/api/Script.h>
+#include <crepe/api/Sprite.h>
+#include <crepe/api/Transform.h>
+#include <crepe/types.h>
+
+using namespace crepe;
+using namespace std;
+
+void GameScene::load_scene() {
+ BackgroundSubScene background(*this);
+
+ GameObject camera = new_object(CAMERA_NAME, "camera", vec2(650, 0));
+ camera.add_component<Camera>(
+ ivec2(990, 720), vec2(VIEWPORT_X, VIEWPORT_Y),
+ Camera::Data {
+ .bg_color = Color::BLACK,
+ }
+ );
+ //camera.add_component<BehaviorScript>().set_script<MoveCameraManualyScript>();
+ camera.add_component<BehaviorScript>().set_script<CoinSystemScript>();
+ camera.add_component<BehaviorScript>().set_script<HudScript>();
+ camera.add_component<BehaviorScript>().set_script<SpeedScript>();
+ camera.add_component<BehaviorScript>().set_script<BattleScript>();
+ camera.add_component<BehaviorScript>().set_script<MissileSpawnEventHandler>();
+ camera.add_component<BehaviorScript>().set_script<ObjectsScheduler>();
+
+ camera.add_component<Rigidbody>(Rigidbody::Data {});
+ AI & enemy_path_1 = camera.add_component<AI>(400);
+ enemy_path_1.make_oval_path(100, 100, camera.transform.position, 1.5708, true);
+ AI & enemy_path_2 = camera.add_component<AI>(400);
+ enemy_path_2.make_oval_path(100, 100, {0, 0}, 1.5708, true);
+ AI & enemy_path_3 = camera.add_component<AI>(400);
+ enemy_path_3.make_oval_path(100, 100, {0, 0}, 1.5708, true);
+ // camer.add_component<AI>
+ PlayerSubScene player(*this);
+ MissilePool missile_pool(*this);
+ WorkersSubScene workers(*this);
+
+ GameObject floor = new_object("floor", "game_world", vec2(0, 325));
+ floor.add_component<Rigidbody>(Rigidbody::Data {
+ .body_type = Rigidbody::BodyType::STATIC,
+ .collision_layer = COLL_LAY_BOT_TOP,
+ });
+ floor.add_component<BoxCollider>(vec2(INFINITY, 200));
+ GameObject floor_low = new_object("floor_low", "game_world", vec2(0, 350));
+ floor_low.add_component<Rigidbody>(Rigidbody::Data {
+ .body_type = Rigidbody::BodyType::STATIC,
+ .collision_layer = COLL_LAY_BOT_LOW,
+ });
+ floor_low.add_component<BoxCollider>(vec2(INFINITY, 200));
+ GameObject floor_high = new_object("floor_high", "game_world", vec2(0, 300));
+ floor_high.add_component<Rigidbody>(Rigidbody::Data {
+ .body_type = Rigidbody::BodyType::STATIC,
+ .collision_layer = COLL_LAY_BOT_HIGH,
+ });
+ GameObject ceiling = new_object("ceiling", "game_world", vec2(0, -325));
+ ceiling.add_component<Rigidbody>(Rigidbody::Data {
+ .body_type = Rigidbody::BodyType::STATIC,
+ .collision_layer = COLL_LAY_BOT_TOP,
+ });
+ ceiling.add_component<BoxCollider>(vec2(INFINITY, 200));
+
+ ZapperPoolSubScene {*this};
+
+ GameObject start_game_script = new_object("start_game_script", "script", vec2(0, 0));
+ start_game_script.add_component<BehaviorScript>().set_script<StartGameScript>();
+
+ //create coin pool
+ CoinPoolSubScene coin_system;
+ coin_system.create_coins(*this);
+ EnemyBulletPool enemy_bullet_pool;
+ enemy_bullet_pool.create_bullets(*this);
+ PlayerBulletPool player_bullet_pool;
+ player_bullet_pool.create_bullets(*this);
+ EnemyPool enemy_pool;
+ enemy_pool.create_enemies(*this);
+ HudSubScene hud;
+ hud.create(*this);
+
+ GameObject background_music = new_object("background_music", "audio", vec2(0, 0));
+ Asset background_music_asset {"asset/music/level.ogg"};
+ background_music.add_component<AudioSource>(background_music_asset);
+
+ GameObject boom_audio = new_object("boom_audio", "audio", vec2(0, 0));
+ Asset boom_audio_asset {"asset/sfx/window_smash.ogg"};
+ boom_audio.add_component<AudioSource>(boom_audio_asset);
+
+ EndGameSubScene endgamewindow;
+ endgamewindow.create(*this);
+}
+
+string GameScene::get_name() const { return "scene1"; }
diff --git a/game/GameScene.h b/game/GameScene.h
new file mode 100644
index 0000000..16e2919
--- /dev/null
+++ b/game/GameScene.h
@@ -0,0 +1,11 @@
+#pragma once
+
+#include <crepe/api/Scene.h>
+#include <string>
+
+class GameScene : public crepe::Scene {
+public:
+ void load_scene();
+
+ std::string get_name() const;
+};
diff --git a/game/MoveCameraManualyScript.cpp b/game/MoveCameraManualyScript.cpp
new file mode 100644
index 0000000..9d75a75
--- /dev/null
+++ b/game/MoveCameraManualyScript.cpp
@@ -0,0 +1,23 @@
+#include "MoveCameraManualyScript.h"
+
+using namespace crepe;
+using namespace std;
+
+void MoveCameraManualyScript::init() {
+ subscribe<KeyPressEvent>([this](const KeyPressEvent & ev) -> bool {
+ return this->keypressed(ev);
+ });
+}
+
+bool MoveCameraManualyScript::keypressed(const KeyPressEvent & event) {
+ if (event.key == Keycode::RIGHT) {
+ Transform & cam = this->get_components_by_name<Transform>("camera").front();
+ cam.position.x += 100;
+ return true;
+ } else if (event.key == Keycode::LEFT) {
+ Transform & cam = this->get_components_by_name<Transform>("camera").front();
+ cam.position.x -= 100;
+ return true;
+ }
+ return false;
+}
diff --git a/game/MoveCameraManualyScript.h b/game/MoveCameraManualyScript.h
new file mode 100644
index 0000000..5a09055
--- /dev/null
+++ b/game/MoveCameraManualyScript.h
@@ -0,0 +1,11 @@
+#pragma once
+
+#include <crepe/api/Script.h>
+
+class MoveCameraManualyScript : public crepe::Script {
+public:
+ void init();
+
+private:
+ bool keypressed(const crepe::KeyPressEvent & event);
+};
diff --git a/game/PreviewScene.cpp b/game/PreviewScene.cpp
new file mode 100644
index 0000000..14a5560
--- /dev/null
+++ b/game/PreviewScene.cpp
@@ -0,0 +1,166 @@
+#include "PreviewScene.h"
+
+#include "Config.h"
+#include "background/BackgroundSubScene.h"
+#include "hud/HudScript.h"
+#include "hud/HudSubScene.h"
+#include "hud/SpeedScript.h"
+#include "menus/ButtonSubScene.h"
+#include "missile/MissilePool.h"
+#include "missile/SpawnEvent.h"
+#include "preview/NpcSubScene.h"
+#include "preview/PrevPlayerSubScene.h"
+#include "preview/SmokeSubScene.h"
+
+#include "missile/MissileSubScene.h"
+
+#include <cmath>
+#include <crepe/api/Animator.h>
+#include <crepe/api/Asset.h>
+#include <crepe/api/AudioSource.h>
+#include <crepe/api/BehaviorScript.h>
+#include <crepe/api/BoxCollider.h>
+#include <crepe/api/Camera.h>
+#include <crepe/api/Color.h>
+#include <crepe/api/Event.h>
+#include <crepe/api/GameObject.h>
+#include <crepe/api/ParticleEmitter.h>
+#include <crepe/api/Rigidbody.h>
+#include <crepe/api/Script.h>
+#include <crepe/api/Sprite.h>
+#include <crepe/api/Transform.h>
+
+#include <crepe/ValueBroker.h>
+#include <crepe/manager/SaveManager.h>
+#include <crepe/types.h>
+#include <iostream>
+
+using namespace crepe;
+using namespace std;
+
+void PreviewScene::load_scene() {
+
+ BackgroundSubScene background(*this);
+
+ GameObject camera = new_object("camera", "camera", vec2(650, 0));
+ camera.add_component<Camera>(
+ ivec2(990, 720), vec2(VIEWPORT_X, VIEWPORT_Y),
+ Camera::Data {
+ .bg_color = Color::RED,
+ }
+ );
+
+ camera.add_component<BehaviorScript>().set_script<MissileSpawnEventHandler>();
+ camera.add_component<BehaviorScript>().set_script<HudScript>();
+ camera.add_component<BehaviorScript>().set_script<SpeedScript>();
+
+ GameObject floor = new_object("floor", "game_world", vec2(0, 325));
+ floor.add_component<Rigidbody>(Rigidbody::Data {
+ .body_type = Rigidbody::BodyType::STATIC,
+ .collision_layer = COLL_LAY_BOT_TOP,
+ });
+ floor.add_component<BoxCollider>(vec2(INFINITY, 200));
+ GameObject floor_low = new_object("floor_low", "game_world", vec2(0, 350));
+ floor_low.add_component<Rigidbody>(Rigidbody::Data {
+ .body_type = Rigidbody::BodyType::STATIC,
+ .collision_layer = COLL_LAY_BOT_LOW,
+ });
+ floor_low.add_component<BoxCollider>(vec2(INFINITY, 200));
+ GameObject floor_high = new_object("floor_high", "game_world", vec2(0, 300));
+ floor_high.add_component<Rigidbody>(Rigidbody::Data {
+ .body_type = Rigidbody::BodyType::STATIC,
+ .collision_layer = COLL_LAY_BOT_HIGH,
+ });
+ GameObject ceiling = new_object("ceiling", "game_world", vec2(0, -325));
+ ceiling.add_component<Rigidbody>(Rigidbody::Data {
+ .body_type = Rigidbody::BodyType::STATIC,
+ .collision_layer = COLL_LAY_BOT_TOP,
+ });
+ ceiling.add_component<BoxCollider>(vec2(INFINITY, 200));
+
+ GameObject world = this->new_object("world", "TAG", vec2 {0, 0}, 0, 1);
+ world.add_component<Rigidbody>(Rigidbody::Data {
+ .body_type = Rigidbody::BodyType::STATIC,
+ .collision_layer = 100,
+ });
+
+ world.add_component<BoxCollider>(vec2(100, INFINITY), vec2(VIEWPORT_X, VIEWPORT_Y));
+ world.add_component<BoxCollider>(vec2(100, INFINITY), vec2(100, VIEWPORT_Y));
+
+ PrevPlayerSubScene player(*this);
+ NpcSubScene npc(*this);
+ SmokeSubScene smoke(*this);
+ MissilePool mpool(*this);
+
+ HudSubScene hud;
+ hud.create(*this);
+
+ const float Y_POS_BUTTONS = -220;
+ const float X_POS_BUTTONS = -150;
+ const float X_POS_BUTTONS_SPACING = 145;
+ ButtonSubScene button;
+ button.create(
+ *this,
+ ButtonSubScene::Data {
+ .text = "BACK",
+ .text_width = 60,
+ .position = {X_POS_BUTTONS, Y_POS_BUTTONS},
+ .script_type = ButtonSubScene::ScriptSelect::NEXT,
+ .button_type = ButtonSubScene::ButtonSelect::BACK,
+ .scale = 0.6,
+ .worldspace = false,
+ .tag = "Next button",
+ .sorting_layer_offset = 20,
+ }
+ );
+
+ button.create(
+ *this,
+ ButtonSubScene::Data {
+ .text = "START REC",
+ .text_width = 130,
+ .position = {X_POS_BUTTONS + X_POS_BUTTONS_SPACING, Y_POS_BUTTONS},
+ .script_type = ButtonSubScene::ScriptSelect::PREVIEW_START,
+ .button_type = ButtonSubScene::ButtonSelect::LARGE,
+ .scale = 0.6,
+ .worldspace = false,
+ .tag = "Next button",
+ .sorting_layer_offset = 20,
+ .btn_side_color = ButtonSubScene::ButtonSideColor::YELLOW,
+ }
+ );
+
+ button.create(
+ *this,
+ ButtonSubScene::Data {
+ .text = "STOP REC",
+ .text_width = 120,
+ .position = {X_POS_BUTTONS + X_POS_BUTTONS_SPACING * 2, Y_POS_BUTTONS},
+ .script_type = ButtonSubScene::ScriptSelect::PREVIEW_STOP,
+ .button_type = ButtonSubScene::ButtonSelect::LARGE,
+ .scale = 0.6,
+ .worldspace = false,
+ .tag = "Next button",
+ .sorting_layer_offset = 20,
+ .btn_side_color = ButtonSubScene::ButtonSideColor::BLUE,
+ }
+ );
+
+ button.create(
+ *this,
+ ButtonSubScene::Data {
+ .text = "REPLAY",
+ .text_width = 90,
+ .position = {X_POS_BUTTONS + X_POS_BUTTONS_SPACING * 3, Y_POS_BUTTONS},
+ .script_type = ButtonSubScene::ScriptSelect::PREVIEW_REPLAY,
+ .button_type = ButtonSubScene::ButtonSelect::LARGE,
+ .scale = 0.6,
+ .worldspace = false,
+ .tag = "Next button",
+ .sorting_layer_offset = 20,
+ .btn_side_color = ButtonSubScene::ButtonSideColor::ORANGE,
+ }
+ );
+}
+
+string PreviewScene::get_name() const { return "preview scene"; }
diff --git a/game/PreviewScene.h b/game/PreviewScene.h
new file mode 100644
index 0000000..afe911e
--- /dev/null
+++ b/game/PreviewScene.h
@@ -0,0 +1,11 @@
+#pragma once
+
+#include <crepe/api/Scene.h>
+#include <string>
+
+class PreviewScene : public crepe::Scene {
+public:
+ void load_scene();
+
+ std::string get_name() const;
+};
diff --git a/game/QuitScript.cpp b/game/QuitScript.cpp
new file mode 100644
index 0000000..0c9f55a
--- /dev/null
+++ b/game/QuitScript.cpp
@@ -0,0 +1,21 @@
+
+
+#include "QuitScript.h"
+
+#include <crepe/api/Event.h>
+#include <crepe/api/KeyCodes.h>
+
+using namespace crepe;
+
+bool QuitScript::on_event(const KeyPressEvent & ev) {
+ if (Keycode::ESCAPE == ev.key) {
+ trigger_event<ShutDownEvent>(ShutDownEvent {});
+ }
+ return false;
+}
+
+void QuitScript::init() {
+ subscribe<KeyPressEvent>([this](const KeyPressEvent & ev) -> bool {
+ return this->on_event(ev);
+ });
+}
diff --git a/game/QuitScript.h b/game/QuitScript.h
new file mode 100644
index 0000000..b79a744
--- /dev/null
+++ b/game/QuitScript.h
@@ -0,0 +1,11 @@
+#pragma once
+
+#include <crepe/api/Script.h>
+
+class QuitScript : public crepe::Script {
+private:
+ bool on_event(const crepe::KeyPressEvent & ev);
+
+public:
+ void init();
+};
diff --git a/game/Random.cpp b/game/Random.cpp
new file mode 100644
index 0000000..64cf1f3
--- /dev/null
+++ b/game/Random.cpp
@@ -0,0 +1,29 @@
+#include <cstdlib>
+
+#include "Random.h"
+
+float Random::f(float upper, float lower) {
+ float range = upper - lower;
+ float x = ((float) rand() / (float) (RAND_MAX)) * range;
+ return x + lower;
+}
+
+double Random::d(double upper, double lower) {
+ double range = upper - lower;
+ double x = ((double) rand() / (double) (RAND_MAX)) * range;
+ return x + lower;
+}
+
+int Random::i(int upper, int lower) {
+ int range = upper - lower;
+ int x = rand() % range;
+ return x + lower;
+}
+
+unsigned Random::u(unsigned upper, unsigned lower) {
+ unsigned range = upper - lower;
+ unsigned x = rand() % range;
+ return x + lower;
+}
+
+bool Random::b() { return rand() % 2; }
diff --git a/game/Random.h b/game/Random.h
new file mode 100644
index 0000000..94f98d2
--- /dev/null
+++ b/game/Random.h
@@ -0,0 +1,10 @@
+#pragma once
+
+class Random {
+public:
+ static float f(float upper = 1.0, float lower = 0.0);
+ static double d(double upper = 1.0, double lower = 0.0);
+ static int i(int upper, int lower = 0);
+ static unsigned u(unsigned upper, unsigned lower = 0);
+ static bool b();
+};
diff --git a/game/StartGameScript.cpp b/game/StartGameScript.cpp
new file mode 100644
index 0000000..6d47e65
--- /dev/null
+++ b/game/StartGameScript.cpp
@@ -0,0 +1,75 @@
+#include "StartGameScript.h"
+#include "Config.h"
+#include "api/BehaviorScript.h"
+
+#include <crepe/api/Animator.h>
+#include <crepe/api/AudioSource.h>
+#include <crepe/api/ParticleEmitter.h>
+#include <crepe/api/Sprite.h>
+
+using namespace crepe;
+using namespace std;
+
+void StartGameScript::fixed_update(crepe::duration_t dt) {
+ Transform & player_transform = this->get_components_by_name<Transform>("player").front();
+ // Create hole in wall and activate panic lamp
+ if (player_transform.position.x > 75 && !this->created_hole) {
+ Sprite & lamp_sprite = this->get_components_by_name<Sprite>("start_end").back();
+ lamp_sprite.active = true;
+ Sprite & hole_sprite = this->get_components_by_name<Sprite>("start_hole").front();
+ hole_sprite.active = true;
+
+ RefVector<Rigidbody> frags_rg
+ = this->get_components_by_tag<Rigidbody>("wall_fragment");
+ RefVector<Sprite> frags_sprite = this->get_components_by_tag<Sprite>("wall_fragment");
+ for (Rigidbody & frag_rg : frags_rg) {
+ frag_rg.active = true;
+ }
+ for (Sprite & frag_sprite : frags_sprite) {
+ frag_sprite.active = true;
+ }
+
+ RefVector<ParticleEmitter> smoke_emitters
+ = this->get_components_by_name<ParticleEmitter>("smoke_particles");
+ for (ParticleEmitter & emitter : smoke_emitters) {
+ emitter.active = true;
+ }
+
+ AudioSource & boom_audio
+ = this->get_components_by_name<AudioSource>("boom_audio").front();
+ boom_audio.play();
+
+ BehaviorScript & player_audio_script
+ = this->get_components_by_name<BehaviorScript>("player_audio").front();
+ player_audio_script.active = true;
+
+ this->created_hole = true;
+ }
+
+ // Take jetpack from jetpack stand
+ if (player_transform.position.x > 275 && !this->took_jetpack) {
+ Animator & jetpack_stand_anim
+ = this->get_components_by_name<Animator>("start_begin").back();
+ jetpack_stand_anim.next_anim();
+ Sprite & jetpack_sprite = this->get_components_by_name<Sprite>("player").back();
+ jetpack_sprite.active = true;
+
+ AudioSource & background_music
+ = this->get_components_by_name<AudioSource>("background_music").front();
+ background_music.play(true);
+
+ this->took_jetpack = true;
+ }
+
+ // Start camera movement, enable player jumping and disable this script
+ if (player_transform.position.x > 500) {
+ Rigidbody & rb = this->get_components_by_name<Rigidbody>("camera").front();
+ rb.data.linear_velocity = vec2(PLAYER_SPEED * 0.02, 0);
+ BehaviorScript & player_script
+ = this->get_components_by_name<BehaviorScript>("player").front();
+ player_script.active = true;
+ BehaviorScript & this_script
+ = this->get_components_by_name<BehaviorScript>("start_game_script").front();
+ this_script.active = false;
+ }
+}
diff --git a/game/StartGameScript.h b/game/StartGameScript.h
new file mode 100644
index 0000000..ad62e1a
--- /dev/null
+++ b/game/StartGameScript.h
@@ -0,0 +1,12 @@
+#pragma once
+
+#include <crepe/api/Script.h>
+
+class StartGameScript : public crepe::Script {
+public:
+ void fixed_update(crepe::duration_t dt);
+
+private:
+ bool created_hole = false;
+ bool took_jetpack = false;
+};
diff --git a/game/background/AquariumScript.cpp b/game/background/AquariumScript.cpp
new file mode 100644
index 0000000..e698e3a
--- /dev/null
+++ b/game/background/AquariumScript.cpp
@@ -0,0 +1,26 @@
+#include "AquariumScript.h"
+
+#include "../Config.h"
+
+#include <crepe/api/Animator.h>
+#include <crepe/api/Sprite.h>
+#include <crepe/api/Transform.h>
+#include <crepe/types.h>
+
+using namespace crepe;
+using namespace std;
+
+void AquariumScript::fixed_update(crepe::duration_t dt) {
+ Transform & trans_cam = this->get_components_by_name<Transform>("camera").front();
+
+ float cam_left_x = trans_cam.position.x - VIEWPORT_X / 2;
+
+ if (cam_left_x > this->start_x + this->lenght) {
+ //Move whole background 12000 to the right
+ RefVector<Transform> trans = this->get_components_by_tag<Transform>("background_aqua");
+ for (Transform & tran : trans) {
+ tran.position.x += 12000;
+ }
+ this->start_x += 12000;
+ }
+}
diff --git a/game/background/AquariumScript.h b/game/background/AquariumScript.h
new file mode 100644
index 0000000..b068628
--- /dev/null
+++ b/game/background/AquariumScript.h
@@ -0,0 +1,12 @@
+#pragma once
+
+#include <crepe/api/Script.h>
+
+class AquariumScript : public crepe::Script {
+public:
+ void fixed_update(crepe::duration_t dt);
+
+private:
+ float start_x = 10200;
+ const float lenght = 3000;
+};
diff --git a/game/background/AquariumSubScene.cpp b/game/background/AquariumSubScene.cpp
new file mode 100644
index 0000000..2a07daf
--- /dev/null
+++ b/game/background/AquariumSubScene.cpp
@@ -0,0 +1,225 @@
+#include "AquariumSubScene.h"
+
+#include "../Config.h"
+
+#include <crepe/api/Animator.h>
+#include <crepe/api/GameObject.h>
+#include <crepe/api/ParticleEmitter.h>
+#include <crepe/api/Scene.h>
+#include <crepe/api/Sprite.h>
+#include <crepe/types.h>
+
+using namespace crepe;
+using namespace std;
+
+float AquariumSubScene::create(Scene & scn, float begin_x) {
+ this->add_background(scn, begin_x);
+
+ GameObject aquarium_begin
+ = scn.new_object("aquarium_begin", "background_aqua", vec2(begin_x, 0));
+ Asset aquarium_begin_asset {"asset/background/aquarium/glassTubeFG_1_TVOS.png"};
+ aquarium_begin.add_component<Sprite>(
+ aquarium_begin_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_BACKGROUND,
+ .order_in_layer = 0,
+ .size = vec2(0, GAME_HEIGHT),
+ }
+ );
+ begin_x += 600;
+
+ GameObject aquarium_middle_1
+ = scn.new_object("aquarium_middle", "background_aqua", vec2(begin_x, 0));
+ Asset aquarium_middle_1_asset {"asset/background/aquarium/glassTubeFG_3_TVOS.png"};
+ aquarium_middle_1.add_component<Sprite>(
+ aquarium_middle_1_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_BACKGROUND,
+ .order_in_layer = 2,
+ .size = vec2(0, GAME_HEIGHT),
+ }
+ );
+ begin_x += 400;
+
+ this->add_background(scn, begin_x - 200);
+ this->add_bubbles(aquarium_middle_1, vec2(-400, 300), 2, 0.7f);
+ this->add_bubbles(aquarium_middle_1, vec2(-100, 300), 4, 1.0f);
+ this->add_bubbles(aquarium_middle_1, vec2(500, 300), 4, 0.9f);
+
+ GameObject aquarium_middle_2
+ = scn.new_object("aquarium_middle", "background_aqua", vec2(begin_x, 0));
+ Asset aquarium_middle_2_asset {"asset/background/aquarium/glassTubeFG_3_TVOS.png"};
+ aquarium_middle_2.add_component<Sprite>(
+ aquarium_middle_2_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_BACKGROUND,
+ .order_in_layer = 3,
+ .size = vec2(0, GAME_HEIGHT),
+ }
+ );
+ begin_x += 400;
+
+ this->add_bubbles(aquarium_middle_2, vec2(300, 300), 2, 0.6f);
+
+ GameObject aquarium_middle_3
+ = scn.new_object("aquarium_middle", "background_aqua", vec2(begin_x, 0));
+ Asset aquarium_middle_3_asset {"asset/background/aquarium/glassTubeFG_3_TVOS.png"};
+ aquarium_middle_3.add_component<Sprite>(
+ aquarium_middle_3_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_BACKGROUND,
+ .order_in_layer = 4,
+ .size = vec2(0, GAME_HEIGHT),
+ }
+ );
+ begin_x += 400;
+
+ this->add_background(scn, begin_x - 200);
+
+ GameObject aquarium_middle_4
+ = scn.new_object("aquarium_middle", "background_aqua", vec2(begin_x, 0));
+ Asset aquarium_middle_4_asset {"asset/background/aquarium/glassTubeFG_3_TVOS.png"};
+ aquarium_middle_4.add_component<Sprite>(
+ aquarium_middle_4_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_BACKGROUND,
+ .order_in_layer = 5,
+ .size = vec2(0, GAME_HEIGHT),
+ }
+ );
+ begin_x += 600;
+
+ this->add_background(scn, begin_x);
+ this->add_bubbles(aquarium_middle_4, vec2(175, 300), 4, 1.0f);
+ this->add_bubbles(aquarium_middle_4, vec2(200, 300), 4, 0.7f);
+
+ GameObject aquarium_end
+ = scn.new_object("aquarium_end", "background_aqua", vec2(begin_x, 0));
+ Asset aquarium_end_asset {"asset/background/aquarium/glassTubeFG_2_TVOS.png"};
+ aquarium_end.add_component<Sprite>(
+ aquarium_end_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_BACKGROUND,
+ .order_in_layer = 1,
+ .size = vec2(0, GAME_HEIGHT),
+ }
+ );
+ begin_x += 600;
+
+ return begin_x;
+}
+
+void AquariumSubScene::add_background(Scene & scn, float begin_x) {
+ GameObject bg_1 = scn.new_object("aquarium_bg_1", "background_aqua", vec2(begin_x, 0));
+ Asset bg_1_1_asset {"asset/background/aquarium/AquariumBG1_1_TVOS.png"};
+ bg_1.add_component<Sprite>(
+ bg_1_1_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_BACK_BACKGROUND,
+ .order_in_layer = 5,
+ .size = vec2(0, 400),
+ .position_offset = vec2(-200, 100),
+ }
+ );
+ Asset bg_1_2_asset {"asset/background/aquarium/AquariumBG1_2_TVOS.png"};
+ bg_1.add_component<Sprite>(
+ bg_1_2_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_BACK_BACKGROUND,
+ .order_in_layer = 5,
+ .size = vec2(0, 400),
+ .position_offset = vec2(200, 100),
+ }
+ );
+ GameObject bg_2 = scn.new_object("aquarium_bg_2", "background_aqua", vec2(begin_x, 0));
+ Asset bg_2_1_asset {"asset/background/aquarium/AquariumBG2_1_TVOS.png"};
+ bg_2.add_component<Sprite>(
+ bg_2_1_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_BACK_BACKGROUND,
+ .order_in_layer = 3,
+ .size = vec2(0, 400),
+ .position_offset = vec2(200, -50),
+ }
+ );
+ Asset bg_2_2_asset {"asset/background/aquarium/AquariumBG2_2_TVOS.png"};
+ bg_2.add_component<Sprite>(
+ bg_2_2_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_BACK_BACKGROUND,
+ .order_in_layer = 3,
+ .size = vec2(0, 400),
+ .position_offset = vec2(-200, -50),
+ }
+ );
+ GameObject bg_3 = scn.new_object("aquarium_bg_3", "background_aqua", vec2(begin_x, 0));
+ Asset bg_3_1_asset {"asset/background/aquarium/AquariumBG3_1_TVOS.png"};
+ bg_3.add_component<Sprite>(
+ bg_3_1_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_BACK_BACKGROUND,
+ .order_in_layer = 1,
+ .size = vec2(0, 400),
+ .position_offset = vec2(200, -200),
+ }
+ );
+ Asset bg_3_2_asset {"asset/background/aquarium/AquariumBG3_2_TVOS.png"};
+ bg_3.add_component<Sprite>(
+ bg_3_2_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_BACK_BACKGROUND,
+ .order_in_layer = 1,
+ .size = vec2(0, 400),
+ .position_offset = vec2(-200, -200),
+ }
+ );
+}
+
+void AquariumSubScene::add_bubbles(
+ GameObject & obj, vec2 offset, int order_in_layer, float scale
+) {
+ Sprite & sprite = obj.add_component<Sprite>(
+ Asset {"asset/background/aquarium/bubble.png"},
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_BACK_BACKGROUND,
+ .order_in_layer = order_in_layer,
+ .size = vec2(0, 12.5),
+ .scale_offset = scale,
+ }
+ );
+ obj.add_component<ParticleEmitter>(
+ sprite,
+ ParticleEmitter::Data {
+ .offset = offset,
+ .max_particles = 20,
+ .emission_rate = 1.2,
+ .min_speed = 50,
+ .max_speed = 100,
+ .min_angle = 265,
+ .max_angle = 275,
+ .force_over_time = vec2(0, -50),
+ }
+ );
+ Sprite & sprite_small = obj.add_component<Sprite>(
+ Asset {"asset/background/aquarium/bubble.png"},
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_BACK_BACKGROUND,
+ .order_in_layer = order_in_layer,
+ .size = vec2(0, 7.5),
+ .scale_offset = scale,
+ }
+ );
+ obj.add_component<ParticleEmitter>(
+ sprite_small,
+ ParticleEmitter::Data {
+ .offset = offset,
+ .max_particles = 20,
+ .emission_rate = 0.8,
+ .min_speed = 50,
+ .max_speed = 100,
+ .min_angle = 265,
+ .max_angle = 275,
+ .force_over_time = vec2(0, -50),
+ }
+ );
+}
diff --git a/game/background/AquariumSubScene.h b/game/background/AquariumSubScene.h
new file mode 100644
index 0000000..9dbb04e
--- /dev/null
+++ b/game/background/AquariumSubScene.h
@@ -0,0 +1,18 @@
+#pragma once
+
+#include <crepe/types.h>
+
+namespace crepe {
+class Scene;
+class GameObject;
+} // namespace crepe
+
+class AquariumSubScene {
+public:
+ float create(crepe::Scene & scn, float begin_x);
+
+private:
+ void add_background(crepe::Scene & scn, float begin_x);
+ void
+ add_bubbles(crepe::GameObject & obj, crepe::vec2 offset, int order_in_layer, float scale);
+};
diff --git a/game/background/BackgroundSubScene.cpp b/game/background/BackgroundSubScene.cpp
new file mode 100644
index 0000000..4bbd977
--- /dev/null
+++ b/game/background/BackgroundSubScene.cpp
@@ -0,0 +1,37 @@
+#include "BackgroundSubScene.h"
+#include "AquariumScript.h"
+#include "AquariumSubScene.h"
+#include "ForestSubScene.h"
+#include "HallwayScript.h"
+#include "HallwaySubScene.h"
+#include "StartSubScene.h"
+
+#include <crepe/api/BehaviorScript.h>
+#include <crepe/api/Color.h>
+#include <crepe/api/Scene.h>
+
+using namespace crepe;
+using namespace std;
+
+BackgroundSubScene::BackgroundSubScene(Scene & scn) {
+ StartSubScene start;
+ HallwaySubScene hallway;
+ ForestSubScene forest;
+ AquariumSubScene aquarium;
+
+ float begin_x = 400;
+
+ begin_x = start.create(scn, begin_x);
+
+ begin_x = hallway.create(scn, begin_x, 1, Color::YELLOW);
+
+ begin_x = forest.create(scn, begin_x, "1");
+
+ begin_x += 3000;
+
+ begin_x = aquarium.create(scn, begin_x);
+
+ GameObject scripts = scn.new_object("scrips_background", "background");
+ scripts.add_component<BehaviorScript>().set_script<HallwayScript>();
+ scripts.add_component<BehaviorScript>().set_script<AquariumScript>();
+}
diff --git a/game/background/BackgroundSubScene.h b/game/background/BackgroundSubScene.h
new file mode 100644
index 0000000..06bdac4
--- /dev/null
+++ b/game/background/BackgroundSubScene.h
@@ -0,0 +1,10 @@
+#pragma once
+
+namespace crepe {
+class Scene;
+}
+
+class BackgroundSubScene {
+public:
+ BackgroundSubScene(crepe::Scene & scn);
+};
diff --git a/game/background/CMakeLists.txt b/game/background/CMakeLists.txt
new file mode 100644
index 0000000..1d705f5
--- /dev/null
+++ b/game/background/CMakeLists.txt
@@ -0,0 +1,9 @@
+target_sources(main PUBLIC
+ AquariumSubScene.cpp
+ BackgroundSubScene.cpp
+ ForestParallaxScript.cpp
+ ForestSubScene.cpp
+ HallwaySubScene.cpp
+ StartSubScene.cpp
+)
+
diff --git a/game/background/ForestParallaxScript.cpp b/game/background/ForestParallaxScript.cpp
new file mode 100644
index 0000000..7470da2
--- /dev/null
+++ b/game/background/ForestParallaxScript.cpp
@@ -0,0 +1,55 @@
+#include "ForestParallaxScript.h"
+
+#include "../Config.h"
+
+using namespace crepe;
+using namespace std;
+
+ForestParallaxScript::ForestParallaxScript(
+ float begin_x, float end_x, std::string unique_bg_name
+)
+ : begin_x(begin_x),
+ end_x(end_x),
+ name(unique_bg_name) {}
+
+void ForestParallaxScript::fixed_update(crepe::duration_t dt) {
+ RefVector<Transform> vec_2
+ = this->get_components_by_name<Transform>("forest_bg_2_" + name);
+ RefVector<Transform> vec_3
+ = this->get_components_by_name<Transform>("forest_bg_3_" + name);
+
+ for (Transform & t : vec_2) {
+ if (t.position.x > end_x - 400) {
+ t.position.x = begin_x - 400;
+ }
+ }
+ for (Transform & t : vec_3) {
+ if (t.position.x > end_x - 400) {
+ t.position.x = begin_x - 400;
+ }
+ }
+
+ //Move whole background 12000 to the right
+ Transform & trans_cam = this->get_components_by_name<Transform>("camera").front();
+
+ float cam_left_x = trans_cam.position.x - VIEWPORT_X / 2;
+
+ if (cam_left_x > this->start_x + this->lenght) {
+ //Move whole background 12000 to the right
+ RefVector<Transform> trans
+ = this->get_components_by_tag<Transform>("background_forest");
+ for (Transform & tran : trans) {
+ tran.position.x += 12000;
+ }
+ this->start_x += 12000;
+
+ RefVector<Transform> trans_back
+ = this->get_components_by_tag<Transform>("forest_background");
+ for (Transform & tran : trans_back) {
+ tran.position.x += 12000;
+ }
+
+ begin_x += 12000;
+ end_x += 12000;
+ }
+}
diff --git a/game/background/ForestParallaxScript.h b/game/background/ForestParallaxScript.h
new file mode 100644
index 0000000..d45fdd9
--- /dev/null
+++ b/game/background/ForestParallaxScript.h
@@ -0,0 +1,17 @@
+#pragma once
+
+#include <crepe/api/Script.h>
+
+class ForestParallaxScript : public crepe::Script {
+public:
+ ForestParallaxScript(float begin_x, float end_x, std::string unique_bg_name);
+
+ void fixed_update(crepe::duration_t dt);
+
+private:
+ float begin_x;
+ float end_x;
+ const std::string name;
+ float start_x = 4200;
+ const float lenght = 3000;
+};
diff --git a/game/background/ForestSubScene.cpp b/game/background/ForestSubScene.cpp
new file mode 100644
index 0000000..83e48dd
--- /dev/null
+++ b/game/background/ForestSubScene.cpp
@@ -0,0 +1,169 @@
+#include "ForestSubScene.h"
+#include "ForestParallaxScript.h"
+
+#include "../Config.h"
+
+#include <crepe/api/Animator.h>
+#include <crepe/api/BehaviorScript.h>
+#include <crepe/api/GameObject.h>
+#include <crepe/api/Scene.h>
+#include <crepe/api/Script.h>
+#include <crepe/api/Sprite.h>
+#include <crepe/types.h>
+
+using namespace crepe;
+using namespace std;
+
+float ForestSubScene::create(Scene & scn, float begin_x, std::string unique_bg_name) {
+ GameObject script = scn.new_object("forest_script", "background_forest");
+ script.add_component<BehaviorScript>().set_script<ForestParallaxScript>(
+ begin_x - 400, begin_x + 3000 + 400, unique_bg_name
+ );
+
+ this->add_background(scn, begin_x, unique_bg_name);
+
+ GameObject begin = scn.new_object("forest_begin", "background_forest", vec2(begin_x, 0));
+ Asset begin_asset {"asset/background/forest/forestFG_1_TVOS.png"};
+ begin.add_component<Sprite>(
+ begin_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_BACKGROUND,
+ .order_in_layer = 0,
+ .size = vec2(0, GAME_HEIGHT),
+ }
+ );
+ begin_x += 800;
+
+ this->add_background(scn, begin_x, unique_bg_name);
+
+ GameObject middle_1
+ = scn.new_object("forest_middle", "background_forest", vec2(begin_x, 0));
+ Asset middle_1_asset {"asset/background/forest/forestFG_3_TVOS.png"};
+ middle_1.add_component<Sprite>(
+ middle_1_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_BACKGROUND,
+ .order_in_layer = 2,
+ .size = vec2(0, GAME_HEIGHT),
+ }
+ );
+ begin_x += 800;
+
+ this->add_background(scn, begin_x, unique_bg_name);
+
+ GameObject middle_2
+ = scn.new_object("forest_middle", "background_forest", vec2(begin_x, 0));
+ Asset middle_2_asset {"asset/background/forest/forestFG_3_TVOS.png"};
+ middle_2.add_component<Sprite>(
+ middle_2_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_BACKGROUND,
+ .order_in_layer = 3,
+ .size = vec2(0, GAME_HEIGHT),
+ }
+ );
+ begin_x += 800;
+
+ this->add_background(scn, begin_x, unique_bg_name);
+
+ GameObject end = scn.new_object("forest_end", "background_forest", vec2(begin_x, 0));
+ Asset end_asset {"asset/background/forest/forestFG_2_TVOS.png"};
+ end.add_component<Sprite>(
+ end_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_BACKGROUND,
+ .order_in_layer = 1,
+ .size = vec2(0, GAME_HEIGHT),
+ }
+ );
+ begin_x += 600;
+
+ this->add_background(scn, begin_x + 200, unique_bg_name);
+
+ return begin_x;
+}
+
+void ForestSubScene::add_background(Scene & scn, float begin_x, std::string name) {
+ GameObject bg_1
+ = scn.new_object("forest_bg_1_" + name, "forest_background", vec2(begin_x, 0));
+ Asset bg_1_asset {"asset/background/forest/forestBG1_1_TVOS.png"};
+ bg_1.add_component<Sprite>(
+ bg_1_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_BACK_BACKGROUND,
+ .order_in_layer = 2,
+ .size = vec2(0, 800),
+ }
+ );
+ GameObject bg_2
+ = scn.new_object("forest_bg_2_" + name, "forest_background", vec2(begin_x, 0));
+ Asset bg_2_1_asset {"asset/background/forest/forestBG2_1_TVOS.png"};
+ bg_2.add_component<Sprite>(
+ bg_2_1_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_BACK_BACKGROUND,
+ .order_in_layer = 1,
+ .size = vec2(0, 400),
+ .position_offset = vec2(200, 0),
+ }
+ );
+ Asset bg_2_2_asset {"asset/background/forest/forestBG2_2_TVOS.png"};
+ bg_2.add_component<Sprite>(
+ bg_2_2_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_BACK_BACKGROUND,
+ .order_in_layer = 1,
+ .size = vec2(0, 400),
+ .position_offset = vec2(-200, 0),
+ }
+ );
+ GameObject bg_3
+ = scn.new_object("forest_bg_3_" + name, "forest_background", vec2(begin_x, 0));
+ Asset bg_3_1_asset {"asset/background/forest/forestBG3_1_TVOS.png"};
+ bg_3.add_component<Sprite>(
+ bg_3_1_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_BACK_BACKGROUND,
+ .order_in_layer = 0,
+ .size = vec2(0, 200),
+ .position_offset = vec2(300, 0),
+ }
+ );
+ Asset bg_3_2_asset {"asset/background/forest/forestBG3_2_TVOS.png"};
+ bg_3.add_component<Sprite>(
+ bg_3_2_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_BACK_BACKGROUND,
+ .order_in_layer = 0,
+ .size = vec2(0, 200),
+ .position_offset = vec2(100, 0),
+ }
+ );
+ Asset bg_3_3_asset {"asset/background/forest/forestBG3_3_TVOS.png"};
+ bg_3.add_component<Sprite>(
+ bg_3_3_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_BACK_BACKGROUND,
+ .order_in_layer = 0,
+ .size = vec2(0, 200),
+ .position_offset = vec2(-100, 0),
+ }
+ );
+ Asset bg_3_4_asset {"asset/background/forest/forestBG3_4_TVOS.png"};
+ bg_3.add_component<Sprite>(
+ bg_3_4_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_BACK_BACKGROUND,
+ .order_in_layer = 0,
+ .size = vec2(0, 200),
+ .position_offset = vec2(-300, 0),
+ }
+ );
+
+ bg_2.add_component<Rigidbody>(Rigidbody::Data {
+ .linear_velocity = vec2(30, 0),
+ });
+ bg_3.add_component<Rigidbody>(Rigidbody::Data {
+ .linear_velocity = vec2(40, 0),
+ });
+}
diff --git a/game/background/ForestSubScene.h b/game/background/ForestSubScene.h
new file mode 100644
index 0000000..0a04001
--- /dev/null
+++ b/game/background/ForestSubScene.h
@@ -0,0 +1,15 @@
+#pragma once
+
+#include <string>
+
+namespace crepe {
+class Scene;
+}
+
+class ForestSubScene {
+public:
+ float create(crepe::Scene & scn, float begin_x, std::string unique_bg_name);
+
+private:
+ void add_background(crepe::Scene & scn, float begin_x, std::string name);
+};
diff --git a/game/background/HallwayScript.cpp b/game/background/HallwayScript.cpp
new file mode 100644
index 0000000..a5bb94c
--- /dev/null
+++ b/game/background/HallwayScript.cpp
@@ -0,0 +1,70 @@
+#include "HallwayScript.h"
+
+#include "../Config.h"
+
+#include <crepe/api/Animator.h>
+#include <crepe/api/Sprite.h>
+#include <crepe/api/Transform.h>
+#include <crepe/types.h>
+
+using namespace crepe;
+using namespace std;
+
+void HallwayScript::fixed_update(crepe::duration_t dt) {
+ Transform & trans_cam = this->get_components_by_name<Transform>("camera").front();
+
+ float cam_left_x = trans_cam.position.x - VIEWPORT_X / 2;
+
+ if (cam_left_x > this->start_x + this->lenght) {
+ //Move whole background 6000 to the right
+ RefVector<Transform> trans = this->get_components_by_tag<Transform>("background_hall");
+ for (Transform & tran : trans) {
+ tran.position.x += 6000;
+ }
+ this->start_x += 6000;
+
+ //Change sector number
+ Animator & anim = this->get_components_by_name<Animator>("hallway_begin").front();
+ int column = (current_sector - 1) / 4;
+ int row = (current_sector - 1) % 4;
+ anim.set_anim(column);
+ for (int i = 0; i < row; i++) {
+ anim.next_anim();
+ }
+ RefVector<Sprite> sprites = this->get_components_by_name<Sprite>("hallway_begin");
+ switch (current_sector % 7) {
+ case 0:
+ sprites[1].get().data.color = Color::YELLOW;
+ sprites[2].get().data.color = Color::YELLOW;
+ break;
+ case 1:
+ sprites[1].get().data.color = Color::MAGENTA;
+ sprites[2].get().data.color = Color::MAGENTA;
+ break;
+ case 2:
+ sprites[1].get().data.color = Color::CYAN;
+ sprites[2].get().data.color = Color::CYAN;
+ break;
+ case 3:
+ sprites[1].get().data.color = Color::GREEN;
+ sprites[2].get().data.color = Color::GREEN;
+ break;
+ case 4:
+ sprites[1].get().data.color = Color::RED;
+ sprites[2].get().data.color = Color::RED;
+ break;
+ case 5:
+ sprites[1].get().data.color = Color::BLUE;
+ sprites[2].get().data.color = Color::BLUE;
+ break;
+ case 6:
+ sprites[1].get().data.color = Color::WHITE;
+ sprites[2].get().data.color = Color::WHITE;
+ break;
+ }
+ current_sector++;
+ if (current_sector > 16) {
+ current_sector = 1;
+ }
+ }
+}
diff --git a/game/background/HallwayScript.h b/game/background/HallwayScript.h
new file mode 100644
index 0000000..04b2933
--- /dev/null
+++ b/game/background/HallwayScript.h
@@ -0,0 +1,13 @@
+#pragma once
+
+#include <crepe/api/Script.h>
+
+class HallwayScript : public crepe::Script {
+public:
+ void fixed_update(crepe::duration_t dt);
+
+private:
+ float start_x = 1200;
+ const float lenght = 3000;
+ int current_sector = 2;
+};
diff --git a/game/background/HallwaySubScene.cpp b/game/background/HallwaySubScene.cpp
new file mode 100644
index 0000000..31af2d5
--- /dev/null
+++ b/game/background/HallwaySubScene.cpp
@@ -0,0 +1,167 @@
+#include "HallwaySubScene.h"
+
+#include "../Config.h"
+
+#include <crepe/api/Animator.h>
+#include <crepe/api/Color.h>
+#include <crepe/api/GameObject.h>
+#include <crepe/api/Scene.h>
+#include <crepe/api/Sprite.h>
+
+using namespace crepe;
+using namespace std;
+
+float HallwaySubScene::create(
+ Scene & scn, float begin_x, unsigned int sector_num, Color sector_color
+) {
+ GameObject begin = scn.new_object("hallway_begin", "background_hall", vec2(begin_x, 0));
+ Asset begin_asset {"asset/background/hallway/hallway1FG_1_TVOS.png"};
+ begin.add_component<Sprite>(
+ begin_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_BACKGROUND,
+ .order_in_layer = 0,
+ .size = vec2(0, GAME_HEIGHT),
+ }
+ );
+ begin_x += 600;
+
+ this->add_sector_number(begin, vec2(-200, 0), sector_num, sector_color);
+ this->add_lamp(begin, vec2(330, -120), 11);
+ this->add_lamp(begin, vec2(430, -120), 9);
+
+ GameObject middle_1
+ = scn.new_object("hallway_middle", "background_hall", vec2(begin_x, 0));
+ Asset middle_asset {"asset/background/hallway/hallway1FG_2_TVOS.png"};
+ middle_1.add_component<Sprite>(
+ middle_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_BACKGROUND,
+ .order_in_layer = 2,
+ .size = vec2(0, GAME_HEIGHT),
+ }
+ );
+ begin_x += 600;
+
+ GameObject middle_2
+ = scn.new_object("hallway_middle", "background_hall", vec2(begin_x, 0));
+ Asset middle_asset_2 {"asset/background/hallway/hallway1FG_2_TVOS.png"};
+ middle_2.add_component<Sprite>(
+ middle_asset_2,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_BACKGROUND,
+ .order_in_layer = 3,
+ .size = vec2(0, GAME_HEIGHT),
+ }
+ );
+ begin_x += 200;
+
+ GameObject middle_3
+ = scn.new_object("hallway_middle", "background_hall", vec2(begin_x, 0));
+ Asset middle_asset_3 {"asset/background/hallway/hallway1FG_2_TVOS.png"};
+ middle_3.add_component<Sprite>(
+ middle_asset_3,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_BACKGROUND,
+ .order_in_layer = 4,
+ .size = vec2(0, GAME_HEIGHT),
+ }
+ );
+ begin_x += 400;
+
+ this->add_lamp(middle_3, vec2(0, -120));
+
+ GameObject middle_4
+ = scn.new_object("hallway_middle", "background_hall", vec2(begin_x, 0));
+ Asset middle_asset_4 {"asset/background/hallway/hallway1FG_2_TVOS.png"};
+ middle_4.add_component<Sprite>(
+ middle_asset_4,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_BACKGROUND,
+ .order_in_layer = 5,
+ .size = vec2(0, GAME_HEIGHT),
+ }
+ );
+ begin_x += 600;
+
+ GameObject end = scn.new_object("hallway_end", "background_hall", vec2(begin_x, 0));
+ Asset end_asset {"asset/background/hallway/hallway1FG_1_TVOS.png"};
+ end.add_component<Sprite>(
+ end_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_BACKGROUND,
+ .order_in_layer = 1,
+ .size = vec2(0, GAME_HEIGHT),
+ }
+ );
+ begin_x += 600;
+
+ return begin_x;
+}
+
+void HallwaySubScene::add_lamp(GameObject & obj, vec2 offset, unsigned int fps) {
+ Asset lamp_asset {"asset/background/hallway/alarmLight_TVOS.png"};
+ obj.add_component<Sprite>(
+ lamp_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_FORE_BACKGROUND,
+ .order_in_layer = 0,
+ .size = vec2(0, 100),
+ .position_offset = offset,
+ }
+ );
+ Asset lamp_glow_asset {"asset/background/hallway/alarmGlow_TVOS.png"};
+ Sprite & lamp_glow_sprite = obj.add_component<Sprite>(
+ lamp_glow_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_FORE_BACKGROUND,
+ .order_in_layer = 1,
+ .size = vec2(0, 300),
+ .position_offset = offset - vec2(65, -30),
+ }
+ );
+ obj.add_component<Animator>(
+ lamp_glow_sprite, ivec2(422, 384), uvec2(6, 1),
+ Animator::Data {
+ .fps = fps,
+ .looping = true,
+ }
+ );
+}
+
+void HallwaySubScene::add_sector_number(
+ GameObject & obj, vec2 offset, unsigned int sector_num, Color sector_color
+) {
+ Asset sector_text_asset {"asset/background/hallway/sectorText_TVOS.png"};
+ obj.add_component<Sprite>(
+ sector_text_asset,
+ Sprite::Data {
+ .color = sector_color,
+ .sorting_in_layer = SORT_IN_LAY_FORE_BACKGROUND,
+ .order_in_layer = 0,
+ .size = vec2(0, 100),
+ .position_offset = offset,
+ }
+ );
+ Asset sector_num_asset {"asset/background/hallway/sectorNumbers_TVOS.png"};
+ Sprite & sector_num_sprite = obj.add_component<Sprite>(
+ sector_num_asset,
+ Sprite::Data {
+ .color = sector_color,
+ .sorting_in_layer = SORT_IN_LAY_FORE_BACKGROUND,
+ .order_in_layer = 0,
+ .size = vec2(0, 100),
+ .position_offset = offset + vec2(200, 0),
+ }
+ );
+ Animator & sector_num_anim = obj.add_component<Animator>(
+ sector_num_sprite, ivec2(256, 128), uvec2(4, 4), Animator::Data {}
+ );
+ int column = (sector_num - 1) / 4;
+ int row = (sector_num - 1) % 4;
+ sector_num_anim.set_anim(column);
+ for (int i = 0; i < row; i++) {
+ sector_num_anim.next_anim();
+ }
+ sector_num_anim.pause();
+}
diff --git a/game/background/HallwaySubScene.h b/game/background/HallwaySubScene.h
new file mode 100644
index 0000000..c38b4a9
--- /dev/null
+++ b/game/background/HallwaySubScene.h
@@ -0,0 +1,24 @@
+#pragma once
+
+#include <crepe/types.h>
+
+namespace crepe {
+class Scene;
+class GameObject;
+class Color;
+} // namespace crepe
+
+class HallwaySubScene {
+public:
+ float create(
+ crepe::Scene & scn, float begin_x, unsigned int sector_num, crepe::Color sector_color
+ );
+
+private:
+ void add_lamp(crepe::GameObject & obj, crepe::vec2 offset, unsigned int fps = 10);
+
+ void add_sector_number(
+ crepe::GameObject & obj, crepe::vec2 offset, unsigned int sector_num,
+ crepe::Color sector_color
+ );
+};
diff --git a/game/background/StartSubScene.cpp b/game/background/StartSubScene.cpp
new file mode 100644
index 0000000..d2d30ea
--- /dev/null
+++ b/game/background/StartSubScene.cpp
@@ -0,0 +1,527 @@
+#include "StartSubScene.h"
+
+#include "../Config.h"
+
+#include <crepe/api/Animator.h>
+#include <crepe/api/CircleCollider.h>
+#include <crepe/api/Color.h>
+#include <crepe/api/GameObject.h>
+#include <crepe/api/ParticleEmitter.h>
+#include <crepe/api/Rigidbody.h>
+#include <crepe/api/Scene.h>
+#include <crepe/api/Sprite.h>
+
+using namespace crepe;
+using namespace std;
+
+float StartSubScene::create(Scene & scn, float begin_x) {
+ this->create_wall_fragments(scn, begin_x - 300);
+
+ GameObject begin = scn.new_object("start_begin", "background", vec2(begin_x, 0));
+ Asset begin_asset {"asset/background/start/titleFG_1_TVOS.png"};
+ begin.add_component<Sprite>(
+ begin_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_BACKGROUND,
+ .order_in_layer = 0,
+ .size = vec2(0, GAME_HEIGHT),
+ }
+ );
+ GameObject hole = scn.new_object("start_hole", "background", vec2(begin_x - 250, 140));
+ Asset hole_asset {"asset/background/start/titleWallHole.png"};
+ Sprite & hole_sprite = hole.add_component<Sprite>(
+ hole_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_BACKGROUND,
+ .order_in_layer = 1,
+ .size = vec2(0, 200),
+ }
+ );
+ hole_sprite.active = false;
+ begin_x += 700;
+
+ this->add_table(begin, vec2(-150, 150));
+ this->add_light(begin, vec2(-125, -150));
+ this->add_jetpack_stand(begin, vec2(-125, 200));
+
+ GameObject end = scn.new_object("start_end", "background", vec2(begin_x, 0));
+ Asset end_asset {"asset/background/start/titleFG_2_TVOS.png"};
+ end.add_component<Sprite>(
+ end_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_BACKGROUND,
+ .order_in_layer = 1,
+ .size = vec2(0, GAME_HEIGHT),
+ }
+ );
+ begin_x += 100;
+
+ this->add_lamp(end, vec2(-350, -95));
+
+ return begin_x;
+}
+
+void StartSubScene::add_lamp(GameObject & obj, vec2 offset, unsigned int fps) {
+ Asset lamp_asset {"asset/background/start/alarmLight_TVOS.png"};
+ obj.add_component<Sprite>(
+ lamp_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_FORE_BACKGROUND,
+ .order_in_layer = 0,
+ .size = vec2(0, 100),
+ .position_offset = offset,
+ }
+ );
+ Asset lamp_glow_asset {"asset/background/start/alarmGlow_TVOS.png"};
+ Sprite & lamp_glow_sprite = obj.add_component<Sprite>(
+ lamp_glow_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_FORE_BACKGROUND,
+ .order_in_layer = 1,
+ .size = vec2(0, 300),
+ .position_offset = offset - vec2(65, -55),
+ }
+ );
+ lamp_glow_sprite.active = false;
+ obj.add_component<Animator>(
+ lamp_glow_sprite, ivec2(422, 384), uvec2(6, 1),
+ Animator::Data {
+ .fps = fps,
+ .looping = true,
+ }
+ );
+}
+
+void StartSubScene::add_table(GameObject & obj, vec2 offset) {
+ Asset table_asset {"asset/background/start/table.png"};
+ obj.add_component<Sprite>(
+ table_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_FORE_BACKGROUND,
+ .order_in_layer = 0,
+ .size = vec2(0, 100),
+ .position_offset = offset,
+ }
+ );
+ Asset gramophone_asset {"asset/background/start/gramophone_TVOS.png"};
+ Sprite & gramophone_sprite = obj.add_component<Sprite>(
+ gramophone_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_FORE_BACKGROUND,
+ .order_in_layer = 1,
+ .size = vec2(0, 100),
+ .position_offset = offset + vec2(0, -50),
+ }
+ );
+ obj.add_component<Animator>(
+ gramophone_sprite, ivec2(64, 128), uvec2(2, 1),
+ Animator::Data {
+ .fps = 10,
+ .looping = true,
+ }
+ );
+}
+
+void StartSubScene::add_light(crepe::GameObject & obj, crepe::vec2 offset) {
+ Asset light_asset {"asset/background/start/title_light_TVOS.png"};
+ obj.add_component<Sprite>(
+ light_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_FORE_BACKGROUND,
+ .order_in_layer = 0,
+ .size = vec2(0, 200),
+ .position_offset = offset,
+ }
+ );
+ Asset light_glow_asset {"asset/background/start/lightEffect2.png"};
+ obj.add_component<Sprite>(
+ light_glow_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_FORE_BACKGROUND,
+ .order_in_layer = 1,
+ .size = vec2(0, 100),
+ .position_offset = offset + vec2(0, 75),
+ }
+ );
+ Asset light_effect_asset {"asset/background/start/lightEffect.png"};
+ obj.add_component<Sprite>(
+ light_effect_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_FORE_BACKGROUND,
+ .order_in_layer = 0,
+ .size = vec2(0, 100),
+ .position_offset = offset + vec2(0, 350),
+ }
+ );
+}
+
+void StartSubScene::add_jetpack_stand(crepe::GameObject & obj, crepe::vec2 offset) {
+ Asset jetpack_stand_asset {"asset/background/start/JetpackStand.png"};
+ Sprite & jetpeck_stand_sprite = obj.add_component<Sprite>(
+ jetpack_stand_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_FORE_BACKGROUND,
+ .order_in_layer = 1,
+ .size = vec2(0, 70),
+ .position_offset = offset,
+ }
+ );
+ obj.add_component<Animator>(
+ jetpeck_stand_sprite, ivec2(34, 46), uvec2(2, 1),
+ Animator::Data {
+ .fps = 10,
+ .looping = true,
+ }
+ )
+ .pause();
+ Asset do_not_steal = {"asset/background/start/doNotTouchSign_TVOS.png"};
+ obj.add_component<Sprite>(
+ do_not_steal,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_FORE_BACKGROUND,
+ .order_in_layer = 1,
+ .size = vec2(0, 100),
+ .position_offset = offset + vec2(-75, -25),
+ }
+ );
+}
+
+void StartSubScene::create_wall_fragments(crepe::Scene & scn, float begin_x) {
+ GameObject frag_1 = scn.new_object("frag_1", "wall_fragment", vec2(begin_x, 200));
+ Asset frag_1_asset {"asset/background/start/StartWall_frag1.png"};
+ Sprite & frag_1_sprite = frag_1.add_component<Sprite>(
+ frag_1_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_FORE_BACKGROUND,
+ .order_in_layer = 5,
+ .size = vec2(0, 50),
+ }
+ );
+ frag_1_sprite.active = false;
+ Rigidbody & frag_1_rb = frag_1.add_component<Rigidbody>(Rigidbody::Data {
+ .gravity_scale = 1.0,
+ .linear_velocity = vec2(400, -400),
+ .linear_velocity_coefficient = vec2(0.5, 0.6),
+ .angular_velocity = 500,
+ .angular_velocity_coefficient = 0.55,
+ .elasticity_coefficient = 0.5,
+ .collision_layers = {COLL_LAY_BOT_TOP},
+ .collision_layer = COLL_LAY_WALL_FRAGS,
+ });
+ frag_1_rb.active = false;
+ frag_1.add_component<CircleCollider>(25);
+
+ GameObject frag_2 = scn.new_object("frag_2", "wall_fragment", vec2(begin_x, 180));
+ Asset frag_2_asset {"asset/background/start/StartWall_frag2.png"};
+ Sprite & frag_2_sprite = frag_2.add_component<Sprite>(
+ frag_2_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_FORE_BACKGROUND,
+ .order_in_layer = 5,
+ .size = vec2(0, 50),
+ }
+ );
+ frag_2_sprite.active = false;
+ Rigidbody & frag_2_rb = frag_2.add_component<Rigidbody>(Rigidbody::Data {
+ .gravity_scale = 1.0,
+ .linear_velocity = vec2(400, -400),
+ .linear_velocity_coefficient = vec2(0.35, 0.4),
+ .angular_velocity = 400,
+ .angular_velocity_coefficient = 0.55,
+ .elasticity_coefficient = 0.5,
+ .collision_layers = {COLL_LAY_BOT_TOP},
+ .collision_layer = COLL_LAY_WALL_FRAGS,
+ });
+ frag_2_rb.active = false;
+ frag_2.add_component<CircleCollider>(55);
+
+ GameObject frag_3 = scn.new_object("frag_3", "wall_fragment", vec2(begin_x, 170));
+ Asset frag_3_asset {"asset/background/start/StartWall_frag3.png"};
+ Sprite & frag_3_sprite = frag_3.add_component<Sprite>(
+ frag_3_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_FORE_BACKGROUND,
+ .order_in_layer = 5,
+ .size = vec2(0, 50),
+ }
+ );
+ frag_3_sprite.active = false;
+ Rigidbody & frag_3_rb = frag_3.add_component<Rigidbody>(Rigidbody::Data {
+ .gravity_scale = 1.0,
+ .linear_velocity = vec2(400, -400),
+ .linear_velocity_coefficient = vec2(0.3, 0.3),
+ .angular_velocity = 300,
+ .angular_velocity_coefficient = 0.55,
+ .elasticity_coefficient = 0.5,
+ .collision_layers = {COLL_LAY_BOT_TOP},
+ .collision_layer = COLL_LAY_WALL_FRAGS,
+ });
+ frag_3_rb.active = false;
+ frag_3.add_component<CircleCollider>(35);
+
+ GameObject frag_4 = scn.new_object("frag_4", "wall_fragment", vec2(begin_x, 160));
+ Asset frag_4_asset {"asset/background/start/StartWall_frag4.png"};
+ Sprite & frag_4_sprite = frag_4.add_component<Sprite>(
+ frag_4_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_FORE_BACKGROUND,
+ .order_in_layer = 5,
+ .size = vec2(0, 50),
+ }
+ );
+ frag_4_sprite.active = false;
+ Rigidbody & frag_4_rb = frag_4.add_component<Rigidbody>(Rigidbody::Data {
+ .gravity_scale = 1.0,
+ .linear_velocity = vec2(700, -400),
+ .linear_velocity_coefficient = vec2(0.2, 0.2),
+ .angular_velocity = 200,
+ .angular_velocity_coefficient = 0.55,
+ .elasticity_coefficient = 0.5,
+ .collision_layers = {COLL_LAY_BOT_TOP},
+ .collision_layer = COLL_LAY_WALL_FRAGS,
+ });
+ frag_4_rb.active = false;
+ frag_4.add_component<CircleCollider>(60);
+
+ GameObject frag_5 = scn.new_object("frag_5", "wall_fragment", vec2(begin_x, 150));
+ Asset frag_5_asset {"asset/background/start/StartWall_frag5.png"};
+ Sprite & frag_5_sprite = frag_5.add_component<Sprite>(
+ frag_5_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_FORE_BACKGROUND,
+ .order_in_layer = 5,
+ .size = vec2(0, 50),
+ }
+ );
+ frag_5_sprite.active = false;
+ Rigidbody & frag_5_rb = frag_5.add_component<Rigidbody>(Rigidbody::Data {
+ .gravity_scale = 1.0,
+ .linear_velocity = vec2(600, -800),
+ .linear_velocity_coefficient = vec2(0.25, 0.15),
+ .angular_velocity = 100,
+ .angular_velocity_coefficient = 0.55,
+ .elasticity_coefficient = 0.5,
+ .collision_layers = {COLL_LAY_BOT_TOP},
+ .collision_layer = COLL_LAY_WALL_FRAGS,
+ });
+ frag_5_rb.active = false;
+ frag_5.add_component<CircleCollider>(5);
+
+ GameObject frag_6 = scn.new_object("frag_6", "wall_fragment", vec2(begin_x, 140));
+ Asset frag_6_asset {"asset/background/start/StartWall_frag6.png"};
+ Sprite & frag_6_sprite = frag_6.add_component<Sprite>(
+ frag_6_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_FORE_BACKGROUND,
+ .order_in_layer = 5,
+ .size = vec2(0, 50),
+ }
+ );
+ frag_6_sprite.active = false;
+ Rigidbody & frag_6_rb = frag_6.add_component<Rigidbody>(Rigidbody::Data {
+ .gravity_scale = 1.0,
+ .linear_velocity = vec2(300, -800),
+ .linear_velocity_coefficient = vec2(0.35, 0.25),
+ .angular_velocity = 100,
+ .angular_velocity_coefficient = 0.55,
+ .elasticity_coefficient = 0.5,
+ .collision_layers = {COLL_LAY_BOT_TOP},
+ .collision_layer = COLL_LAY_WALL_FRAGS,
+ });
+ frag_6_rb.active = false;
+ frag_6.add_component<CircleCollider>(30);
+
+ GameObject frag_7 = scn.new_object("frag_7", "wall_fragment", vec2(begin_x, 130));
+ Asset frag_7_asset {"asset/background/start/StartWall_frag7.png"};
+ Sprite & frag_7_sprite = frag_7.add_component<Sprite>(
+ frag_7_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_FORE_BACKGROUND,
+ .order_in_layer = 5,
+ .size = vec2(0, 50),
+ }
+ );
+ frag_7_sprite.active = false;
+ Rigidbody & frag_7_rb = frag_7.add_component<Rigidbody>(Rigidbody::Data {
+ .gravity_scale = 1.0,
+ .linear_velocity = vec2(400, -500),
+ .linear_velocity_coefficient = vec2(0.45, 0.6),
+ .angular_velocity = 800,
+ .angular_velocity_coefficient = 0.55,
+ .elasticity_coefficient = 0.5,
+ .collision_layers = {COLL_LAY_BOT_TOP},
+ .collision_layer = COLL_LAY_WALL_FRAGS,
+ });
+ frag_7_rb.active = false;
+ frag_7.add_component<CircleCollider>(45);
+
+ GameObject frag_8 = scn.new_object("frag_8", "wall_fragment", vec2(begin_x, 120));
+ Asset frag_8_asset {"asset/background/start/StartWall_frag8.png"};
+ Sprite & frag_8_sprite = frag_8.add_component<Sprite>(
+ frag_8_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_FORE_BACKGROUND,
+ .order_in_layer = 5,
+ .size = vec2(0, 50),
+ }
+ );
+ frag_8_sprite.active = false;
+ Rigidbody & frag_8_rb = frag_8.add_component<Rigidbody>(Rigidbody::Data {
+ .gravity_scale = 1.0,
+ .linear_velocity = vec2(400, -400),
+ .linear_velocity_coefficient = vec2(0.5, 0.6),
+ .angular_velocity = 500,
+ .angular_velocity_coefficient = 0.55,
+ .elasticity_coefficient = 0.5,
+ .collision_layers = {COLL_LAY_BOT_TOP},
+ .collision_layer = COLL_LAY_WALL_FRAGS,
+ });
+ frag_8_rb.active = false;
+ frag_8.add_component<CircleCollider>(25);
+
+ GameObject frag_9 = scn.new_object("frag_9", "wall_fragment", vec2(begin_x, 110));
+ Asset frag_9_asset {"asset/background/start/StartWall_frag9.png"};
+ Sprite & frag_9_sprite = frag_9.add_component<Sprite>(
+ frag_9_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_FORE_BACKGROUND,
+ .order_in_layer = 5,
+ .size = vec2(0, 50),
+ }
+ );
+ frag_9_sprite.active = false;
+ Rigidbody & frag_9_rb = frag_9.add_component<Rigidbody>(Rigidbody::Data {
+ .gravity_scale = 1.0,
+ .linear_velocity = vec2(200, -400),
+ .linear_velocity_coefficient = vec2(0.5, 0.25),
+ .angular_velocity = 500,
+ .angular_velocity_coefficient = 0.55,
+ .elasticity_coefficient = 0.5,
+ .collision_layers = {COLL_LAY_BOT_TOP},
+ .collision_layer = COLL_LAY_WALL_FRAGS,
+ });
+ frag_9_rb.active = false;
+ frag_9.add_component<CircleCollider>(15);
+
+ GameObject frag_10 = scn.new_object("frag_10", "wall_fragment", vec2(begin_x, 100));
+ Asset frag_10_asset {"asset/background/start/StartWall_frag10.png"};
+ Sprite & frag_10_sprite = frag_10.add_component<Sprite>(
+ frag_10_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_FORE_BACKGROUND,
+ .order_in_layer = 5,
+ .size = vec2(0, 50),
+ }
+ );
+ frag_10_sprite.active = false;
+ Rigidbody & frag_10_rb = frag_10.add_component<Rigidbody>(Rigidbody::Data {
+ .gravity_scale = 1.0,
+ .linear_velocity = vec2(400, -900),
+ .linear_velocity_coefficient = vec2(0.35, 0.4),
+ .angular_velocity = 300,
+ .angular_velocity_coefficient = 0.55,
+ .elasticity_coefficient = 0.5,
+ .collision_layers = {COLL_LAY_BOT_TOP},
+ .collision_layer = COLL_LAY_WALL_FRAGS,
+ });
+ frag_10_rb.active = false;
+ frag_10.add_component<CircleCollider>(60);
+
+ GameObject frag_11 = scn.new_object("frag_11", "wall_fragment", vec2(begin_x, 70));
+ Asset frag_11_asset {"asset/background/start/StartWall_frag11.png"};
+ Sprite & frag_11_sprite = frag_11.add_component<Sprite>(
+ frag_11_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_FORE_BACKGROUND,
+ .order_in_layer = 5,
+ .size = vec2(0, 50),
+ }
+ );
+ frag_11_sprite.active = false;
+ Rigidbody & frag_11_rb = frag_11.add_component<Rigidbody>(Rigidbody::Data {
+ .gravity_scale = 1.0,
+ .linear_velocity = vec2(600, -800),
+ .linear_velocity_coefficient = vec2(0.3, 0.3),
+ .angular_velocity = 200,
+ .angular_velocity_coefficient = 0.55,
+ .elasticity_coefficient = 0.5,
+ .collision_layers = {COLL_LAY_BOT_TOP},
+ .collision_layer = COLL_LAY_WALL_FRAGS,
+ });
+ frag_11_rb.active = false;
+ frag_11.add_component<CircleCollider>(5);
+
+ GameObject frag_12 = scn.new_object("frag_12", "wall_fragment", vec2(begin_x, 80));
+ Asset frag_12_asset {"asset/background/start/StartWall_frag12.png"};
+ Sprite & frag_12_sprite = frag_12.add_component<Sprite>(
+ frag_12_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_FORE_BACKGROUND,
+ .order_in_layer = 5,
+ .size = vec2(0, 50),
+ }
+ );
+ frag_12_sprite.active = false;
+ Rigidbody & frag_12_rb = frag_12.add_component<Rigidbody>(Rigidbody::Data {
+ .gravity_scale = 1.0,
+ .linear_velocity = vec2(500, -800),
+ .linear_velocity_coefficient = vec2(0.25, 0.15),
+ .angular_velocity = 100,
+ .angular_velocity_coefficient = 0.55,
+ .elasticity_coefficient = 0.5,
+ .collision_layers = {COLL_LAY_BOT_TOP},
+ .collision_layer = COLL_LAY_WALL_FRAGS,
+ });
+ frag_12_rb.active = false;
+ frag_12.add_component<CircleCollider>(50);
+
+ GameObject smoke_particles_1
+ = scn.new_object("smoke_particles", "particle_emitter", vec2(begin_x - 100, 200));
+ Asset smoke_asset_1 {"asset/particles/smoke.png"};
+ Sprite & smoke_sprite_1 = smoke_particles_1.add_component<Sprite>(
+ smoke_asset_1,
+ Sprite::Data {
+ .color = Color(255, 255, 255, 50),
+ .sorting_in_layer = SORT_IN_LAY_PARTICLES_FOREGROUND,
+ .order_in_layer = 0,
+ .size = vec2(0, 100),
+ }
+ );
+ ParticleEmitter & emitter_1 = smoke_particles_1.add_component<ParticleEmitter>(
+ smoke_sprite_1,
+ ParticleEmitter::Data {
+ .emission_rate = 20,
+ .min_speed = 40,
+ .max_speed = 100,
+ .min_angle = -30,
+ .max_angle = 10,
+ .end_lifespan = 4,
+ }
+ );
+ emitter_1.active = false;
+
+ GameObject smoke_particles_2
+ = scn.new_object("smoke_particles", "particle_emitter", vec2(begin_x - 100, 200));
+ Asset smoke_asset_2 {"asset/particles/smoke.png"};
+ Sprite & smoke_sprite_2 = smoke_particles_2.add_component<Sprite>(
+ smoke_asset_2,
+ Sprite::Data {
+ .color = Color(255, 255, 255, 50),
+ .sorting_in_layer = SORT_IN_LAY_PARTICLES_FOREGROUND,
+ .order_in_layer = 0,
+ .size = vec2(0, 70),
+ }
+ );
+ ParticleEmitter & emitter_2 = smoke_particles_2.add_component<ParticleEmitter>(
+ smoke_sprite_2,
+ ParticleEmitter::Data {
+ .emission_rate = 30,
+ .min_speed = 40,
+ .max_speed = 100,
+ .min_angle = -45,
+ .max_angle = 5,
+ .end_lifespan = 3,
+ }
+ );
+ emitter_2.active = false;
+}
diff --git a/game/background/StartSubScene.h b/game/background/StartSubScene.h
new file mode 100644
index 0000000..c83e3d5
--- /dev/null
+++ b/game/background/StartSubScene.h
@@ -0,0 +1,20 @@
+#pragma once
+
+#include <crepe/types.h>
+
+namespace crepe {
+class Scene;
+class GameObject;
+} // namespace crepe
+
+class StartSubScene {
+public:
+ float create(crepe::Scene & scn, float begin_x);
+
+private:
+ void add_lamp(crepe::GameObject & obj, crepe::vec2 offset, unsigned int fps = 10);
+ void add_table(crepe::GameObject & obj, crepe::vec2 offset);
+ void add_light(crepe::GameObject & obj, crepe::vec2 offset);
+ void add_jetpack_stand(crepe::GameObject & obj, crepe::vec2 offset);
+ void create_wall_fragments(crepe::Scene & scn, float begin_x);
+};
diff --git a/game/coins/CoinPoolSubScene.cpp b/game/coins/CoinPoolSubScene.cpp
new file mode 100644
index 0000000..f8b5b70
--- /dev/null
+++ b/game/coins/CoinPoolSubScene.cpp
@@ -0,0 +1,13 @@
+#include "CoinPoolSubScene.h"
+#include "CoinSubScene.h"
+
+using namespace crepe;
+using namespace std;
+
+void CoinPoolSubScene::create_coins(crepe::Scene & scn) {
+ int amount = 0;
+ CoinSubScene coin;
+ while (amount < this->MAXIMUM_AMOUNT) {
+ amount = coin.create(scn, amount);
+ }
+}
diff --git a/game/coins/CoinPoolSubScene.h b/game/coins/CoinPoolSubScene.h
new file mode 100644
index 0000000..07626d6
--- /dev/null
+++ b/game/coins/CoinPoolSubScene.h
@@ -0,0 +1,11 @@
+#pragma once
+
+#include <crepe/api/Scene.h>
+
+class CoinPoolSubScene {
+public:
+ void create_coins(crepe::Scene & scn);
+
+private:
+ static constexpr int MAXIMUM_AMOUNT = 100;
+};
diff --git a/game/coins/CoinScript.cpp b/game/coins/CoinScript.cpp
new file mode 100644
index 0000000..514f4de
--- /dev/null
+++ b/game/coins/CoinScript.cpp
@@ -0,0 +1,71 @@
+#include "CoinScript.h"
+
+#include "manager/SaveManager.h"
+
+#include "../Config.h"
+#include "../hud/HudScript.h"
+
+#include <crepe/api/Animator.h>
+#include <crepe/api/AudioSource.h>
+#include <crepe/api/CircleCollider.h>
+#include <crepe/api/Sprite.h>
+
+using namespace crepe;
+using namespace std;
+
+bool CoinScript::on_collision(const CollisionEvent & collisionData) {
+ if (collisionData.info.other.metadata.tag != "coin") return false;
+ if (!this->get_components_by_name<Sprite>(collisionData.info.other.metadata.name)
+ .front()
+ .get()
+ .active)
+ return false;
+ this->get_components_by_name<Sprite>(collisionData.info.other.metadata.name)
+ .front()
+ .get()
+ .active
+ = false;
+ this->get_components_by_name<CircleCollider>(collisionData.info.other.metadata.name)
+ .front()
+ .get()
+ .active
+ = false;
+ this->amount++;
+
+ AudioSource & audio = this->get_components_by_id<AudioSource>(
+ collisionData.info.other.metadata.game_object_id
+ )
+ .front();
+ audio.play();
+
+ this->get_components_by_name<Sprite>(collisionData.info.other.metadata.name)
+ .back()
+ .get()
+ .active
+ = true;
+ this->get_components_by_name<Animator>(collisionData.info.other.metadata.name)
+ .back()
+ .get()
+ .active
+ = true;
+
+ return false;
+}
+
+void CoinScript::init() {
+ this->subscribe<CollisionEvent>([this](const CollisionEvent & ev) -> bool {
+ return this->on_collision(ev);
+ });
+}
+
+void CoinScript::fixed_update(crepe::duration_t dt) {
+ this->trigger_event(GetCoinEvent {
+ .amount_of_coins = this->amount,
+ });
+}
+
+bool CoinScript::save() {
+ SaveManager & savemgr = this->get_save_manager();
+ savemgr.set(TOTAL_COINS_RUN, this->amount);
+ return false;
+}
diff --git a/game/coins/CoinScript.h b/game/coins/CoinScript.h
new file mode 100644
index 0000000..5718025
--- /dev/null
+++ b/game/coins/CoinScript.h
@@ -0,0 +1,14 @@
+#pragma once
+
+#include "api/Script.h"
+
+class CoinScript : public crepe::Script {
+public:
+ void init() override;
+ void fixed_update(crepe::duration_t dt) override;
+ bool on_collision(const crepe::CollisionEvent & collisionData);
+ bool save();
+
+private:
+ int amount = 0;
+};
diff --git a/game/coins/CoinSubScene.cpp b/game/coins/CoinSubScene.cpp
new file mode 100644
index 0000000..d154819
--- /dev/null
+++ b/game/coins/CoinSubScene.cpp
@@ -0,0 +1,64 @@
+#include "CoinSubScene.h"
+
+#include "../Config.h"
+
+#include <crepe/api/Animator.h>
+#include <crepe/api/AudioSource.h>
+#include <crepe/api/CircleCollider.h>
+#include <crepe/api/Rigidbody.h>
+#include <crepe/api/Scene.h>
+
+using namespace crepe;
+using namespace std;
+
+int CoinSubScene::create(Scene & scn, int coin_counter) {
+ vec2 size = {20, 20};
+
+ string unique_name = "coin_" + to_string(coin_counter++);
+
+ GameObject coin = scn.new_object(unique_name.c_str(), "coin", vec2 {650, 0}, 0, 1);
+ coin.add_component<Rigidbody>(Rigidbody::Data {
+ .body_type = Rigidbody::BodyType::KINEMATIC,
+ .kinematic_collision = false,
+ .collision_layers = {COLL_LAY_PLAYER},
+ });
+ coin.add_component<CircleCollider>((size.x / 2) - 3).active = false;
+ crepe::OptionalRef<crepe::Sprite> coin_sprite = coin.add_component<Sprite>(
+ Asset {"asset/coin/coin1_TVOS.png"},
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_COINS,
+ .order_in_layer = 0,
+ .size = size,
+ }
+ );
+ coin_sprite->active = false;
+ coin.add_component<Animator>(
+ coin_sprite, ivec2 {32, 32}, uvec2 {8, 1},
+ Animator::Data {
+ .fps = 15,
+ .looping = true,
+ }
+ );
+ coin.add_component<AudioSource>(Asset {"asset/sfx/coin_pickup_1.ogg"}).volume = 3;
+
+ Sprite & pick_up = coin.add_component<Sprite>(
+ Asset {"asset/coin/coinCollect1_TVOS.png"},
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_COINS,
+ .order_in_layer = 1,
+ .size = size * 2,
+ }
+ );
+ pick_up.active = false;
+ coin.add_component<Animator>(
+ pick_up, ivec2 {64, 64}, uvec2 {5, 1},
+ Animator::Data {
+ .fps = 5,
+ .looping = false,
+ }
+ )
+ .active
+ = false;
+
+ return coin_counter;
+}
diff --git a/game/coins/CoinSubScene.h b/game/coins/CoinSubScene.h
new file mode 100644
index 0000000..7a1c60a
--- /dev/null
+++ b/game/coins/CoinSubScene.h
@@ -0,0 +1,12 @@
+#pragma once
+
+#include <crepe/api/GameObject.h>
+
+namespace crepe {
+class Scene;
+}
+
+class CoinSubScene {
+public:
+ int create(crepe::Scene & scn, int coin_counter);
+};
diff --git a/game/coins/CoinSystemScript.cpp b/game/coins/CoinSystemScript.cpp
new file mode 100644
index 0000000..f9816c9
--- /dev/null
+++ b/game/coins/CoinSystemScript.cpp
@@ -0,0 +1,258 @@
+#include "CoinSystemScript.h"
+
+#include <random>
+
+#include <crepe/api/CircleCollider.h>
+#include <crepe/api/Metadata.h>
+#include <crepe/api/Sprite.h>
+#include <crepe/api/Transform.h>
+
+using namespace crepe;
+using namespace std;
+
+void CoinSystemScript::init() { engine.seed(rd()); }
+
+void CoinSystemScript::add_location(const crepe::vec2 & location) {
+ coin_locations.push_back(CoinData(location));
+}
+
+float CoinSystemScript::preset_1(const vec2 & begin_position) {
+ vec2 top = {begin_position.x, begin_position.y - (this->ROW_OFFSET_1)};
+ vec2 bottom = {begin_position.x, begin_position.y + (this->ROW_OFFSET_1)};
+
+ // Add locations for the top row
+ for (int i = 0; i < COLUM_AMOUNT_1; ++i) {
+ add_location(top);
+ top.x += this->COLUM_OFFSET_1;
+ }
+
+ // Add locations for the bottom row
+ bottom.x += this->COLUM_OFFSET_1 * COLUM_AMOUNT_1;
+ for (int i = 0; i < COLUM_AMOUNT_1; ++i) {
+ add_location(bottom);
+ bottom.x += this->COLUM_OFFSET_1;
+ }
+
+ // Add locations for the next set of the top row
+ top.x += this->COLUM_OFFSET_1 * COLUM_AMOUNT_1;
+ for (int i = 0; i < COLUM_AMOUNT_1; ++i) {
+ add_location(top);
+ top.x += this->COLUM_OFFSET_1;
+ }
+
+ // Add locations for the next set of the bottom row
+ bottom.x += this->COLUM_OFFSET_1 * COLUM_AMOUNT_1;
+ for (int i = 0; i < COLUM_AMOUNT_1; ++i) {
+ add_location(bottom);
+ bottom.x += this->COLUM_OFFSET_1;
+ }
+
+ return bottom.x - begin_position.x;
+}
+
+float CoinSystemScript::preset_2(const vec2 & begin_position) {
+ vec2 top
+ = {begin_position.x + this->COLUM_OFFSET_2, begin_position.y - this->ROW_OFFSET_2};
+ vec2 middle = begin_position;
+ vec2 bottom
+ = {begin_position.x + this->COLUM_OFFSET_2, begin_position.y + this->ROW_OFFSET_2};
+
+ // Add locations for the next set of the bottom row
+ for (int i = 0; i < COLUM_AMOUNT_2 - 2; ++i) {
+ add_location(bottom);
+ bottom.x += this->COLUM_OFFSET_2;
+ }
+
+ // Add locations for the next set of the middle row
+ for (int i = 0; i < COLUM_AMOUNT_2; ++i) {
+ add_location(middle);
+ middle.x += this->COLUM_OFFSET_2;
+ }
+
+ // Add locations for the next set of the top row
+ for (int i = 0; i < COLUM_AMOUNT_2 - 2; ++i) {
+ add_location(top);
+ top.x += this->COLUM_OFFSET_2;
+ }
+
+ return middle.x - begin_position.x;
+}
+
+float CoinSystemScript::preset_3(const vec2 & begin_position) {
+ vec2 location = {begin_position.x, begin_position.y - (this->ROW_OFFSET_3)};
+
+ // Add locations for the top row
+ for (int i = 0; i < COLUM_AMOUNT_3; ++i) {
+ add_location(location);
+ location.x += this->COLUM_OFFSET_3;
+ }
+
+ // Add locations for the bottom row
+ location.y += this->ROW_OFFSET_3;
+ location.x += this->COLUM_OFFSET_3;
+ for (int i = 0; i < COLUM_AMOUNT_3; ++i) {
+ add_location(location);
+ location.x += this->COLUM_OFFSET_3;
+ }
+
+ // Add locations for the next set of the top row
+ location.y += this->ROW_OFFSET_3;
+ location.x += this->COLUM_OFFSET_3;
+ for (int i = 0; i < COLUM_AMOUNT_3; ++i) {
+ add_location(location);
+ location.x += this->COLUM_OFFSET_3;
+ }
+
+ return location.x - begin_position.x;
+}
+
+float CoinSystemScript::preset_4(const vec2 & begin_position) {
+ vec2 location = {begin_position.x, begin_position.y + (this->ROW_OFFSET_4)};
+
+ // Add locations for the top row
+ for (int i = 0; i < COLUM_AMOUNT_4; ++i) {
+ add_location(location);
+ location.x += this->COLUM_OFFSET_4;
+ }
+
+ // Add locations for the bottom row
+ location.y -= this->ROW_OFFSET_4;
+ location.x += this->COLUM_OFFSET_4;
+ for (int i = 0; i < COLUM_AMOUNT_4; ++i) {
+ add_location(location);
+ location.x += this->COLUM_OFFSET_4;
+ }
+
+ // Add locations for the next set of the top row
+ location.y -= this->ROW_OFFSET_4;
+ location.x += this->COLUM_OFFSET_4;
+ for (int i = 0; i < COLUM_AMOUNT_4; ++i) {
+ add_location(location);
+ location.x += this->COLUM_OFFSET_4;
+ }
+
+ return location.x - begin_position.x;
+}
+
+float CoinSystemScript::preset_5(const vec2 & begin_position) {
+ vec2 location = {begin_position.x, begin_position.y - ROW_OFFSET_5 / 2};
+ for (int i = 0; i < COLUM_AMOUNT_5; ++i) {
+ add_location(location);
+ location.x += this->COLUM_OFFSET_5;
+ }
+ return location.x - begin_position.x;
+}
+
+void CoinSystemScript::frame_update(crepe::duration_t dt) {
+ this->despawn_coins();
+ this->generate_locations();
+ this->spawn_coins();
+}
+
+void CoinSystemScript::despawn_coins() {
+ // Get the current x-position of the CoinSystem's Transform component
+ float position = this->get_component<Transform>().position.x;
+
+ // Retrieve all active coin sprites tagged as "coin"
+ RefVector<Sprite> coin_sprites = this->get_components_by_tag<Sprite>("coin");
+
+ for (Sprite & coin_sprite : coin_sprites) {
+ if (!coin_sprite.active) continue; // Skip inactive sprites
+
+ // Retrieve the corresponding Transform, Metadata, and CircleCollider components
+ Transform & coin_transform
+ = this->get_components_by_id<Transform>(coin_sprite.game_object_id).front().get();
+ Metadata & coin_metadata
+ = this->get_components_by_id<Metadata>(coin_sprite.game_object_id).front().get();
+ CircleCollider & coin_collider
+ = this->get_components_by_id<CircleCollider>(coin_sprite.game_object_id)
+ .front()
+ .get();
+
+ // Check if the coin is out of bounds based on DESPAWN_DISTANCE
+ if (coin_transform.position.x < position - this->DESPAWN_DISTANCE) {
+ // Find the coin in the coin_locations vector using its name
+ auto it = std::find_if(
+ coin_locations.begin(), coin_locations.end(),
+ [&coin_metadata](const CoinData & data) {
+ return data.name == coin_metadata.name;
+ }
+ );
+
+ // If a match is found, erase it from coin_locations
+ if (it != coin_locations.end()) {
+ coin_locations.erase(it);
+ coin_sprite.active = false;
+ coin_collider.active = false;
+ }
+ }
+ }
+}
+
+void CoinSystemScript::spawn_coins() {
+ // Get the current x-position of the CoinSystem's Transform component
+ float position = this->get_component<Transform>().position.x;
+
+ // Iterate through the list of coin locations
+ for (auto & coin : coin_locations) {
+ // Skip this coin if it is already active
+ if (coin.active) continue;
+ // Skip this coin if it is not within the defined spawn area
+ if (coin.start_location.x < this->SPAWN_DISTANCE + position
+ || coin.start_location.x > this->SPAWN_AREA + this->SPAWN_DISTANCE + position)
+ continue;
+
+ // Retrieve all sprites tagged as "coin"
+ RefVector<Sprite> coin_sprites = this->get_components_by_tag<Sprite>("coin");
+
+ // Check for an available (inactive) coin sprite
+ for (Sprite & coin_sprite : coin_sprites) {
+ // Skip this sprite if it is already active
+ if (coin_sprite.active) continue;
+ if (coin_sprite.data.order_in_layer == 1) {
+ coin_sprite.active = false;
+ continue;
+ }
+
+ // Found an available (inactive) coin sprite
+ // Retrieve its associated components
+ Transform & coin_transform
+ = this->get_components_by_id<Transform>(coin_sprite.game_object_id)
+ .front()
+ .get();
+ Metadata & coin_metadata
+ = this->get_components_by_id<Metadata>(coin_sprite.game_object_id)
+ .front()
+ .get();
+ CircleCollider & coin_collider
+ = this->get_components_by_id<CircleCollider>(coin_sprite.game_object_id)
+ .front()
+ .get();
+
+ // Assign data and set active
+ coin.name = coin_metadata.name;
+ coin.active = true;
+ coin_sprite.active = true;
+ coin_collider.active = true;
+ coin_transform.position = coin.start_location;
+
+ // Break out of the inner loop since we've assigned this coin to an available sprite
+ break;
+ }
+ }
+}
+
+void CoinSystemScript::generate_locations() {
+ float position = this->get_component<Transform>().position.x;
+ if (position + SPAWN_DISTANCE + SYSTEM_POSITION_OFFSET < this->system_position) return;
+
+ std::discrete_distribution<int> dist(weights.begin(), weights.end());
+ int selected_index = dist(engine);
+
+ std::uniform_real_distribution<float> space_dist(SPAWN_SPACING_MIN, SPAWN_SPACING_MAX);
+ float spacing = space_dist(engine);
+
+ // Call the corresponding function and return the new x position
+ this->system_position += functions[selected_index]({this->system_position, 0});
+ this->system_position += spacing;
+}
diff --git a/game/coins/CoinSystemScript.h b/game/coins/CoinSystemScript.h
new file mode 100644
index 0000000..5c94273
--- /dev/null
+++ b/game/coins/CoinSystemScript.h
@@ -0,0 +1,107 @@
+#pragma once
+
+#include <random>
+#include <string>
+
+#include <crepe/api/CircleCollider.h>
+#include <crepe/api/Script.h>
+#include <crepe/api/Sprite.h>
+#include <crepe/api/Transform.h>
+#include <crepe/types.h>
+
+class CoinSystemScript : public crepe::Script {
+private:
+ struct CoinData {
+ crepe::vec2 start_location = {0, 0};
+ std::string name = "";
+ bool active = false;
+ CoinData(crepe::vec2 start_location)
+ : start_location(start_location),
+ name(""),
+ active(false) {}
+ };
+
+public:
+ CoinSystemScript() {};
+ void init() override;
+ void frame_update(crepe::duration_t dt) override;
+
+private:
+ void add_location(const crepe::vec2 & location);
+ void despawn_coins();
+ void spawn_coins();
+ void generate_locations();
+ float preset_1(const crepe::vec2 & begin_position);
+ float preset_2(const crepe::vec2 & begin_position);
+ float preset_3(const crepe::vec2 & begin_position);
+ float preset_4(const crepe::vec2 & begin_position);
+ float preset_5(const crepe::vec2 & begin_position);
+
+private:
+ std::vector<std::function<float(const crepe::vec2 &)>> functions
+ = {[this](const crepe::vec2 & pos) { return preset_1(pos); },
+ [this](const crepe::vec2 & pos) { return preset_2(pos); },
+ [this](const crepe::vec2 & pos) { return preset_3(pos); },
+ [this](const crepe::vec2 & pos) { return preset_4(pos); },
+ [this](const crepe::vec2 & pos) { return preset_5(pos); }};
+ std::vector<int> weights = {20, 20, 20, 20, 20};
+ std::random_device rd;
+ std::default_random_engine engine;
+ float system_position = 1400;
+ static constexpr float SYSTEM_POSITION_OFFSET = 200;
+
+private:
+ static constexpr float SPAWN_SPACING_MIN = 400;
+ static constexpr float SPAWN_SPACING_MAX = 1000;
+ static constexpr float SPAWN_DISTANCE = 600;
+ static constexpr float DESPAWN_DISTANCE = 600;
+ static constexpr float SPAWN_AREA = 50;
+ std::vector<CoinData> coin_locations;
+
+private:
+ // preset one settings
+ // ***** *****
+ //
+ //
+ //
+ // ***** *****
+ static constexpr float ROW_OFFSET_1 = 100;
+ static constexpr float COLUM_OFFSET_1 = 25;
+ static constexpr int COLUM_AMOUNT_1 = 5;
+
+private:
+ // preset two settings
+ //
+ // ********
+ // **********
+ // ********
+ //
+ static constexpr float ROW_OFFSET_2 = 25;
+ static constexpr float COLUM_OFFSET_2 = 25;
+ static constexpr int COLUM_AMOUNT_2 = 10;
+ // preset three settings
+ // ***
+ //
+ // ***
+ //
+ // ***
+ static constexpr float ROW_OFFSET_3 = 100;
+ static constexpr float COLUM_OFFSET_3 = 25;
+ static constexpr int COLUM_AMOUNT_3 = 3;
+ // preset four settings
+ // ***
+ //
+ // ***
+ //
+ // ***
+ static constexpr float ROW_OFFSET_4 = 100;
+ static constexpr float COLUM_OFFSET_4 = 25;
+ static constexpr int COLUM_AMOUNT_4 = 3;
+ // preset five settings
+ //
+ // ***
+ //
+ static constexpr float ROW_OFFSET_5 = 25;
+ static constexpr float COLUM_OFFSET_5 = 25;
+ static constexpr int COLUM_AMOUNT_5 = 3;
+};
diff --git a/game/enemy/BattleScript.cpp b/game/enemy/BattleScript.cpp
new file mode 100644
index 0000000..798cbb9
--- /dev/null
+++ b/game/enemy/BattleScript.cpp
@@ -0,0 +1,61 @@
+#include "BattleScript.h"
+#include "EnemyConfig.h"
+#include "EnemyScript.h"
+#include "api/Transform.h"
+#include <crepe/api/AI.h>
+#include <crepe/api/BehaviorScript.h>
+#include <crepe/api/Metadata.h>
+using namespace std;
+using namespace crepe;
+
+BattleScript::BattleScript() { engine.seed(rd()); }
+void BattleScript::init() {
+ std::uniform_int_distribution<int> dist(2, 10);
+ int random_enemy_amount = dist(this->engine);
+ // this->create_battle(random_enemy_amount);
+ this->subscribe<BattleStartEvent>([this](const BattleStartEvent & e) -> bool {
+ return this->create_battle(e);
+ });
+}
+void BattleScript::fixed_update(duration_t dt) {
+ if (!battle_active) return;
+ bool enemies_alive = false;
+ RefVector<AI> enemy_ai = this->get_components_by_tag<AI>("enemy");
+
+ for (AI & ai : enemy_ai) {
+ if (ai.active) {
+ enemies_alive = true;
+ }
+ }
+ if (!enemies_alive) {
+ this->battle_active = false;
+ this->trigger_event<BattleWonEvent>();
+ }
+}
+bool BattleScript::create_battle(const BattleStartEvent & e) {
+ this->battle_active = e.battle;
+ this->spawn_enemies(e.num_enemies);
+ return false;
+}
+void BattleScript::spawn_enemies(int amount) {
+ RefVector<AI> enemy_ai = this->get_components_by_tag<AI>("enemy");
+ std::uniform_real_distribution<float> dist(70, 150);
+ int spawned = 0;
+ for (int i = 0; i < ENEMY_POOL_MAX; i++) {
+ AI & ai = enemy_ai[i];
+ Transform & enemy_transform
+ = this->get_components_by_id<Transform>(ai.game_object_id).front();
+ if (ai.active == true || enemy_transform.position != ENEMY_POOL_LOCATION) continue;
+ this->queue_event<SpawnEnemyEvent>(
+ SpawnEnemyEvent {
+ .speed = dist(engine),
+ .column = i,
+ },
+ ai.game_object_id
+ );
+ spawned++;
+ if (spawned >= amount) {
+ return;
+ }
+ }
+}
diff --git a/game/enemy/BattleScript.h b/game/enemy/BattleScript.h
new file mode 100644
index 0000000..57aa16c
--- /dev/null
+++ b/game/enemy/BattleScript.h
@@ -0,0 +1,26 @@
+#pragma once
+
+#include <crepe/api/BehaviorScript.h>
+#include <crepe/api/Event.h>
+#include <crepe/api/Script.h>
+#include <random>
+struct BattleWonEvent : public crepe::Event {};
+
+struct BattleStartEvent : public crepe::Event {
+public:
+ int num_enemies = 0;
+ bool battle = false;
+};
+class BattleScript : public crepe::Script {
+public:
+ BattleScript();
+ void init() override;
+ void fixed_update(crepe::duration_t dt) override;
+
+private:
+ bool battle_active = false;
+ std::random_device rd;
+ std::default_random_engine engine;
+ void spawn_enemies(int amount);
+ bool create_battle(const BattleStartEvent & e);
+};
diff --git a/game/enemy/EnemyBulletPool.cpp b/game/enemy/EnemyBulletPool.cpp
new file mode 100644
index 0000000..3ee4816
--- /dev/null
+++ b/game/enemy/EnemyBulletPool.cpp
@@ -0,0 +1,11 @@
+#include "EnemyBulletPool.h"
+#include "EnemyBulletSubScene.h"
+using namespace std;
+
+void EnemyBulletPool::create_bullets(crepe::Scene & scn) {
+ EnemyBulletSubScene bullet;
+ int amount = 0;
+ while (amount < this->MAXIMUM_AMOUNT) {
+ amount = bullet.create(scn, amount);
+ }
+}
diff --git a/game/enemy/EnemyBulletPool.h b/game/enemy/EnemyBulletPool.h
new file mode 100644
index 0000000..ee53fc4
--- /dev/null
+++ b/game/enemy/EnemyBulletPool.h
@@ -0,0 +1,11 @@
+#pragma once
+
+#include <crepe/api/Scene.h>
+
+class EnemyBulletPool {
+public:
+ void create_bullets(crepe::Scene & scn);
+
+private:
+ static constexpr int MAXIMUM_AMOUNT = 20;
+};
diff --git a/game/enemy/EnemyBulletScript.cpp b/game/enemy/EnemyBulletScript.cpp
new file mode 100644
index 0000000..d208a88
--- /dev/null
+++ b/game/enemy/EnemyBulletScript.cpp
@@ -0,0 +1,40 @@
+#include "EnemyBulletScript.h"
+#include <crepe/api/Camera.h>
+#include <crepe/api/Metadata.h>
+#include <crepe/api/Rigidbody.h>
+#include <iostream>
+
+#include "EnemyConfig.h"
+using namespace crepe;
+using namespace std;
+void EnemyBulletScript::init() {
+ this->subscribe<CollisionEvent>([this](const CollisionEvent & e) -> bool {
+ return this->on_collide(e);
+ });
+}
+void EnemyBulletScript::fixed_update(crepe::duration_t dt) {
+ Transform & transform = this->get_component<Transform>();
+ Camera & camera = this->get_components_by_name<Camera>("camera").front();
+ Transform & cam_transform = this->get_components_by_name<Transform>("camera").front();
+ Rigidbody & bullet_body = this->get_component<Rigidbody>();
+ //move
+ transform.position += bullet_body.data.linear_velocity * dt.count();
+ vec2 half_screen = camera.viewport_size / 2;
+ float despawn_location = cam_transform.position.x - half_screen.x - 50;
+ if (transform.position.x < despawn_location) {
+ this->despawn_bullet();
+ }
+}
+
+void EnemyBulletScript::despawn_bullet() {
+ Transform & transform = this->get_component<Transform>();
+ Rigidbody & bullet_body = this->get_component<Rigidbody>();
+ bullet_body.active = false;
+ transform.position = ENEMY_BULLET_POOL_LOCATION;
+}
+
+bool EnemyBulletScript::on_collide(const CollisionEvent & e) {
+ //cout << "collision happened with " << e.info.other.metadata.tag << endl;
+ this->despawn_bullet();
+ return false;
+}
diff --git a/game/enemy/EnemyBulletScript.h b/game/enemy/EnemyBulletScript.h
new file mode 100644
index 0000000..7dab751
--- /dev/null
+++ b/game/enemy/EnemyBulletScript.h
@@ -0,0 +1,11 @@
+#pragma once
+#include <crepe/api/BehaviorScript.h>
+#include <crepe/api/Script.h>
+
+class EnemyBulletScript : public crepe::Script {
+public:
+ void init() override;
+ void fixed_update(crepe::duration_t dt) override;
+ bool on_collide(const crepe::CollisionEvent & e);
+ void despawn_bullet();
+};
diff --git a/game/enemy/EnemyBulletSubScene.cpp b/game/enemy/EnemyBulletSubScene.cpp
new file mode 100644
index 0000000..bc31ba8
--- /dev/null
+++ b/game/enemy/EnemyBulletSubScene.cpp
@@ -0,0 +1,52 @@
+#include <string>
+
+#include "../Config.h"
+#include "EnemyConfig.h"
+#include <crepe/api/AI.h>
+#include <crepe/api/Animator.h>
+#include <crepe/api/BehaviorScript.h>
+#include <crepe/api/BoxCollider.h>
+#include <crepe/api/CircleCollider.h>
+#include <crepe/api/GameObject.h>
+#include <crepe/api/Rigidbody.h>
+#include <crepe/api/Scene.h>
+#include <crepe/api/Sprite.h>
+
+#include "../Random.h"
+#include "EnemyBulletScript.h"
+#include "EnemyBulletSubScene.h"
+#include "EnemyScript.h"
+#include "api/Color.h"
+using namespace crepe;
+using namespace std;
+int EnemyBulletSubScene::create(Scene & scn, int counter) {
+ string unique_name = "enemy_bullet_" + to_string(counter++);
+ GameObject bullet = scn.new_object(
+ unique_name.c_str(), "enemy_bullet", ENEMY_BULLET_POOL_LOCATION, 0, 1
+ );
+
+ Rigidbody & bullet_body = bullet.add_component<Rigidbody>(Rigidbody::Data {
+ .gravity_scale = 0,
+ .body_type = Rigidbody::BodyType::KINEMATIC,
+ .linear_velocity = vec2 {-300, 0},
+ .kinematic_collision = false,
+ .collision_layers = {COLL_LAY_MISSILE, COLL_LAY_ZAPPER},
+ .collision_layer = COLL_LAY_BULLET
+ });
+ bullet_body.active = false;
+ BoxCollider & bullet_collider = bullet.add_component<BoxCollider>(vec2(40, 10));
+ //bullet_collider.active = false;
+ Asset bullet_asset {"asset/other_effects/effect_smgbullet_x2.png"};
+ Sprite & bullet_sprite = bullet.add_component<Sprite>(
+ bullet_asset,
+ Sprite::Data {
+ .color = Color::BLUE,
+ .flip = {true, false},
+ .sorting_in_layer = SORT_IN_LAY_OBSTACLES,
+ .order_in_layer = 1,
+ .size = vec2(60, 0),
+ }
+ );
+ bullet.add_component<BehaviorScript>().set_script<EnemyBulletScript>();
+ return counter;
+}
diff --git a/game/enemy/EnemyBulletSubScene.h b/game/enemy/EnemyBulletSubScene.h
new file mode 100644
index 0000000..ac78ad9
--- /dev/null
+++ b/game/enemy/EnemyBulletSubScene.h
@@ -0,0 +1,10 @@
+#pragma once
+
+namespace crepe {
+class Scene;
+}
+
+class EnemyBulletSubScene {
+public:
+ int create(crepe::Scene & scn, int counter);
+};
diff --git a/game/enemy/EnemyConfig.h b/game/enemy/EnemyConfig.h
new file mode 100644
index 0000000..f9fb469
--- /dev/null
+++ b/game/enemy/EnemyConfig.h
@@ -0,0 +1,8 @@
+#pragma once
+#include <crepe/types.h>
+
+//button config
+// static constexpr crepe::vec2 PLAYER_BULLET_POOL_LOCATION = {0, -850};
+static constexpr crepe::vec2 ENEMY_BULLET_POOL_LOCATION = {0, -750};
+static constexpr crepe::vec2 ENEMY_POOL_LOCATION = {0, -650};
+static constexpr int ENEMY_POOL_MAX = 12;
diff --git a/game/enemy/EnemyPool.cpp b/game/enemy/EnemyPool.cpp
new file mode 100644
index 0000000..135fc35
--- /dev/null
+++ b/game/enemy/EnemyPool.cpp
@@ -0,0 +1,10 @@
+#include "EnemyPool.h"
+#include "EnemySubScene.h"
+using namespace std;
+void EnemyPool::create_enemies(crepe::Scene & scn) {
+ EnemySubScene enemy;
+ int amount = 0;
+ while (amount < ENEMY_POOL_MAX) {
+ amount = enemy.create(scn, amount);
+ }
+}
diff --git a/game/enemy/EnemyPool.h b/game/enemy/EnemyPool.h
new file mode 100644
index 0000000..cfd0b1c
--- /dev/null
+++ b/game/enemy/EnemyPool.h
@@ -0,0 +1,8 @@
+#pragma once
+
+#include "EnemyConfig.h"
+#include <crepe/api/Scene.h>
+class EnemyPool {
+public:
+ void create_enemies(crepe::Scene & scn);
+};
diff --git a/game/enemy/EnemyScript.cpp b/game/enemy/EnemyScript.cpp
new file mode 100644
index 0000000..0822c29
--- /dev/null
+++ b/game/enemy/EnemyScript.cpp
@@ -0,0 +1,206 @@
+#include "EnemyScript.h"
+#include "../Config.h"
+#include "../Random.h"
+#include "../enemy/EnemyConfig.h"
+#include "api/Color.h"
+#include "api/Sprite.h"
+#include <crepe/api/AI.h>
+#include <crepe/api/Animator.h>
+#include <crepe/api/AudioSource.h>
+#include <crepe/api/BoxCollider.h>
+#include <crepe/api/ParticleEmitter.h>
+#include <crepe/api/Rigidbody.h>
+#include <crepe/api/Sprite.h>
+#include <crepe/api/Transform.h>
+#include <crepe/types.h>
+#include <random>
+using namespace crepe;
+using namespace std;
+EnemyScript::EnemyScript() {
+ engine.seed(rd());
+ this->last_fired = std::chrono::steady_clock::now();
+ this->shot_delay = std::chrono::duration<float>(1.5 + Random::f(2, 0));
+}
+void EnemyScript::init() {
+ Metadata & meta = this->get_component<Metadata>();
+ this->subscribe<SpawnEnemyEvent>(
+ [this](const SpawnEnemyEvent & e) -> bool { return this->spawn_enemy(e); },
+ meta.game_object_id
+ );
+ this->subscribe<CollisionEvent>([this](const CollisionEvent & e) -> bool {
+ return this->on_collide(e);
+ });
+};
+void EnemyScript::fixed_update(duration_t dt) {
+ if (!spawned) return;
+ auto now = std::chrono::steady_clock::now();
+ std::chrono::duration<float> elapsed_hit = now - last_hit;
+ //hit blink timer
+ if (elapsed_hit > blink_time) {
+ set_hit_blink(false);
+ }
+ Transform & transform = this->get_component<Transform>();
+ if (!this->alive) {
+ Camera & camera = this->get_components_by_name<Camera>("camera").front();
+ Transform & cam_transform = this->get_components_by_name<Transform>("camera").front();
+ vec2 half_screen = camera.viewport_size / 2;
+ float x_value = cam_transform.position.x - half_screen.x - 100;
+ if (transform.position.x < x_value) {
+ this->despawn_enemy();
+ }
+ return;
+ }
+
+ Transform & player_transform = this->get_components_by_name<Transform>("player").front();
+ Rigidbody & enemy_body = this->get_component<Rigidbody>();
+ AI & ai_component = this->get_component<AI>();
+
+ float direction_to_player_y = player_transform.position.y - transform.position.y;
+ float distance_to_player_y = std::abs(direction_to_player_y);
+
+ float adjustment_speed = speed * (distance_to_player_y / MAX_DISTANCE);
+ adjustment_speed = std::clamp(adjustment_speed, MIN_SPEED, MAX_SPEED);
+ Rigidbody & player_body = this->get_components_by_tag<Rigidbody>("player").front();
+ // move path nodes
+ for (vec2 & path_node : ai_component.path) {
+ path_node.y += (direction_to_player_y > 0 ? 1 : -1) * adjustment_speed * dt.count();
+ path_node.x += player_body.data.linear_velocity.x * dt.count();
+ }
+ //bullet fire logic:
+
+ std::chrono::duration<float> elapsed = now - last_fired;
+ if (elapsed > shot_delay) {
+ this->shoot(transform.position);
+ last_fired = now;
+ this->shot_delay = std::chrono::duration<float>(Random::f(3, 1.5));
+ }
+}
+
+bool EnemyScript::spawn_enemy(const SpawnEnemyEvent & e) {
+
+ this->speed = e.speed;
+ this->alive = true;
+ this->spawned = true;
+ this->health = 2;
+ RefVector<Animator> animators = this->get_components<Animator>();
+ for (Animator & anim : animators) {
+ anim.active = false;
+ anim.set_anim(0);
+ }
+ RefVector<Sprite> sprites = this->get_components<Sprite>();
+ for (Sprite & sprite : sprites) {
+ sprite.data.position_offset.x = 0;
+ }
+ Sprite & jetpack = sprites[2];
+ jetpack.data.position_offset.x = 20;
+ Sprite & gun = sprites[3];
+ gun.data.position_offset.x = -20;
+ AI & ai_component = this->get_component<AI>();
+ Transform & transform = this->get_component<Transform>();
+ Camera & camera = this->get_components_by_name<Camera>("camera").front();
+ Transform & cam_transform = this->get_components_by_name<Transform>("camera").front();
+ Rigidbody & rb = this->get_component<Rigidbody>();
+ rb.data.collision_layers = {COLL_LAY_BOT_TOP, COLL_LAY_PLAYER_BULLET};
+ rb.data.collision_layer = COLL_LAY_ENEMY;
+ vec2 half_screen = camera.viewport_size / 2;
+ float x_value = cam_transform.position.x + half_screen.x - 40 * (1 + e.column);
+ uniform_real_distribution<float> dist(
+ cam_transform.position.y - half_screen.y + 100,
+ cam_transform.position.y + half_screen.y - 100
+ );
+ float random_height = dist(engine);
+ vec2 spawn_location
+ = {cam_transform.position.x + camera.viewport_size.x / 2 + 100, random_height};
+ transform.position = spawn_location;
+ ai_component.path.clear();
+ ai_component.make_oval_path(10, 30, vec2 {x_value, random_height}, 1.5708, true);
+ ai_component.active = true;
+ this->last_fired = std::chrono::steady_clock::now();
+
+ return false;
+}
+
+void EnemyScript::set_hit_blink(bool status) {
+ RefVector<Sprite> sprites = this->get_components<Sprite>();
+ for (Sprite & sprite : sprites) {
+ if (status) {
+ sprite.data.color = Color::RED;
+ continue;
+ }
+ sprite.data.color = Color::WHITE;
+ }
+}
+
+bool EnemyScript::on_collide(const CollisionEvent & e) {
+ if (!this->alive) return false;
+ if (e.info.other.metadata.tag == "player_bullet") {
+ this->health--;
+ last_hit = std::chrono::steady_clock::now();
+ //Sprite& sprite;
+ set_hit_blink(true);
+ if (health <= 0) {
+ this->death();
+ }
+ }
+
+ //body_animator.play();
+
+ return false;
+}
+void EnemyScript::death() {
+
+ Rigidbody & rb = this->get_component<Rigidbody>();
+ Transform & tr = this->get_component<Transform>();
+ RefVector<Animator> animators = this->get_components<Animator>();
+ for (Animator & anim : animators) {
+ anim.active = false;
+ anim.set_anim(3);
+ }
+ RefVector<Sprite> sprites = this->get_components<Sprite>();
+ for (Sprite & sprite : sprites) {
+ sprite.data.position_offset.x = 15;
+ }
+ rb.data.linear_velocity_coefficient = {0.5, 1};
+ rb.data.collision_layers = {COLL_LAY_BOT_TOP};
+ rb.data.collision_layer = 0;
+
+ rb.data.gravity_scale = 1;
+ tr.rotation = 90;
+ AI & ai = this->get_component<AI>();
+ ai.active = false;
+ this->alive = false;
+ AI & ai_component = this->get_component<AI>();
+ ai_component.active = false;
+}
+void EnemyScript::despawn_enemy() {
+ Transform & transform = this->get_component<Transform>();
+ Rigidbody & rb = this->get_component<Rigidbody>();
+ rb.data.gravity_scale = 0;
+ rb.data.linear_velocity = {0, 0};
+ transform.rotation = 0;
+ transform.position = ENEMY_POOL_LOCATION;
+
+ this->spawned = false;
+}
+
+void EnemyScript::shoot(const vec2 & location) {
+ RefVector<Transform> bullet_transforms
+ = this->get_components_by_tag<Transform>("enemy_bullet");
+
+ for (Transform & bullet_pos : bullet_transforms) {
+ if (bullet_pos.position.x == 0 && bullet_pos.position.y == -750) {
+
+ bullet_pos.position = location;
+ bullet_pos.position.x -= 20;
+ Rigidbody & bullet_body
+ = this->get_components_by_id<Rigidbody>(bullet_pos.game_object_id).front();
+ BoxCollider bullet_collider
+ = this->get_components_by_id<BoxCollider>(bullet_pos.game_object_id).front();
+ bullet_collider.active = true;
+ bullet_body.active = true;
+ AudioSource & audio = this->get_component<AudioSource>();
+ audio.play();
+ return;
+ }
+ }
+}
diff --git a/game/enemy/EnemyScript.h b/game/enemy/EnemyScript.h
new file mode 100644
index 0000000..be71a78
--- /dev/null
+++ b/game/enemy/EnemyScript.h
@@ -0,0 +1,37 @@
+#pragma once
+#include <chrono>
+#include <crepe/api/Camera.h>
+#include <crepe/api/Event.h>
+#include <crepe/api/Script.h>
+#include <random>
+struct SpawnEnemyEvent : public crepe::Event {
+ float speed = 0;
+ int column = 0;
+};
+class EnemyScript : public crepe::Script {
+public:
+ EnemyScript();
+ void init() override;
+ void fixed_update(crepe::duration_t dt) override;
+ void shoot(const crepe::vec2 & position);
+ bool on_collide(const crepe::CollisionEvent & collisionData);
+ void despawn_enemy();
+ bool spawn_enemy(const SpawnEnemyEvent & e);
+ void death();
+ void set_hit_blink(bool status);
+
+private:
+ std::random_device rd;
+ std::default_random_engine engine;
+ bool alive = false;
+ bool spawned = false;
+ float speed = 50;
+ int health = 2;
+ const float MIN_SPEED = 20;
+ const float MAX_SPEED = 150;
+ const float MAX_DISTANCE = 200;
+ std::chrono::time_point<std::chrono::steady_clock> last_fired;
+ std::chrono::time_point<std::chrono::steady_clock> last_hit;
+ std::chrono::duration<float> shot_delay = std::chrono::duration<float>(0);
+ std::chrono::duration<float> blink_time = std::chrono::duration<float>(0.1);
+};
diff --git a/game/enemy/EnemySubScene.cpp b/game/enemy/EnemySubScene.cpp
new file mode 100644
index 0000000..c6aecec
--- /dev/null
+++ b/game/enemy/EnemySubScene.cpp
@@ -0,0 +1,117 @@
+#include <string>
+
+#include <crepe/api/AI.h>
+#include <crepe/api/Animator.h>
+#include <crepe/api/AudioSource.h>
+#include <crepe/api/BehaviorScript.h>
+#include <crepe/api/BoxCollider.h>
+#include <crepe/api/CircleCollider.h>
+#include <crepe/api/GameObject.h>
+#include <crepe/api/Rigidbody.h>
+#include <crepe/api/Scene.h>
+#include <crepe/api/Sprite.h>
+
+#include "../Config.h"
+#include "EnemyConfig.h"
+#include "EnemyScript.h"
+#include "EnemySubScene.h"
+using namespace crepe;
+using namespace std;
+//#TODO add sound
+int EnemySubScene::create(Scene & scn, int enemy_counter) {
+
+ string unique_name = "enemy_" + to_string(enemy_counter++);
+ GameObject enemy = scn.new_object(unique_name.c_str(), "enemy", ENEMY_POOL_LOCATION, 0, 1);
+
+ enemy.add_component<Rigidbody>(Rigidbody::Data {
+ .gravity_scale = 0,
+ .body_type = Rigidbody::BodyType::DYNAMIC,
+ .max_linear_velocity = 400,
+ .collision_layers = {COLL_LAY_BOT_TOP, COLL_LAY_PLAYER_BULLET},
+ .collision_layer = COLL_LAY_ENEMY,
+
+ });
+ // normal body
+ Asset enemy_body_asset {"asset/workers/worker2Body.png"};
+ enemy.add_component<BoxCollider>(vec2(40, 60));
+ Sprite & enemy_body_sprite = enemy.add_component<Sprite>(
+ enemy_body_asset,
+ Sprite::Data {
+ .flip = {true, false},
+ .sorting_in_layer = SORT_IN_LAY_WORKERS_FRONT,
+ .order_in_layer = 0,
+ .size = vec2(0, 50),
+ }
+ );
+ Animator & body_animator = enemy.add_component<Animator>(
+ enemy_body_sprite, ivec2(32, 32), uvec2(4, 8),
+ Animator::Data {
+ .fps = 5,
+ .col = 1,
+ .row = 0,
+ .looping = false,
+ }
+ );
+ body_animator.pause();
+
+ Asset enemy_head_asset {"asset/workers/worker2Head.png"};
+ Sprite & enemy_head_sprite = enemy.add_component<Sprite>(
+ enemy_head_asset,
+ Sprite::Data {
+ .flip = {true, false},
+ .sorting_in_layer = SORT_IN_LAY_WORKERS_FRONT,
+ .order_in_layer = 1,
+ .size = vec2(0, 50),
+ .position_offset = vec2(0, -20),
+ }
+ );
+ enemy.add_component<Animator>(
+ enemy_head_sprite, ivec2(32, 32), uvec2(4, 8),
+ Animator::Data {
+ .fps = 5,
+ .looping = true,
+ }
+ );
+
+ //jetpack
+ //enemy.add_component<CircleCollider>(25, vec2(0, -20));
+ Asset enemy_jetpack_asset {"asset/barry/jetpackDefault.png"};
+ Sprite & enemy_jetpack_sprite = enemy.add_component<Sprite>(
+ enemy_jetpack_asset,
+ Sprite::Data {
+ .flip = {true, false},
+ .sorting_in_layer = SORT_IN_LAY_WORKERS_FRONT,
+ .order_in_layer = 2,
+ .size = vec2(0, 60),
+ .position_offset = vec2(20, 0),
+ }
+ );
+ enemy_jetpack_sprite.active = true;
+ enemy.add_component<Animator>(
+ enemy_jetpack_sprite, ivec2(32, 44), uvec2(4, 4),
+ Animator::Data {
+ .fps = 5,
+ .looping = true,
+ }
+ );
+ //gun
+ Asset enemy_pistol_asset {"asset/workers/gun.png"};
+ Sprite & enemy_pistol_sprite = enemy.add_component<Sprite>(
+ enemy_pistol_asset,
+ Sprite::Data {
+ .flip = {false, false},
+ .sorting_in_layer = SORT_IN_LAY_WORKERS_FRONT,
+ .order_in_layer = 2,
+ .size = vec2(0, 20),
+ .position_offset = vec2(-20, 0),
+ }
+ );
+ enemy.add_component<AudioSource>(Asset("asset/sfx/bike_gun_2.ogg")).volume = 0.1;
+ AI & ai_component = enemy.add_component<AI>(3000);
+ ai_component.path_follow_on();
+ ai_component.active = false;
+ BehaviorScript & enemy_script
+ = enemy.add_component<BehaviorScript>().set_script<EnemyScript>();
+ //enemy_script.active = false;
+ return enemy_counter;
+}
diff --git a/game/enemy/EnemySubScene.h b/game/enemy/EnemySubScene.h
new file mode 100644
index 0000000..3899250
--- /dev/null
+++ b/game/enemy/EnemySubScene.h
@@ -0,0 +1,10 @@
+#pragma once
+
+namespace crepe {
+class Scene;
+}
+
+class EnemySubScene {
+public:
+ int create(crepe::Scene & scn, int enemy_counter);
+};
diff --git a/game/hud/HudConfig.h b/game/hud/HudConfig.h
new file mode 100644
index 0000000..facc298
--- /dev/null
+++ b/game/hud/HudConfig.h
@@ -0,0 +1,33 @@
+#pragma once
+#include <crepe/types.h>
+
+static constexpr crepe::vec2 TOP_LEFT = {-530, -230};
+static constexpr const char * HUD_DISTANCE = "hud_distance";
+static constexpr const char * HUD_BEST = "hud_best";
+static constexpr const char * HUD_COINS = "hud_coins";
+static constexpr const char * HUD_FPS = "hud_fps";
+
+// Distance
+static constexpr const char * DISTANCE_PLACEHOLDER = "0000m";
+static constexpr const char * DISTANCE_UNIT = "m";
+static constexpr int DISTANCE_LENGTH = 5;
+static constexpr float DISTANCE_CHAR_WIDTH = 12;
+static constexpr float STEP_SIZE_DISTANCE = 100;
+
+// BEST
+static constexpr const char * BEST = "BEST:";
+static constexpr int BEST_LENGTH = 5;
+static constexpr float BEST_CHAR_WIDTH = 10;
+static constexpr crepe::vec2 BEST_OFFSET = {0, 25};
+
+// COINS
+static constexpr const char * COINS = "0000";
+static constexpr int COINS_LENGTH = 4;
+static constexpr float COINS_CHAR_WIDTH = 10;
+static constexpr crepe::vec2 COINS_OFFSET = {0, 50};
+
+// FPS
+static constexpr const char * FPS = "00";
+static constexpr int FPS_LENGTH = 2;
+static constexpr float FPS_CHAR_WIDTH = 10;
+static constexpr crepe::vec2 FPS_OFFSET = {1030, 0};
diff --git a/game/hud/HudScript.cpp b/game/hud/HudScript.cpp
new file mode 100644
index 0000000..e4aeae7
--- /dev/null
+++ b/game/hud/HudScript.cpp
@@ -0,0 +1,98 @@
+#include "HudScript.h"
+#include "HudConfig.h"
+
+#include "../Config.h"
+#include "../Events.h"
+#include "api/KeyCodes.h"
+#include "menus/endgame/EndGameSubScript.h"
+
+#include <climits>
+
+#include <crepe/api/Text.h>
+#include <crepe/api/Transform.h>
+#include <crepe/manager/SaveManager.h>
+
+using namespace crepe;
+using namespace std;
+
+void HudScript::init() {
+ savemgr = &this->get_save_manager();
+ savemgr->set(TOTAL_COINS_RUN, 0);
+ Text & txt = this->get_components_by_name<Text>(HUD_BEST).front();
+ string record
+ = BEST + to_string(savemgr->get<int>(DISTANCE_GAME, 0).get()) + DISTANCE_UNIT;
+ txt.text = record;
+ txt.dimensions = {BEST_CHAR_WIDTH * record.size(), (BEST_CHAR_WIDTH) * 2};
+ txt.offset
+ = TOP_LEFT + FONTOFFSET + BEST_OFFSET + vec2 {record.size() * BEST_CHAR_WIDTH / 2, 0};
+
+ this->subscribe<GetCoinEvent>([this](const GetCoinEvent e) -> bool {
+ return this->get_coin(e);
+ });
+ this->subscribe<KeyPressEvent>([this](const KeyPressEvent & ev) -> bool {
+ return this->toggle_fps(ev);
+ });
+ this->subscribe<EndGameEvent>([this](const EndGameEvent e) -> bool {
+ return this->save();
+ });
+}
+
+bool HudScript::toggle_fps(crepe::KeyPressEvent ev) {
+ if (ev.key != Keycode::D1) return false;
+ Text & txt_fps = this->get_components_by_name<Text>(HUD_FPS).front();
+ this->show_fps = !this->show_fps;
+ if (this->show_fps) {
+ txt_fps.active = true;
+ } else {
+ txt_fps.active = false;
+ }
+ return true;
+}
+
+void HudScript::frame_update(crepe::duration_t dt) {
+
+ // Distance
+ Text & txt_dt = this->get_components_by_name<Text>(HUD_DISTANCE).front();
+ Transform & tf = this->get_components_by_name<Transform>(PLAYER_NAME).front();
+ string distance
+ = to_string(static_cast<int>(tf.position.x / STEP_SIZE_DISTANCE)) + DISTANCE_UNIT;
+ this->distance_st = distance;
+ txt_dt.text = distance;
+ txt_dt.dimensions = {DISTANCE_CHAR_WIDTH * distance.size(), (DISTANCE_CHAR_WIDTH) * 2};
+ txt_dt.offset
+ = TOP_LEFT + FONTOFFSET + vec2 {distance.size() * DISTANCE_CHAR_WIDTH / 2, 0};
+
+ // Coins
+ Text & txt_co = this->get_components_by_name<Text>(HUD_COINS).front();
+ string amount_of_coins = to_string(this->coin_amount);
+ this->coin_amount_st = amount_of_coins;
+ txt_co.text = amount_of_coins;
+ txt_co.dimensions = {COINS_CHAR_WIDTH * amount_of_coins.size(), (COINS_CHAR_WIDTH) * 2};
+ txt_co.offset = TOP_LEFT + FONTOFFSET + COINS_OFFSET
+ + vec2 {amount_of_coins.size() * COINS_CHAR_WIDTH / 2, 0};
+
+ // FPS
+ Text & txt_fps = this->get_components_by_name<Text>(HUD_FPS).front();
+ float fps = this->get_loop_timer().get_fps();
+ string fps_amount = to_string(this->get_loop_timer().get_fps());
+ txt_fps.text = fps_amount;
+ txt_fps.dimensions = {FPS_CHAR_WIDTH * fps_amount.size(), (FPS_CHAR_WIDTH) * 2};
+ txt_fps.offset = TOP_LEFT + FONTOFFSET + FPS_OFFSET
+ + vec2 {fps_amount.size() * FPS_CHAR_WIDTH / 2, 0};
+ if (fps >= 30) txt_fps.data.text_color = Color::YELLOW;
+ if (fps >= 50) txt_fps.data.text_color = Color::GREEN;
+ if (fps < 30) txt_fps.data.text_color = Color::RED;
+}
+
+bool HudScript::get_coin(const GetCoinEvent e) {
+ this->coin_amount = e.amount_of_coins;
+ return true;
+}
+
+bool HudScript::save() {
+ SaveManager & savemgr = this->get_save_manager();
+ savemgr.set(TOTAL_COINS_RUN, this->coin_amount);
+ savemgr.set(DISTANCE_RUN, this->distance_st);
+ this->trigger_event<ShowScoreEvent>();
+ return false;
+}
diff --git a/game/hud/HudScript.h b/game/hud/HudScript.h
new file mode 100644
index 0000000..2b789db
--- /dev/null
+++ b/game/hud/HudScript.h
@@ -0,0 +1,25 @@
+#pragma once
+
+#include <crepe/api/Event.h>
+#include <crepe/api/Script.h>
+#include <crepe/manager/SaveManager.h>
+
+struct GetCoinEvent : public crepe::Event {
+ int amount_of_coins;
+};
+
+class HudScript : public crepe::Script {
+public:
+ void init() override;
+ void frame_update(crepe::duration_t dt) override;
+ bool get_coin(const GetCoinEvent e);
+ bool toggle_fps(crepe::KeyPressEvent ev);
+ bool save();
+
+private:
+ crepe::SaveManager * savemgr;
+ bool show_fps = false;
+ int coin_amount = 0;
+ std::string coin_amount_st = "";
+ std::string distance_st = "";
+};
diff --git a/game/hud/HudSubScene.cpp b/game/hud/HudSubScene.cpp
new file mode 100644
index 0000000..dcc07b4
--- /dev/null
+++ b/game/hud/HudSubScene.cpp
@@ -0,0 +1,68 @@
+#include "HudSubScene.h"
+#include "HudConfig.h"
+
+#include "../Config.h"
+
+#include <crepe/api/GameObject.h>
+#include <crepe/api/Text.h>
+
+using namespace crepe;
+using namespace std;
+
+void HudSubScene::create(Scene & scn) {
+
+ // Distance
+ GameObject hud_dis = scn.new_object(HUD_DISTANCE);
+
+ crepe::vec2 size_distance
+ = {DISTANCE_CHAR_WIDTH * DISTANCE_LENGTH, (DISTANCE_CHAR_WIDTH) * 2};
+ hud_dis.add_component<Text>(
+ size_distance, FONT,
+ Text::Data {
+ .world_space = false,
+ .text_color = Color::WHITE,
+ },
+ TOP_LEFT + FONTOFFSET + vec2 {DISTANCE_LENGTH * DISTANCE_CHAR_WIDTH / 2, 0},
+ DISTANCE_PLACEHOLDER
+ );
+
+ // Best
+ GameObject hud_best = scn.new_object(HUD_BEST);
+ crepe::vec2 size_best = {BEST_CHAR_WIDTH * BEST_LENGTH, (BEST_CHAR_WIDTH) * 2};
+ hud_best.add_component<Text>(
+ size_best, FONT,
+ Text::Data {
+ .world_space = false,
+ .text_color = Color::GREY,
+ },
+ TOP_LEFT + FONTOFFSET + BEST_OFFSET + vec2 {BEST_LENGTH * BEST_CHAR_WIDTH / 2, 0}, BEST
+ );
+
+ // Coins
+ GameObject hud_coin = scn.new_object(HUD_COINS);
+ crepe::vec2 size_coin = {COINS_CHAR_WIDTH * COINS_LENGTH, (COINS_CHAR_WIDTH) * 2};
+ hud_coin.add_component<Text>(
+ size_coin, FONT,
+ Text::Data {
+ .world_space = false,
+ .text_color = Color::GOLD,
+ },
+ TOP_LEFT + FONTOFFSET + COINS_OFFSET + vec2 {COINS_LENGTH * COINS_CHAR_WIDTH / 2, 0},
+ COINS
+ );
+
+ // Fps
+ GameObject hud_fps = scn.new_object(HUD_FPS);
+ crepe::vec2 size_fps = {FPS_CHAR_WIDTH * FPS_LENGTH, (FPS_CHAR_WIDTH) * 2};
+ hud_fps
+ .add_component<Text>(
+ size_fps, FONT,
+ Text::Data {
+ .world_space = false,
+ .text_color = Color::GREEN,
+ },
+ TOP_LEFT + FONTOFFSET + FPS_OFFSET + vec2 {FPS_LENGTH * FPS_CHAR_WIDTH / 2, 0}, FPS
+ )
+ .active
+ = false;
+}
diff --git a/game/hud/HudSubScene.h b/game/hud/HudSubScene.h
new file mode 100644
index 0000000..0cd368e
--- /dev/null
+++ b/game/hud/HudSubScene.h
@@ -0,0 +1,8 @@
+#pragma once
+
+#include <crepe/api/Scene.h>
+
+class HudSubScene {
+public:
+ void create(crepe::Scene & scn);
+};
diff --git a/game/hud/SpeedScript.cpp b/game/hud/SpeedScript.cpp
new file mode 100644
index 0000000..2ced47a
--- /dev/null
+++ b/game/hud/SpeedScript.cpp
@@ -0,0 +1,42 @@
+#include "SpeedScript.h"
+
+#include "../Events.h"
+#include "api/BehaviorScript.h"
+#include <crepe/api/Event.h>
+#include <crepe/api/KeyCodes.h>
+#include <crepe/manager/LoopTimerManager.h>
+
+using namespace crepe;
+using namespace std;
+
+void SpeedScript::init() {
+ this->subscribe<KeyPressEvent>([this](const KeyPressEvent & ev) -> bool {
+ if (ev.key != Keycode::HOME) return false;
+ LoopTimerManager & lp = this->get_loop_timer();
+ this->toggle = !this->toggle;
+ if (this->toggle) {
+ this->timescale = lp.get_time_scale();
+ lp.set_time_scale(0);
+ } else {
+ lp.set_time_scale(this->timescale);
+ }
+
+ return true;
+ });
+ this->subscribe<EndGameEvent>([this](const EndGameEvent e) {
+ this->get_component<BehaviorScript>().active = false;
+ return false;
+ });
+}
+
+void SpeedScript::fixed_update(crepe::duration_t dt) {
+ LoopTimerManager & lp = this->get_loop_timer();
+ if (this->get_key_state(Keycode::PAGE_UP)) {
+ if (lp.get_time_scale() >= 2) return;
+ lp.set_time_scale(lp.get_time_scale() + 0.1);
+ }
+ if (this->get_key_state(Keycode::PAGE_DOWN)) {
+ if (lp.get_time_scale() <= 0.5) return;
+ lp.set_time_scale(lp.get_time_scale() - 0.1);
+ }
+}
diff --git a/game/hud/SpeedScript.h b/game/hud/SpeedScript.h
new file mode 100644
index 0000000..b40f7cc
--- /dev/null
+++ b/game/hud/SpeedScript.h
@@ -0,0 +1,15 @@
+#pragma once
+
+#include <crepe/api/Script.h>
+#include <crepe/manager/SaveManager.h>
+
+class SpeedScript : public crepe::Script {
+public:
+ void init() override;
+ void fixed_update(crepe::duration_t dt) override;
+
+private:
+ crepe::SaveManager * savemgr;
+ bool toggle = false;
+ float timescale = 1;
+};
diff --git a/game/main.cpp b/game/main.cpp
new file mode 100644
index 0000000..95cb35c
--- /dev/null
+++ b/game/main.cpp
@@ -0,0 +1,27 @@
+#include <cstdlib>
+
+#include <crepe/api/Engine.h>
+#include <crepe/api/Script.h>
+
+#include "EngineConfig.h"
+#include "GameScene.h"
+#include "PreviewScene.h"
+#include "menus/mainmenu/MainMenuScene.h"
+#include "menus/shop/ShopMenuScene.h"
+
+using namespace crepe;
+
+int main() {
+ srand(time(NULL));
+
+ Config::get_instance() = ENGINE_CONFIG;
+
+ Engine gameloop;
+
+ gameloop.add_scene<MainMenuScene>();
+ gameloop.add_scene<ShopMenuScene>();
+ gameloop.add_scene<GameScene>();
+ gameloop.add_scene<PreviewScene>();
+
+ return gameloop.main();
+}
diff --git a/game/makefile b/game/makefile
new file mode 100644
index 0000000..3fedf7f
--- /dev/null
+++ b/game/makefile
@@ -0,0 +1,5 @@
+.PHONY: FORCE
+
+format: FORCE
+ $(MAKE) -C .. $@
+
diff --git a/game/menus/BannerSubScene.cpp b/game/menus/BannerSubScene.cpp
new file mode 100644
index 0000000..006a829
--- /dev/null
+++ b/game/menus/BannerSubScene.cpp
@@ -0,0 +1,49 @@
+#include "BannerSubScene.h"
+#include "MenusConfig.h"
+
+#include "../Config.h"
+
+#include <crepe/api/Scene.h>
+#include <crepe/api/Sprite.h>
+#include <crepe/api/Text.h>
+
+using namespace crepe;
+using namespace std;
+
+void BannerSubScene::create(Scene & scn, const Data & data) {
+ GameObject menu_banner = scn.new_object("menu_banner", "", {0, -414});
+ menu_banner.add_component<Sprite>(
+ Asset("asset/ui/settings_container/top_middle_setting.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 1,
+ .size = {1100, 88},
+ }
+ );
+ menu_banner.add_component<Sprite>(
+ Asset("asset/ui/settings_container/top_2_middle_setting.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 1,
+ .size = {1100, 66},
+ .position_offset {0, 77},
+ }
+ );
+ menu_banner.add_component<Sprite>(
+ Asset("asset/ui/settings_container/banner_bottom.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 1,
+ .size = {1100, 7},
+ .position_offset {0, 113},
+ }
+ );
+ crepe::vec2 size
+ = {data.banner_title_width, (data.banner_title_width / data.banner_title.size()) * 2};
+
+ menu_banner.add_component<Text>(
+ size, FONT,
+ Text::Data {
+ .world_space = true,
+ .text_color = Color::WHITE,
+ },
+ data.banner_title_offset + FONTOFFSET, data.banner_title
+ );
+}
diff --git a/game/menus/BannerSubScene.h b/game/menus/BannerSubScene.h
new file mode 100644
index 0000000..c194dfc
--- /dev/null
+++ b/game/menus/BannerSubScene.h
@@ -0,0 +1,20 @@
+#pragma once
+
+#include <crepe/api/GameObject.h>
+#include <crepe/types.h>
+
+namespace crepe {
+class Scene;
+}
+
+class BannerSubScene {
+public:
+ struct Data {
+ const std::string & banner_title = "NODATA";
+ const float banner_title_width = 100;
+ const crepe::vec2 & banner_title_offset = {0, 0};
+ };
+
+public:
+ void create(crepe::Scene & scn, const Data & data);
+};
diff --git a/game/menus/ButtonNextMainMenuSubScript.cpp b/game/menus/ButtonNextMainMenuSubScript.cpp
new file mode 100644
index 0000000..63a2777
--- /dev/null
+++ b/game/menus/ButtonNextMainMenuSubScript.cpp
@@ -0,0 +1,44 @@
+#include "ButtonNextMainMenuSubScript.h"
+#include "ButtonReplaySubScript.h"
+#include "MenusConfig.h"
+#include "ValueBroker.h"
+
+#include "manager/SaveManager.h"
+
+#include "../Config.h"
+
+#include <crepe/api/AudioSource.h>
+#include <crepe/types.h>
+
+using namespace crepe;
+using namespace std;
+
+void ButtonNextMainMenuSubScript::init() {
+ IButtonScript::init();
+ this->subscribe<ButtonPressEvent>([this](const ButtonPressEvent & e) {
+ return this->on_button_press(e);
+ });
+}
+
+bool ButtonNextMainMenuSubScript::on_button_press(const ButtonPressEvent & e) {
+ RefVector<AudioSource> audios
+ = this->get_components_by_name<AudioSource>("background_music");
+
+ for (AudioSource & audio : audios) {
+ audio.stop();
+ }
+
+ this->trigger_event<DeleteRecordingEvent>();
+ SaveManager & savemgr = this->get_save_manager();
+
+ ValueBroker<int> coins = savemgr.get<int>(TOTAL_COINS_RUN, 0);
+ ValueBroker<int> coins_game = savemgr.get<int>(TOTAL_COINS_GAME, 0);
+ savemgr.set(TOTAL_COINS_GAME, coins_game.get() + coins.get());
+
+ ValueBroker<int> distance = savemgr.get<int>(DISTANCE_RUN, 0);
+ ValueBroker<int> distance_game = savemgr.get<int>(DISTANCE_GAME, 0);
+ if (distance.get() > distance_game.get()) savemgr.set(DISTANCE_GAME, distance.get());
+
+ this->set_next_scene(MAINMENU_SCENE);
+ return false;
+}
diff --git a/game/menus/ButtonNextMainMenuSubScript.h b/game/menus/ButtonNextMainMenuSubScript.h
new file mode 100644
index 0000000..3bc3f52
--- /dev/null
+++ b/game/menus/ButtonNextMainMenuSubScript.h
@@ -0,0 +1,14 @@
+#pragma once
+
+#include "IButtonScript.h"
+
+#include <crepe/api/Script.h>
+
+class ButtonNextMainMenuSubScript : public IButtonScript {
+public:
+ void init() override;
+ bool on_button_press(const crepe::ButtonPressEvent & e);
+
+protected:
+ bool transition = false;
+};
diff --git a/game/menus/ButtonReplaySubScript.cpp b/game/menus/ButtonReplaySubScript.cpp
new file mode 100644
index 0000000..01cccbf
--- /dev/null
+++ b/game/menus/ButtonReplaySubScript.cpp
@@ -0,0 +1,43 @@
+#include "ButtonReplaySubScript.h"
+#include "Config.h"
+#include "MenusConfig.h"
+
+#include "../Events.h"
+#include <crepe/api/AudioSource.h>
+#include <crepe/types.h>
+
+using namespace crepe;
+using namespace std;
+
+void ButtonReplaySubScript::init() {
+ IButtonScript::init();
+ this->subscribe<ButtonPressEvent>([this](const ButtonPressEvent & e) {
+ return this->on_button_press(e);
+ });
+ this->subscribe<EndGameEvent>([this](const EndGameEvent & e) {
+ return this->set_recording();
+ });
+ this->subscribe<DeleteRecordingEvent>([this](const DeleteRecordingEvent & e) {
+ return this->delete_recording();
+ });
+ if (DISABLE_REPLAY) return;
+ replay.record_start();
+}
+
+bool ButtonReplaySubScript::on_button_press(const ButtonPressEvent & e) {
+ if (DISABLE_REPLAY) return false;
+ replay.play(this->recording);
+ return false;
+}
+
+bool ButtonReplaySubScript::set_recording() {
+ if (DISABLE_REPLAY) return false;
+ this->recording = replay.record_end();
+ return false;
+}
+
+bool ButtonReplaySubScript::delete_recording() {
+ if (DISABLE_REPLAY) return false;
+ replay.release(this->recording);
+ return false;
+}
diff --git a/game/menus/ButtonReplaySubScript.h b/game/menus/ButtonReplaySubScript.h
new file mode 100644
index 0000000..3eb8aa9
--- /dev/null
+++ b/game/menus/ButtonReplaySubScript.h
@@ -0,0 +1,21 @@
+#pragma once
+
+#include "IButtonScript.h"
+
+#include <crepe/api/Script.h>
+
+struct DeleteRecordingEvent : public crepe::Event {};
+
+class ButtonReplaySubScript : public IButtonScript {
+public:
+ void init() override;
+ bool on_button_press(const crepe::ButtonPressEvent & e);
+
+private:
+ crepe::recording_t recording = 0;
+ bool set_recording();
+ bool delete_recording();
+
+protected:
+ bool transition = false;
+};
diff --git a/game/menus/ButtonSetMainMenuSubScript.cpp b/game/menus/ButtonSetMainMenuSubScript.cpp
new file mode 100644
index 0000000..1c6bcb2
--- /dev/null
+++ b/game/menus/ButtonSetMainMenuSubScript.cpp
@@ -0,0 +1,23 @@
+#include "ButtonSetMainMenuSubScript.h"
+#include "MenusConfig.h"
+
+#include <crepe/api/AudioSource.h>
+#include <crepe/types.h>
+
+using namespace crepe;
+using namespace std;
+
+void ButtonSetMainMenuSubScript::init() {
+ IButtonScript::init();
+ this->subscribe<ButtonPressEvent>([this](const ButtonPressEvent & e) {
+ return this->on_button_press(e);
+ });
+}
+
+bool ButtonSetMainMenuSubScript::on_button_press(const ButtonPressEvent & e) {
+ RefVector<AudioSource> audios
+ = this->get_components_by_name<AudioSource>("background_music");
+
+ this->set_next_scene(MAINMENU_SCENE);
+ return false;
+}
diff --git a/game/menus/ButtonSetMainMenuSubScript.h b/game/menus/ButtonSetMainMenuSubScript.h
new file mode 100644
index 0000000..2fb2634
--- /dev/null
+++ b/game/menus/ButtonSetMainMenuSubScript.h
@@ -0,0 +1,14 @@
+#pragma once
+
+#include "IButtonScript.h"
+
+#include <crepe/api/Script.h>
+
+class ButtonSetMainMenuSubScript : public IButtonScript {
+public:
+ void init() override;
+ bool on_button_press(const crepe::ButtonPressEvent & e);
+
+protected:
+ bool transition = false;
+};
diff --git a/game/menus/ButtonSetShopSubScript.cpp b/game/menus/ButtonSetShopSubScript.cpp
new file mode 100644
index 0000000..4f395eb
--- /dev/null
+++ b/game/menus/ButtonSetShopSubScript.cpp
@@ -0,0 +1,17 @@
+#include "ButtonSetShopSubScript.h"
+#include "MenusConfig.h"
+
+using namespace crepe;
+using namespace std;
+
+void ButtonSetShopSubScript::init() {
+ IButtonScript::init();
+ this->subscribe<ButtonPressEvent>([this](const ButtonPressEvent & e) {
+ return this->on_button_press(e);
+ });
+}
+
+bool ButtonSetShopSubScript::on_button_press(const ButtonPressEvent & e) {
+ this->set_next_scene(SHOP_SCENE);
+ return false;
+}
diff --git a/game/menus/ButtonSetShopSubScript.h b/game/menus/ButtonSetShopSubScript.h
new file mode 100644
index 0000000..4017a4c
--- /dev/null
+++ b/game/menus/ButtonSetShopSubScript.h
@@ -0,0 +1,14 @@
+#pragma once
+
+#include "IButtonScript.h"
+
+#include <crepe/api/Script.h>
+
+class ButtonSetShopSubScript : public IButtonScript {
+public:
+ void init() override;
+ bool on_button_press(const crepe::ButtonPressEvent & e);
+
+protected:
+ bool transition = false;
+};
diff --git a/game/menus/ButtonShowCreditsSubScript.cpp b/game/menus/ButtonShowCreditsSubScript.cpp
new file mode 100644
index 0000000..ec0e980
--- /dev/null
+++ b/game/menus/ButtonShowCreditsSubScript.cpp
@@ -0,0 +1,20 @@
+#include "ButtonShowCreditsSubScript.h"
+#include "MenusConfig.h"
+#include "mainmenu/CreditsSubScript.h"
+#include <crepe/api/AudioSource.h>
+#include <crepe/types.h>
+
+using namespace crepe;
+using namespace std;
+
+void ButtonShowCreditsSubScript::init() {
+ IButtonScript::init();
+ this->subscribe<ButtonPressEvent>([this](const ButtonPressEvent & e) {
+ return this->on_button_press(e);
+ });
+}
+
+bool ButtonShowCreditsSubScript::on_button_press(const ButtonPressEvent & e) {
+ this->trigger_event<ShowCreditsEvent>();
+ return false;
+}
diff --git a/game/menus/ButtonShowCreditsSubScript.h b/game/menus/ButtonShowCreditsSubScript.h
new file mode 100644
index 0000000..3c73c44
--- /dev/null
+++ b/game/menus/ButtonShowCreditsSubScript.h
@@ -0,0 +1,14 @@
+#pragma once
+
+#include "IButtonScript.h"
+
+#include <crepe/api/Script.h>
+
+class ButtonShowCreditsSubScript : public IButtonScript {
+public:
+ void init() override;
+ bool on_button_press(const crepe::ButtonPressEvent & e);
+
+protected:
+ bool transition = false;
+};
diff --git a/game/menus/ButtonSubScene.cpp b/game/menus/ButtonSubScene.cpp
new file mode 100644
index 0000000..1fe6b03
--- /dev/null
+++ b/game/menus/ButtonSubScene.cpp
@@ -0,0 +1,266 @@
+#include "ButtonSubScene.h"
+#include "ButtonNextMainMenuSubScript.h"
+#include "ButtonReplaySubScript.h"
+#include "ButtonSetMainMenuSubScript.h"
+#include "ButtonSetShopSubScript.h"
+#include "ButtonShowCreditsSubScript.h"
+#include "IButtonScript.h"
+#include "MenusConfig.h"
+
+#include "../preview/PreviewReplaySubScript.h"
+#include "../preview/PreviewStartRecSubScript.h"
+#include "../preview/PreviewStopRecSubScript.h"
+#include "api/Asset.h"
+#include "mainmenu/ButtonTransitionPreviewSubScript.h"
+
+#include "../Config.h"
+#include "mainmenu/CreditsSubScript.h"
+#include "menus/shop/ButtonBuySelectBubbleScript.h"
+#include "menus/shop/ButtonBuySelectBulletScript.h"
+
+#include <crepe/api/BehaviorScript.h>
+#include <crepe/api/Button.h>
+#include <crepe/api/Color.h>
+#include <crepe/api/Scene.h>
+#include <crepe/api/Sprite.h>
+#include <crepe/api/Text.h>
+
+using namespace crepe;
+using namespace std;
+
+void ButtonSubScene::create(Scene & scn, const Data & data) {
+ GameObject button_object
+ = scn.new_object("button", data.tag, data.position, 0, data.scale);
+ this->set_button_overlay(button_object, data);
+ this->btn_text(button_object, data);
+ this->set_script(button_object, data);
+ this->set_icon(button_object, data);
+}
+
+void ButtonSubScene::btn_text(crepe::GameObject & button_object, const Data & data) {
+
+ crepe::vec2 size = {data.text_width, (data.text_width / data.text.size()) * 2};
+ button_object.add_component<Text>(
+ size, FONT,
+ Text::Data {
+ .world_space = data.worldspace,
+ .text_color = Color::WHITE,
+ },
+ data.text_offset + FONTOFFSET, data.text
+ );
+}
+
+void ButtonSubScene::set_script(crepe::GameObject & button_object, const Data & data) {
+ switch (data.script_type) {
+ case ScriptSelect::PREVIEW:
+ button_object.add_component<BehaviorScript>()
+ .set_script<ButtonTransitionPreviewSubScript>();
+ break;
+ case ScriptSelect::SHOP:
+ button_object.add_component<BehaviorScript>().set_script<ButtonSetShopSubScript>();
+ break;
+ case ScriptSelect::MAINMENU:
+ button_object.add_component<BehaviorScript>()
+ .set_script<ButtonSetMainMenuSubScript>();
+ break;
+ case ScriptSelect::NEXT:
+ button_object.add_component<BehaviorScript>()
+ .set_script<ButtonNextMainMenuSubScript>();
+ break;
+ case ScriptSelect::REPLAY:
+ button_object.add_component<BehaviorScript>().set_script<ButtonReplaySubScript>();
+ break;
+ case ScriptSelect::CREDITS_BACK:
+ button_object.add_component<BehaviorScript>().set_script<CreditsSubScript>(data.tag
+ );
+ break;
+ case ScriptSelect::CREDITS_SHOW:
+ button_object.add_component<BehaviorScript>()
+ .set_script<ButtonShowCreditsSubScript>();
+ break;
+ case ScriptSelect::PREVIEW_REPLAY:
+ button_object.add_component<BehaviorScript>().set_script<PreviewReplaySubScript>();
+ break;
+ case ScriptSelect::PREVIEW_START:
+ button_object.add_component<BehaviorScript>().set_script<PreviewStartRecSubScript>(
+ );
+ break;
+ case ScriptSelect::PREVIEW_STOP:
+ button_object.add_component<BehaviorScript>().set_script<PreviewStopRecSubScript>(
+ );
+ break;
+ case ScriptSelect::SHOP_BULLET:
+ button_object.add_component<BehaviorScript>()
+ .set_script<ButtonBuySelectBulletScript>();
+ break;
+ case ScriptSelect::SHOP_BUBBLE:
+ button_object.add_component<BehaviorScript>()
+ .set_script<ButtonBuySelectBubbleScript>();
+ break;
+ case ScriptSelect::NONE:
+ button_object.add_component<BehaviorScript>().set_script<IButtonScript>();
+ break;
+ }
+}
+
+void ButtonSubScene::set_icon(crepe::GameObject & button_object, const Data & data) {
+ switch (data.icon_type) {
+ case IconSelect::SHOP:
+ button_object.add_component<Sprite>(
+ Asset("asset/ui/buttonCoinsSmall.png"),
+ Sprite::Data {
+ .sorting_in_layer
+ = STARTING_SORTING_IN_LAYER + 3 + data.sorting_layer_offset,
+ .size = ICON_SIZE,
+ .position_offset = data.icon_offset,
+ .world_space = data.worldspace,
+ }
+ );
+ break;
+ case IconSelect::COINS:
+ button_object.add_component<Sprite>(
+ Asset("asset/ui/buttonCoinsSmall.png"),
+ Sprite::Data {
+ .sorting_in_layer
+ = STARTING_SORTING_IN_LAYER + 3 + data.sorting_layer_offset,
+ .size = ICON_SIZE,
+ .position_offset = data.icon_offset,
+ .world_space = data.worldspace,
+ }
+ );
+ break;
+ case IconSelect::NONE:
+ break;
+ }
+}
+
+void ButtonSubScene::set_button_overlay(crepe::GameObject & button_object, const Data & data) {
+ switch (data.button_type) {
+ case ButtonSelect::LARGE:
+ this->large_btn_overlay(button_object, data);
+ break;
+ case ButtonSelect::BACK:
+ this->back_btn_overlay(button_object, data);
+ break;
+ case ButtonSelect::NEXT:
+ this->next_btn_overlay(button_object, data);
+ break;
+ }
+}
+
+void ButtonSubScene::large_btn_overlay(crepe::GameObject & button_object, const Data & data) {
+ button_object.add_component<Sprite>(
+ Asset("asset/ui/buttonBacking.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 1 + data.sorting_layer_offset,
+ .size = LARGE_OVERLAY_SIZE,
+ .world_space = data.worldspace,
+ }
+ );
+ button_object.add_component<Button>(LARGE_OVERLAY_SIZE, Button::Data {});
+ if (!data.color_side) return;
+ this->btn_color_side(button_object, SIDE_PANEL_OFFSET, data);
+}
+
+void ButtonSubScene::back_btn_overlay(crepe::GameObject & button_object, const Data & data) {
+ button_object.add_component<Sprite>(
+ Asset("asset/ui/backbuttonright.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 1 + data.sorting_layer_offset,
+ .size = SMALL_OVERLAY_SIZE_RIGHT,
+ .position_offset = {20, 0},
+ .world_space = data.worldspace,
+ }
+ );
+ button_object.add_component<Sprite>(
+ Asset("asset/ui/backbuttonleft.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 1 + data.sorting_layer_offset,
+ .size = SMALL_OVERLAY_SIZE_LEFT,
+ .position_offset = {-80, 0},
+ .world_space = data.worldspace,
+ }
+ );
+ button_object.add_component<Button>(
+ vec2 {
+ SMALL_OVERLAY_SIZE_LEFT.x + SMALL_OVERLAY_SIZE_RIGHT.x, SMALL_OVERLAY_SIZE_LEFT.y
+ },
+ Button::Data {}
+ );
+}
+
+void ButtonSubScene::next_btn_overlay(crepe::GameObject & button_object, const Data & data) {
+ button_object.add_component<Sprite>(
+ Asset("asset/ui/backbuttonright.png"),
+ Sprite::Data {
+ .flip = {true, false},
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 1 + data.sorting_layer_offset,
+ .size = SMALL_OVERLAY_SIZE_RIGHT,
+ .position_offset = {-20, 0},
+ .world_space = data.worldspace,
+ }
+ );
+ button_object.add_component<Sprite>(
+ Asset("asset/ui/backbuttonleft.png"),
+ Sprite::Data {
+ .flip = {true, false},
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 1 + data.sorting_layer_offset,
+ .size = SMALL_OVERLAY_SIZE_LEFT,
+ .position_offset = {80, 0},
+ .world_space = data.worldspace,
+ }
+ );
+ button_object.add_component<Button>(
+ vec2 {
+ SMALL_OVERLAY_SIZE_LEFT.x + SMALL_OVERLAY_SIZE_RIGHT.x, SMALL_OVERLAY_SIZE_LEFT.y
+ },
+ Button::Data {}
+ );
+}
+
+void ButtonSubScene::btn_color_side(
+ crepe::GameObject & button_object, const vec2 & offset, const Data & data
+) {
+ Asset * selected;
+ Asset blue = Asset("asset/ui/buttonSmallBlue.png");
+ Asset orange = Asset("asset/ui/buttonSmallOrange.png");
+ Asset purple = Asset("asset/ui/buttonSmallPurple.png");
+ Asset yellow = Asset("asset/ui/buttonSmallYellow.png");
+ switch (data.btn_side_color) {
+ case ButtonSideColor::BLUE:
+ selected = &blue;
+ break;
+ case ButtonSideColor::ORANGE:
+ selected = &orange;
+ break;
+ case ButtonSideColor::PURPLE:
+ selected = &purple;
+ break;
+ case ButtonSideColor::YELLOW:
+ selected = &yellow;
+ break;
+ case ButtonSideColor::NONE:
+ selected = &blue;
+ break;
+ }
+
+ button_object.add_component<Sprite>(
+ *selected,
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 2 + data.sorting_layer_offset,
+ .size = SIDE_PANEL_SIZE,
+ .position_offset = offset,
+ .world_space = data.worldspace,
+ }
+ );
+ button_object.add_component<Sprite>(
+ *selected,
+ Sprite::Data {
+ .flip = {true, false},
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 2 + data.sorting_layer_offset,
+ .size = SIDE_PANEL_SIZE,
+ .position_offset = {-offset.x, offset.y},
+ .world_space = data.worldspace,
+ }
+ );
+}
diff --git a/game/menus/ButtonSubScene.h b/game/menus/ButtonSubScene.h
new file mode 100644
index 0000000..d4c7223
--- /dev/null
+++ b/game/menus/ButtonSubScene.h
@@ -0,0 +1,84 @@
+#pragma once
+
+#include <crepe/api/GameObject.h>
+
+#include <string>
+
+namespace crepe {
+class Scene;
+}
+
+class ButtonSubScene {
+public:
+ //script enum
+ enum class ScriptSelect {
+ PREVIEW,
+ SHOP,
+ MAINMENU,
+ NEXT,
+ REPLAY,
+ CREDITS_SHOW,
+ CREDITS_BACK,
+ PREVIEW_START,
+ PREVIEW_STOP,
+ PREVIEW_REPLAY,
+ SHOP_BULLET,
+ SHOP_BUBBLE,
+ NONE,
+ };
+ //icon enum
+ enum class IconSelect {
+ SHOP,
+ COINS,
+ NONE,
+ };
+ //icon enum
+ enum class ButtonSelect {
+ BACK,
+ NEXT,
+ LARGE,
+ };
+
+ enum class ButtonSideColor {
+ BLUE,
+ ORANGE,
+ PURPLE,
+ YELLOW,
+ NONE,
+ };
+ //data struct
+ struct Data {
+ const std::string & text = "NODATA";
+ const crepe::vec2 & text_offset = {0, 0};
+ const float text_width = 200;
+ const crepe::vec2 & icon_offset = {0, 0};
+ const IconSelect icon_type = IconSelect::NONE;
+ const crepe::vec2 & position = {0, 0};
+ const ScriptSelect script_type = ScriptSelect::NONE;
+ const ButtonSelect button_type = ButtonSelect::LARGE;
+ const float scale = 1;
+ const bool worldspace = true;
+ const bool color_side = true;
+ const std::string & tag = "";
+ const int sorting_layer_offset = 0;
+ const ButtonSideColor btn_side_color = ButtonSideColor::NONE;
+ };
+
+public:
+ void create(crepe::Scene & scn, const Data & data);
+
+private:
+ void large_btn_overlay(crepe::GameObject & button_object, const Data & data);
+ void back_btn_overlay(crepe::GameObject & button_object, const Data & data);
+ void next_btn_overlay(crepe::GameObject & button_object, const Data & data);
+ void btn_color_side(
+ crepe::GameObject & button_object, const crepe::vec2 & offset, const Data & data
+ );
+ void btn_text(crepe::GameObject & button_object, const Data & data);
+ void set_script(crepe::GameObject & button_object, const Data & data);
+ void set_icon(crepe::GameObject & button_object, const Data & data);
+ void set_button_overlay(crepe::GameObject & button_object, const Data & data);
+
+private:
+ static constexpr crepe::vec2 SIDE_PANEL_OFFSET = {113, 0};
+};
diff --git a/game/menus/FloatingWindowSubScene.cpp b/game/menus/FloatingWindowSubScene.cpp
new file mode 100644
index 0000000..4420bfa
--- /dev/null
+++ b/game/menus/FloatingWindowSubScene.cpp
@@ -0,0 +1,220 @@
+
+#include "FloatingWindowSubScene.h"
+#include "MenusConfig.h"
+
+#include <crepe/api/Camera.h>
+#include <crepe/api/GameObject.h>
+#include <crepe/api/Scene.h>
+#include <crepe/api/Sprite.h>
+#include <crepe/types.h>
+
+using namespace crepe;
+using namespace std;
+
+void FloatingWindowSubScene::create(Scene & scn, const Data & data) {
+ const vec2 SIZE = {data.width, data.width * 0.75f};
+ const vec2 POSITION_CORRECTION = vec2 {0, -SIZE.y / 2} + data.offset;
+ const float THICKNESS_BANNER = 34;
+ const float MIDDLE_OFFSET_FACTOR_TICKNESS = 0.83;
+ const float MIDDLE_OFFSET_FACTOR_OFFSET = 1.2;
+ const float MIDDLE_OFFSET_FACTOR_MIDDLE_WIDTH = 0.86;
+ const float MIDDLE_OFFSET_OFFSET_ADDITION = -0.5;
+ const float BOTTOM_OFFSET_X = 3;
+ const float BOTTOM_OFFSET_Y = -3;
+
+ GameObject floatingwindow = scn.new_object("FloatingWindow", data.group_tag);
+
+ // Top_middle
+ floatingwindow.add_component<Sprite>(
+ Asset("asset/ui/settings_container/top_middle_setting.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 8,
+ .size = {SIZE.x, THICKNESS_BANNER},
+ .position_offset = POSITION_CORRECTION + vec2 {0, 0},
+ .world_space = false,
+ }
+ );
+
+ // Top_Left
+ floatingwindow.add_component<Sprite>(
+ Asset("asset/ui/settings_container/top_left_setting.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 8,
+ .size = {THICKNESS_BANNER, THICKNESS_BANNER},
+ .position_offset
+ = POSITION_CORRECTION + vec2 {-SIZE.x / 2 - THICKNESS_BANNER / 2, 0},
+ .world_space = false,
+ }
+ );
+
+ // Top_Right
+ floatingwindow.add_component<Sprite>(
+ Asset("asset/ui/settings_container/top_right_setting.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 8,
+ .size = {THICKNESS_BANNER, THICKNESS_BANNER},
+ .position_offset
+ = POSITION_CORRECTION + vec2 {SIZE.x / 2 + THICKNESS_BANNER / 2, 0},
+ .world_space = false,
+ }
+ );
+
+ // Top_middle_2
+ floatingwindow.add_component<Sprite>(
+ Asset("asset/ui/settings_container/top_2_middle_setting.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 8,
+ .size = {SIZE.x, THICKNESS_BANNER},
+ .position_offset = POSITION_CORRECTION + vec2 {0, THICKNESS_BANNER},
+ .world_space = false,
+ }
+ );
+
+ // Top_Left_2
+ floatingwindow.add_component<Sprite>(
+ Asset("asset/ui/settings_container/top_2_left_setting.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 8,
+ .size = {THICKNESS_BANNER, THICKNESS_BANNER},
+ .position_offset = POSITION_CORRECTION
+ + vec2 {-SIZE.x / 2 - THICKNESS_BANNER / 2, THICKNESS_BANNER},
+ .world_space = false,
+ }
+ );
+
+ // Top_Right_2
+ floatingwindow.add_component<Sprite>(
+ Asset("asset/ui/settings_container/top_2_right_setting.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 8,
+ .size = {THICKNESS_BANNER, THICKNESS_BANNER},
+ .position_offset
+ = POSITION_CORRECTION + vec2 {SIZE.x / 2 + THICKNESS_BANNER / 2, THICKNESS_BANNER},
+ .world_space = false,
+ }
+ );
+
+ // Top_middle_3
+ floatingwindow.add_component<Sprite>(
+ Asset("asset/ui/settings_container/top_3_middle_setting.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 8,
+ .size = {SIZE.x, THICKNESS_BANNER},
+ .position_offset = POSITION_CORRECTION + vec2 {0, THICKNESS_BANNER * 2},
+ .world_space = false,
+ }
+ );
+
+ // Top_Left_3
+ floatingwindow.add_component<Sprite>(
+ Asset("asset/ui/settings_container/top_3_left_setting.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 8,
+ .size = {THICKNESS_BANNER, THICKNESS_BANNER},
+ .position_offset
+ = POSITION_CORRECTION
+ + vec2 {-SIZE.x / 2 - THICKNESS_BANNER / 2, THICKNESS_BANNER * 2},
+ .world_space = false,
+ }
+ );
+
+ // Top_Right_3
+ floatingwindow.add_component<Sprite>(
+ Asset("asset/ui/settings_container/top_3_right_setting.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 8,
+ .size = {THICKNESS_BANNER, THICKNESS_BANNER},
+ .position_offset
+ = POSITION_CORRECTION
+ + vec2 {SIZE.x / 2 + THICKNESS_BANNER / 2, THICKNESS_BANNER * 2},
+ .world_space = false,
+ }
+ );
+
+ // Middle_Mid
+ floatingwindow.add_component<Sprite>(
+ Asset("asset/ui/settings_container/middle_mid_setting.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 7,
+ .size
+ = {SIZE.x * MIDDLE_OFFSET_FACTOR_OFFSET * MIDDLE_OFFSET_FACTOR_MIDDLE_WIDTH
+ + data.width_middle_offset,
+ SIZE.y},
+ .position_offset
+ = POSITION_CORRECTION
+ + vec2 {0, THICKNESS_BANNER * 3 + SIZE.y / 2 - THICKNESS_BANNER / 2},
+ .world_space = false,
+ }
+ );
+
+ // Middle_Left
+ floatingwindow.add_component<Sprite>(
+ Asset("asset/ui/settings_container/middle_left_setting.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 8,
+ .size = {THICKNESS_BANNER * MIDDLE_OFFSET_FACTOR_TICKNESS, SIZE.y},
+ .position_offset
+ = POSITION_CORRECTION
+ + vec2 {-SIZE.x / 2 - THICKNESS_BANNER / 2 * MIDDLE_OFFSET_FACTOR_OFFSET - MIDDLE_OFFSET_OFFSET_ADDITION, THICKNESS_BANNER * 3 + SIZE.y / 2 - THICKNESS_BANNER / 2},
+ .world_space = false,
+ }
+ );
+
+ // Middle_Right
+ floatingwindow.add_component<Sprite>(
+ Asset("asset/ui/settings_container/middle_right_setting.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 8,
+ .size = {THICKNESS_BANNER * MIDDLE_OFFSET_FACTOR_TICKNESS, SIZE.y},
+ .position_offset
+ = POSITION_CORRECTION
+ + vec2 {SIZE.x / 2 + THICKNESS_BANNER / 2 * MIDDLE_OFFSET_FACTOR_OFFSET + MIDDLE_OFFSET_OFFSET_ADDITION, THICKNESS_BANNER * 3 + SIZE.y / 2 - THICKNESS_BANNER / 2},
+ .world_space = false,
+ }
+ );
+
+ // Bot_Middle
+ floatingwindow.add_component<Sprite>(
+ Asset("asset/ui/settings_container/bot_middle_setting.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 7,
+ .size
+ = {SIZE.x * MIDDLE_OFFSET_FACTOR_OFFSET * MIDDLE_OFFSET_FACTOR_MIDDLE_WIDTH
+ + data.width_middle_offset,
+ THICKNESS_BANNER * MIDDLE_OFFSET_FACTOR_TICKNESS},
+ .position_offset
+ = POSITION_CORRECTION + vec2 {0, THICKNESS_BANNER * 3 + SIZE.y + BOTTOM_OFFSET_Y},
+ .world_space = false,
+ }
+ );
+
+ // Bot_Left
+ floatingwindow.add_component<Sprite>(
+ Asset("asset/ui/settings_container/bot_left_setting.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 8,
+ .size
+ = {THICKNESS_BANNER * MIDDLE_OFFSET_FACTOR_TICKNESS,
+ THICKNESS_BANNER * MIDDLE_OFFSET_FACTOR_TICKNESS},
+ .position_offset
+ = POSITION_CORRECTION
+ + vec2 {-SIZE.x / 2 - THICKNESS_BANNER / 2 - BOTTOM_OFFSET_X, THICKNESS_BANNER * 3 + SIZE.y + BOTTOM_OFFSET_Y},
+ .world_space = false,
+ }
+ );
+
+ // Bot_Right
+ floatingwindow.add_component<Sprite>(
+ Asset("asset/ui/settings_container/bot_right_setting.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 8,
+ .size
+ = {THICKNESS_BANNER * MIDDLE_OFFSET_FACTOR_TICKNESS,
+ THICKNESS_BANNER * MIDDLE_OFFSET_FACTOR_TICKNESS},
+ .position_offset
+ = POSITION_CORRECTION
+ + vec2 {SIZE.x / 2 + THICKNESS_BANNER / 2 + BOTTOM_OFFSET_X, THICKNESS_BANNER * 3 + SIZE.y + BOTTOM_OFFSET_Y},
+ .world_space = false,
+ }
+ );
+}
diff --git a/game/menus/FloatingWindowSubScene.h b/game/menus/FloatingWindowSubScene.h
new file mode 100644
index 0000000..7b9de96
--- /dev/null
+++ b/game/menus/FloatingWindowSubScene.h
@@ -0,0 +1,17 @@
+#pragma once
+
+#include <crepe/api/Scene.h>
+#include <crepe/types.h>
+
+class FloatingWindowSubScene {
+public:
+ struct Data {
+ const std::string group_tag = "";
+ float width = 200;
+ crepe::vec2 offset = {0, 0};
+ float width_middle_offset = 0;
+ };
+
+public:
+ void create(crepe::Scene & scn, const Data & data);
+};
diff --git a/game/menus/IButtonScript.cpp b/game/menus/IButtonScript.cpp
new file mode 100644
index 0000000..34efbd0
--- /dev/null
+++ b/game/menus/IButtonScript.cpp
@@ -0,0 +1,32 @@
+#include "IButtonScript.h"
+
+#include "system/InputSystem.h"
+
+#include <crepe/api/Sprite.h>
+#include <crepe/types.h>
+
+using namespace crepe;
+using namespace std;
+
+void IButtonScript::init() {
+ this->subscribe<ButtonExitEvent>([this](const ButtonExitEvent & e) {
+ return this->on_button_exit(e);
+ });
+ this->subscribe<ButtonEnterEvent>([this](const ButtonEnterEvent & e) {
+ return this->on_button_enter(e);
+ });
+}
+bool IButtonScript::on_button_exit(const ButtonExitEvent & e) {
+ RefVector<Sprite> sprites = this->get_components<Sprite>();
+ for (Sprite & sprite : sprites) {
+ sprite.data.color = Color {255, 255, 255, 255};
+ }
+ return false;
+}
+bool IButtonScript::on_button_enter(const ButtonEnterEvent & e) {
+ RefVector<Sprite> sprites = this->get_components<Sprite>();
+ for (Sprite & sprite : sprites) {
+ sprite.data.color = Color {200, 200, 200, 255};
+ }
+ return false;
+}
diff --git a/game/menus/IButtonScript.h b/game/menus/IButtonScript.h
new file mode 100644
index 0000000..e45375b
--- /dev/null
+++ b/game/menus/IButtonScript.h
@@ -0,0 +1,10 @@
+#pragma once
+
+#include <crepe/api/Script.h>
+
+class IButtonScript : public virtual crepe::Script {
+public:
+ virtual void init();
+ virtual bool on_button_exit(const crepe::ButtonExitEvent & e);
+ virtual bool on_button_enter(const crepe::ButtonEnterEvent & e);
+};
diff --git a/game/menus/IFloatingWindowScript.cpp b/game/menus/IFloatingWindowScript.cpp
new file mode 100644
index 0000000..4b538ef
--- /dev/null
+++ b/game/menus/IFloatingWindowScript.cpp
@@ -0,0 +1,22 @@
+#include "IFloatingWindowScript.h"
+
+#include <crepe/api/Sprite.h>
+#include <crepe/types.h>
+
+using namespace crepe;
+
+void IFloatingWindowScript::init() { this->disable_all_sprites(); }
+
+void IFloatingWindowScript::disable_all_sprites() {
+ RefVector<Sprite> sprites = this->get_components_by_tag<Sprite>(this->tag);
+ for (Sprite & sprite : sprites) {
+ sprite.active = false;
+ }
+}
+
+void IFloatingWindowScript::enable_all_sprites() {
+ RefVector<Sprite> sprites = this->get_components_by_tag<Sprite>(this->tag);
+ for (Sprite & sprite : sprites) {
+ sprite.active = true;
+ }
+}
diff --git a/game/menus/IFloatingWindowScript.h b/game/menus/IFloatingWindowScript.h
new file mode 100644
index 0000000..e39378f
--- /dev/null
+++ b/game/menus/IFloatingWindowScript.h
@@ -0,0 +1,15 @@
+#pragma once
+
+#include <crepe/api/Script.h>
+
+#include <string>
+
+class IFloatingWindowScript : public virtual crepe::Script {
+public:
+ virtual void init();
+ void disable_all_sprites();
+ void enable_all_sprites();
+
+protected:
+ std::string tag = "";
+};
diff --git a/game/menus/MenusConfig.h b/game/menus/MenusConfig.h
new file mode 100644
index 0000000..3e357a5
--- /dev/null
+++ b/game/menus/MenusConfig.h
@@ -0,0 +1,16 @@
+#pragma once
+#include <crepe/types.h>
+
+//generic menu config
+static constexpr int STARTING_SORTING_IN_LAYER = 7;
+//Scene names
+static constexpr const char * START_SCENE = "scene1";
+static constexpr const char * PREVIEW_SCENE = "preview scene";
+static constexpr const char * SHOP_SCENE = "shopmenu";
+static constexpr const char * MAINMENU_SCENE = "mainmenu";
+//button config
+static constexpr crepe::vec2 LARGE_OVERLAY_SIZE = {250, 100};
+static constexpr crepe::vec2 SMALL_OVERLAY_SIZE_RIGHT = {150, 100};
+static constexpr crepe::vec2 SMALL_OVERLAY_SIZE_LEFT = {50, 100};
+static constexpr crepe::vec2 SIDE_PANEL_SIZE = {50, 150};
+static constexpr crepe::vec2 ICON_SIZE = {50, 50};
diff --git a/game/menus/endgame/EndGameSubScene.cpp b/game/menus/endgame/EndGameSubScene.cpp
new file mode 100644
index 0000000..b33072a
--- /dev/null
+++ b/game/menus/endgame/EndGameSubScene.cpp
@@ -0,0 +1,127 @@
+
+#include "EndGameSubScene.h"
+#include "EndGameSubScript.h"
+
+#include "../../Config.h"
+#include "../ButtonSubScene.h"
+#include "../FloatingWindowSubScene.h"
+
+#include <string>
+
+#include <crepe/api/BehaviorScript.h>
+#include <crepe/api/GameObject.h>
+#include <crepe/api/Text.h>
+#include <crepe/types.h>
+
+using namespace crepe;
+using namespace std;
+
+void EndGameSubScene::create(Scene & scn) {
+
+ const std::string TAG = "end_game_tag";
+ GameObject script = scn.new_object("script");
+ script.add_component<BehaviorScript>().set_script<EndGameSubScript>(TAG);
+
+ // Window
+ FloatingWindowSubScene window;
+ window.create(
+ scn,
+ FloatingWindowSubScene::Data {
+ .group_tag = TAG,
+ .width = 500,
+ .offset = {0, -50},
+ .width_middle_offset = -2,
+ }
+ );
+
+ // Titel
+ const string TITEL_STRING = "GAME OVER";
+ GameObject titel = scn.new_object("titel", TAG);
+ crepe::vec2 size = {200, (200.0f / TITEL_STRING.size()) * 2};
+ titel.add_component<Text>(
+ size, FONT,
+ Text::Data {
+ .world_space = false,
+ .text_color = Color::WHITE,
+ },
+ vec2 {0, -207} + FONTOFFSET, TITEL_STRING
+ );
+
+ const float Y_SPACING = 50;
+ const float Y_OFFSET = -100;
+
+ // Gold gathered
+ const string GOLD_STRING = "gold:0";
+ GameObject gold = scn.new_object("gold_endgame", TAG);
+ crepe::vec2 size_gold = {200, (200.0f / GOLD_STRING.size()) * 2};
+ gold.add_component<Text>(
+ size_gold, FONT,
+ Text::Data {
+ .world_space = false,
+ .text_color = Color::GOLD,
+ },
+ vec2 {0, Y_OFFSET} + FONTOFFSET, GOLD_STRING
+ );
+
+ // Distance
+ const string DISTANCE_STRING = "0M";
+ GameObject distance = scn.new_object("distance_endgame", TAG);
+ crepe::vec2 size_distance = {200, (200.0f / DISTANCE_STRING.size()) * 2};
+ distance.add_component<Text>(
+ size_distance, FONT,
+ Text::Data {
+ .world_space = false,
+ .text_color = Color::WHITE,
+ },
+ vec2 {0, Y_SPACING + Y_OFFSET} + FONTOFFSET, DISTANCE_STRING
+ );
+
+ // Highscore
+ const string HIGHSCORE_STRING = "NEW HIGHSCORE";
+ GameObject highscore = scn.new_object("highscore_endgame", "highscore_tag_end");
+ crepe::vec2 size_highscore = {200, (200.0f / HIGHSCORE_STRING.size()) * 2};
+ highscore
+ .add_component<Text>(
+ size_highscore, FONT,
+ Text::Data {
+ .world_space = false,
+ .text_color = Color::WHITE,
+ },
+ vec2 {0, Y_SPACING * 2 + Y_OFFSET} + FONTOFFSET, HIGHSCORE_STRING
+ )
+ .active
+ = false;
+
+ // Buttons
+ vec2 button_position = {190, 190};
+ ButtonSubScene button;
+ button.create(
+ scn,
+ ButtonSubScene::Data {
+ .text = "NEXT",
+ .text_width = 100,
+ .position = button_position,
+ .script_type = ButtonSubScene::ScriptSelect::NEXT,
+ .button_type = ButtonSubScene::ButtonSelect::NEXT,
+ .scale = 0.6,
+ .worldspace = false,
+ .tag = TAG,
+ .sorting_layer_offset = 20,
+ }
+ );
+
+ button.create(
+ scn,
+ ButtonSubScene::Data {
+ .text = "REPLAY",
+ .text_width = 150,
+ .position = {-button_position.x, button_position.y},
+ .script_type = ButtonSubScene::ScriptSelect::REPLAY,
+ .button_type = ButtonSubScene::ButtonSelect::BACK,
+ .scale = 0.6,
+ .worldspace = false,
+ .tag = TAG,
+ .sorting_layer_offset = 20,
+ }
+ );
+}
diff --git a/game/menus/endgame/EndGameSubScene.h b/game/menus/endgame/EndGameSubScene.h
new file mode 100644
index 0000000..204f3b7
--- /dev/null
+++ b/game/menus/endgame/EndGameSubScene.h
@@ -0,0 +1,9 @@
+#pragma once
+
+#include <crepe/api/Scene.h>
+
+class EndGameSubScene {
+
+public:
+ void create(crepe::Scene & scn);
+};
diff --git a/game/menus/endgame/EndGameSubScript.cpp b/game/menus/endgame/EndGameSubScript.cpp
new file mode 100644
index 0000000..6793f3e
--- /dev/null
+++ b/game/menus/endgame/EndGameSubScript.cpp
@@ -0,0 +1,101 @@
+#include "EndGameSubScript.h"
+
+#include "../../Config.h"
+#include "../../Events.h"
+#include "../ButtonReplaySubScript.h"
+#include "../IFloatingWindowScript.h"
+#include "ValueBroker.h"
+#include "manager/SaveManager.h"
+
+#include <string>
+
+#include <crepe/api/Button.h>
+#include <crepe/api/Sprite.h>
+#include <crepe/api/Text.h>
+#include <crepe/types.h>
+
+using namespace crepe;
+
+EndGameSubScript::EndGameSubScript(const std::string & tag) { this->tag = tag; }
+
+void EndGameSubScript::init() {
+ this->disable_all();
+ this->subscribe<EndGameEvent>([this](const EndGameEvent e) { return this->enable_all(); });
+ this->subscribe<EndGameEvent>([this](const EndGameEvent e) {
+ return this->reset_timescale();
+ });
+ this->subscribe<ShowScoreEvent>([this](const ShowScoreEvent e) {
+ return this->showscore();
+ });
+}
+
+bool EndGameSubScript::disable_all() {
+ IFloatingWindowScript::disable_all_sprites();
+ RefVector<Button> buttons = this->get_components_by_tag<Button>(this->tag);
+ for (Button & button : buttons) {
+ button.active = false;
+ }
+ RefVector<Text> texts = this->get_components_by_tag<Text>(this->tag);
+ for (Text & text : texts) {
+ text.active = false;
+ }
+ return false;
+}
+
+bool EndGameSubScript::enable_all() {
+ IFloatingWindowScript::enable_all_sprites();
+ RefVector<Button> buttons = this->get_components_by_tag<Button>(this->tag);
+ for (Button & button : buttons) {
+ button.active = true;
+ }
+ RefVector<Text> texts = this->get_components_by_tag<Text>(this->tag);
+ for (Text & text : texts) {
+ text.active = true;
+ }
+ return false;
+}
+
+bool EndGameSubScript::reset_timescale() {
+ this->get_loop_timer().set_time_scale(1);
+ return false;
+}
+
+bool EndGameSubScript::showscore() {
+ // Gather text
+ Text & coins_text = this->get_components_by_name<Text>("gold_endgame").front().get();
+ Text & distance_text
+ = this->get_components_by_name<Text>("distance_endgame").front().get();
+ Text & highscore_text
+ = this->get_components_by_name<Text>("highscore_endgame").front().get();
+ highscore_text.active = false;
+
+ // Gather saved data
+ SaveManager & savemgr = this->get_save_manager();
+ ValueBroker<std::string> coins = savemgr.get<std::string>(TOTAL_COINS_RUN, "0");
+ ValueBroker<std::string> distance = savemgr.get<std::string>(DISTANCE_RUN, "0");
+ int distance_run = savemgr.get<int>(DISTANCE_RUN, 0).get();
+ int distance_game = savemgr.get<int>(DISTANCE_GAME, 0).get();
+
+ // Show highscore
+ if (distance_run > distance_game) highscore_text.active = true;
+
+ const float CHAR_SIZE_DIS = 20;
+ // Show distance
+ std::string distance_string = "DISTANCE:" + distance.get();
+ distance_text.text = distance_string;
+ crepe::vec2 size_distance
+ = {CHAR_SIZE_DIS * distance_string.size(),
+ (CHAR_SIZE_DIS * distance_string.size() / distance_string.size()) * 2};
+ distance_text.dimensions = size_distance;
+
+ const float CHAR_SIZE_COIN = 16;
+ // Show coins
+ std::string coins_string = "Coins:" + coins.get();
+ coins_text.text = coins_string;
+ crepe::vec2 size_coins
+ = {CHAR_SIZE_COIN * coins_string.size(),
+ (CHAR_SIZE_COIN * coins_string.size() / coins_string.size()) * 2};
+ coins_text.dimensions = size_coins;
+
+ return false;
+}
diff --git a/game/menus/endgame/EndGameSubScript.h b/game/menus/endgame/EndGameSubScript.h
new file mode 100644
index 0000000..c25f6ca
--- /dev/null
+++ b/game/menus/endgame/EndGameSubScript.h
@@ -0,0 +1,18 @@
+#pragma once
+
+#include "../IFloatingWindowScript.h"
+
+#include <crepe/api/Event.h>
+#include <crepe/api/Script.h>
+
+struct ShowScoreEvent : public crepe::Event {};
+
+class EndGameSubScript : public IFloatingWindowScript {
+public:
+ EndGameSubScript(const std::string & tag);
+ void init() override;
+ bool disable_all();
+ bool enable_all();
+ bool reset_timescale();
+ bool showscore();
+};
diff --git a/game/menus/mainmenu/ButtonTransitionPreviewSubScript.cpp b/game/menus/mainmenu/ButtonTransitionPreviewSubScript.cpp
new file mode 100644
index 0000000..4c4dfc1
--- /dev/null
+++ b/game/menus/mainmenu/ButtonTransitionPreviewSubScript.cpp
@@ -0,0 +1,23 @@
+#include "ButtonTransitionPreviewSubScript.h"
+
+#include "../MenusConfig.h"
+
+using namespace crepe;
+using namespace std;
+
+void ButtonTransitionPreviewSubScript::init() {
+ IButtonScript::init();
+ this->subscribe<ButtonPressEvent>([this](const ButtonPressEvent & e) {
+ return this->on_button_press(e);
+ });
+}
+
+bool ButtonTransitionPreviewSubScript::on_button_press(const ButtonPressEvent & e) {
+ if (!this->transition) this->transition = true;
+ return false;
+}
+
+const char * ButtonTransitionPreviewSubScript::get_scene_name() const {
+ // Provide the next scene defined in MainMenuConfig
+ return PREVIEW_SCENE;
+}
diff --git a/game/menus/mainmenu/ButtonTransitionPreviewSubScript.h b/game/menus/mainmenu/ButtonTransitionPreviewSubScript.h
new file mode 100644
index 0000000..d6d8149
--- /dev/null
+++ b/game/menus/mainmenu/ButtonTransitionPreviewSubScript.h
@@ -0,0 +1,12 @@
+#pragma once
+
+#include "ITransitionScript.h"
+
+#include "../IButtonScript.h"
+
+class ButtonTransitionPreviewSubScript : public ITransitionScript, public IButtonScript {
+public:
+ void init() override;
+ bool on_button_press(const crepe::ButtonPressEvent & e);
+ const char * get_scene_name() const override;
+};
diff --git a/game/menus/mainmenu/CreditsSubScene.cpp b/game/menus/mainmenu/CreditsSubScene.cpp
new file mode 100644
index 0000000..65576ee
--- /dev/null
+++ b/game/menus/mainmenu/CreditsSubScene.cpp
@@ -0,0 +1,132 @@
+
+#include "CreditsSubScene.h"
+#include "CreditsSubScript.h"
+
+#include "../../Config.h"
+#include "../ButtonSubScene.h"
+#include "../FloatingWindowSubScene.h"
+
+#include <string>
+
+#include <crepe/api/BehaviorScript.h>
+#include <crepe/api/GameObject.h>
+#include <crepe/api/Text.h>
+#include <crepe/types.h>
+
+using namespace crepe;
+using namespace std;
+
+void CreditsSubScene::create(Scene & scn) {
+
+ const std::string TAG = "credits_tag";
+ GameObject script = scn.new_object("script");
+ script.add_component<BehaviorScript>().set_script<CreditsSubScript>(TAG);
+
+ // Window
+ FloatingWindowSubScene window;
+ window.create(
+ scn,
+ FloatingWindowSubScene::Data {
+ .group_tag = TAG,
+ .width = 500,
+ .offset = {150, -50},
+ .width_middle_offset = -2,
+ }
+ );
+
+ // Titel
+ const string TITEL_STRING = "Credits";
+ GameObject titel = scn.new_object("titel", TAG);
+ crepe::vec2 size = {200, (200.0f / TITEL_STRING.size()) * 2};
+ titel.add_component<Text>(
+ size, FONT,
+ Text::Data {
+ .world_space = false,
+ .text_color = Color::WHITE,
+ },
+ vec2 {150, -207} + FONTOFFSET, TITEL_STRING
+ );
+
+ // Buttons
+ vec2 button_position = {190, 190};
+ ButtonSubScene button;
+ button.create(
+ scn,
+ ButtonSubScene::Data {
+ .text = "Back",
+ .text_width = 150,
+ .position = {-button_position.x + 150, button_position.y},
+ .script_type = ButtonSubScene::ScriptSelect::CREDITS_BACK,
+ .button_type = ButtonSubScene::ButtonSelect::BACK,
+ .scale = 0.6,
+ .worldspace = false,
+ .tag = TAG,
+ .sorting_layer_offset = 20,
+ }
+ );
+
+ const float SIZE_CHAR_NAMES = 10;
+ const float Y_OFFSET_NAMES_BEGIN = 100;
+ const float Y_OFFSET_NAMES = 30;
+ const string LOEK = "Loek Le Blansch";
+ crepe::vec2 size_loek
+ = {LOEK.size() * SIZE_CHAR_NAMES, (LOEK.size() * SIZE_CHAR_NAMES / LOEK.size()) * 2};
+ titel.add_component<Text>(
+ size_loek, FONT,
+ Text::Data {
+ .world_space = false,
+ .text_color = Color::WHITE,
+ },
+ vec2 {150, -207 + Y_OFFSET_NAMES + Y_OFFSET_NAMES_BEGIN} + FONTOFFSET, LOEK
+ );
+
+ const string WOUTER = "Wouter Boerenkamps";
+ crepe::vec2 size_wouter
+ = {WOUTER.size() * SIZE_CHAR_NAMES,
+ (WOUTER.size() * SIZE_CHAR_NAMES / WOUTER.size()) * 2};
+ titel.add_component<Text>(
+ size_wouter, FONT,
+ Text::Data {
+ .world_space = false,
+ .text_color = Color::WHITE,
+ },
+ vec2 {150, -207 + Y_OFFSET_NAMES * 2 + Y_OFFSET_NAMES_BEGIN} + FONTOFFSET, WOUTER
+ );
+
+ const string JARO = "Jaro Rutjes";
+ crepe::vec2 size_jaro
+ = {JARO.size() * SIZE_CHAR_NAMES, (JARO.size() * SIZE_CHAR_NAMES / JARO.size()) * 2};
+ titel.add_component<Text>(
+ size_jaro, FONT,
+ Text::Data {
+ .world_space = false,
+ .text_color = Color::WHITE,
+ },
+ vec2 {150, -207 + Y_OFFSET_NAMES * 3 + Y_OFFSET_NAMES_BEGIN} + FONTOFFSET, JARO
+ );
+
+ const string MAX = "Max Smits";
+ crepe::vec2 size_max
+ = {MAX.size() * SIZE_CHAR_NAMES, (MAX.size() * SIZE_CHAR_NAMES / MAX.size()) * 2};
+ titel.add_component<Text>(
+ size_max, FONT,
+ Text::Data {
+ .world_space = false,
+ .text_color = Color::WHITE,
+ },
+ vec2 {150, -207 + Y_OFFSET_NAMES * 4 + Y_OFFSET_NAMES_BEGIN} + FONTOFFSET, MAX
+ );
+
+ const string NIELS = "Niels Stunnebrink";
+ crepe::vec2 size_niels
+ = {NIELS.size() * SIZE_CHAR_NAMES, (NIELS.size() * SIZE_CHAR_NAMES / NIELS.size()) * 2
+ };
+ titel.add_component<Text>(
+ size_niels, FONT,
+ Text::Data {
+ .world_space = false,
+ .text_color = Color::WHITE,
+ },
+ vec2 {150, -207 + Y_OFFSET_NAMES * 5 + Y_OFFSET_NAMES_BEGIN} + FONTOFFSET, NIELS
+ );
+}
diff --git a/game/menus/mainmenu/CreditsSubScene.h b/game/menus/mainmenu/CreditsSubScene.h
new file mode 100644
index 0000000..e7ff735
--- /dev/null
+++ b/game/menus/mainmenu/CreditsSubScene.h
@@ -0,0 +1,9 @@
+#pragma once
+
+#include <crepe/api/Scene.h>
+
+class CreditsSubScene {
+
+public:
+ void create(crepe::Scene & scn);
+};
diff --git a/game/menus/mainmenu/CreditsSubScript.cpp b/game/menus/mainmenu/CreditsSubScript.cpp
new file mode 100644
index 0000000..4224dc8
--- /dev/null
+++ b/game/menus/mainmenu/CreditsSubScript.cpp
@@ -0,0 +1,58 @@
+#include "CreditsSubScript.h"
+
+#include "../../Events.h"
+#include "../ButtonReplaySubScript.h"
+#include "../IFloatingWindowScript.h"
+
+#include <string>
+
+#include <crepe/api/Button.h>
+#include <crepe/api/Sprite.h>
+#include <crepe/api/Text.h>
+#include <crepe/types.h>
+
+using namespace crepe;
+
+CreditsSubScript::CreditsSubScript(const std::string & tag) { this->tag = tag; }
+
+void CreditsSubScript::init() {
+ IButtonScript::init();
+ this->subscribe<ButtonPressEvent>([this](const ButtonPressEvent & e) {
+ return this->on_button_press(e);
+ });
+ this->subscribe<ShowCreditsEvent>([this](const ShowCreditsEvent & e) {
+ this->enable_all();
+ return false;
+ });
+ this->disable_all();
+}
+
+bool CreditsSubScript::disable_all() {
+ IFloatingWindowScript::disable_all_sprites();
+ RefVector<Button> buttons = this->get_components_by_tag<Button>(this->tag);
+ for (Button & button : buttons) {
+ button.active = false;
+ }
+ RefVector<Text> texts = this->get_components_by_tag<Text>(this->tag);
+ for (Text & text : texts) {
+ text.active = false;
+ }
+ return false;
+}
+
+bool CreditsSubScript::enable_all() {
+ IFloatingWindowScript::enable_all_sprites();
+ RefVector<Button> buttons = this->get_components_by_tag<Button>(this->tag);
+ for (Button & button : buttons) {
+ button.active = true;
+ }
+ RefVector<Text> texts = this->get_components_by_tag<Text>(this->tag);
+ for (Text & text : texts) {
+ text.active = true;
+ }
+ return false;
+}
+
+bool CreditsSubScript::on_button_press(const ButtonPressEvent & e) {
+ return this->disable_all();
+}
diff --git a/game/menus/mainmenu/CreditsSubScript.h b/game/menus/mainmenu/CreditsSubScript.h
new file mode 100644
index 0000000..81f941a
--- /dev/null
+++ b/game/menus/mainmenu/CreditsSubScript.h
@@ -0,0 +1,18 @@
+#pragma once
+
+#include "../IButtonScript.h"
+#include "../IFloatingWindowScript.h"
+
+#include <crepe/api/Event.h>
+#include <crepe/api/Script.h>
+
+struct ShowCreditsEvent : public crepe::Event {};
+
+class CreditsSubScript : public IFloatingWindowScript, public IButtonScript {
+public:
+ CreditsSubScript(const std::string & tag);
+ void init() override;
+ bool disable_all();
+ bool enable_all();
+ bool on_button_press(const crepe::ButtonPressEvent & e);
+};
diff --git a/game/menus/mainmenu/ITransitionScript.cpp b/game/menus/mainmenu/ITransitionScript.cpp
new file mode 100644
index 0000000..54b0875
--- /dev/null
+++ b/game/menus/mainmenu/ITransitionScript.cpp
@@ -0,0 +1,30 @@
+#include "ITransitionScript.h"
+#include "MainMenuConfig.h"
+
+#include "../../Config.h"
+#include "../MenusConfig.h"
+
+#include <crepe/api/Camera.h>
+#include <crepe/api/Transform.h>
+#include <crepe/types.h>
+
+using namespace crepe;
+using namespace std;
+
+void ITransitionScript::frame_update(crepe::duration_t delta_time) {
+ if (this->transition) {
+ // cout << "transition:" << velocity << std::endl;
+ Transform & cam = this->get_components_by_name<Transform>(CAMERA_NAME).front();
+ RefVector<Transform> info_tf = this->get_components_by_tag<Transform>(MENU_INFO_TAG);
+ for (Transform & tf : info_tf) {
+ tf.position.y -= VELOCITY_INFO_UP * delta_time.count();
+ }
+ if (velocity < VELOCITY_MAX && cam.position.x < SLOW_DOWN)
+ velocity += VELOCITY_STEP * delta_time.count();
+ else if (velocity > 20) velocity -= VELOCITY_STEP * delta_time.count();
+ if (cam.position.x < END) cam.position.x += (velocity * delta_time.count());
+ if (cam.position.x >= END) {
+ this->set_next_scene(this->get_scene_name());
+ }
+ }
+}
diff --git a/game/menus/mainmenu/ITransitionScript.h b/game/menus/mainmenu/ITransitionScript.h
new file mode 100644
index 0000000..9a2ef90
--- /dev/null
+++ b/game/menus/mainmenu/ITransitionScript.h
@@ -0,0 +1,15 @@
+#pragma once
+
+#include <crepe/api/Script.h>
+
+class ITransitionScript : public virtual crepe::Script {
+public:
+ void frame_update(crepe::duration_t delta_time) override;
+ virtual const char * get_scene_name() const = 0;
+
+private:
+ float velocity = 20;
+
+protected:
+ bool transition = false;
+};
diff --git a/game/menus/mainmenu/MainMenuConfig.h b/game/menus/mainmenu/MainMenuConfig.h
new file mode 100644
index 0000000..f4ca5a6
--- /dev/null
+++ b/game/menus/mainmenu/MainMenuConfig.h
@@ -0,0 +1,19 @@
+#pragma once
+#include <crepe/types.h>
+
+//main menu config
+static constexpr float STARTMAP_OFFSET = 50;
+static constexpr crepe::vec2 MENU_OFFSET = {0, 0};
+static constexpr float MENU_BUTTON_SPACING = 10;
+static constexpr const char * MENU_BUTTON_NAME = "menu_button_background";
+static constexpr crepe::vec2 MENU_OFFSET_BUTTON = {-400, -200};
+static constexpr crepe::vec2 MENU_OFFSET_BUTTON_BACKGROUND = {-400, 0};
+static constexpr const char * MENU_INFO_TAG = "menu_info";
+static constexpr crepe::vec2 MENU_OFFSET_INFO = {350, -365};
+static constexpr crepe::vec2 MENU_OFFSET_INFO_BACKGROUND = {350, -365}; //375
+//Moving to new scene (Start and Preview)
+static constexpr float SLOW_DOWN = 200;
+static constexpr float END = 300;
+static constexpr float VELOCITY_MAX = 200;
+static constexpr float VELOCITY_STEP = 200;
+static constexpr float VELOCITY_INFO_UP = 40;
diff --git a/game/menus/mainmenu/MainMenuScene.cpp b/game/menus/mainmenu/MainMenuScene.cpp
new file mode 100644
index 0000000..c5d2030
--- /dev/null
+++ b/game/menus/mainmenu/MainMenuScene.cpp
@@ -0,0 +1,137 @@
+
+#include "MainMenuScene.h"
+#include "CreditsSubScene.h"
+#include "MainMenuConfig.h"
+#include "QuitScript.h"
+#include "TransitionStartSubScript.h"
+
+#include "../ButtonSubScene.h"
+#include "../MenusConfig.h"
+
+#include "../../Config.h"
+#include "../../background/HallwaySubScene.h"
+#include "../../background/StartSubScene.h"
+
+#include "../endgame/EndGameSubScene.h"
+
+#include <crepe/api/BehaviorScript.h>
+#include <crepe/api/Camera.h>
+#include <crepe/api/GameObject.h>
+#include <crepe/api/Sprite.h>
+#include <crepe/manager/SaveManager.h>
+
+using namespace crepe;
+using namespace std;
+
+void MainMenuScene::load_scene() {
+ ButtonSubScene button;
+
+ GameObject camera_object = this->new_object(CAMERA_NAME);
+ camera_object.add_component<Camera>(
+ ivec2(990, 720), vec2(1100, 800),
+ Camera::Data {
+ .bg_color = Color::BLACK,
+ }
+ );
+ camera_object.add_component<BehaviorScript>().set_script<TransitionStartSubScript>();
+ camera_object.add_component<BehaviorScript>().set_script<QuitScript>();
+
+ //Button menu
+ GameObject menu_button = this->new_object(MENU_BUTTON_NAME, MENU_BUTTON_NAME, MENU_OFFSET);
+ menu_button.add_component<Sprite>(
+ Asset("asset/ui/background.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 0,
+ .size = {300, 860},
+ .position_offset = MENU_OFFSET_BUTTON_BACKGROUND,
+ }
+ );
+
+ vec2 pos_btn = MENU_OFFSET_BUTTON;
+
+ //Preview btn
+ button.create(
+ *this,
+ ButtonSubScene::Data {
+ .text = "PREVIEW",
+ .text_width = 200,
+ .position = pos_btn,
+ .script_type = ButtonSubScene::ScriptSelect::PREVIEW,
+ .btn_side_color = ButtonSubScene::ButtonSideColor::PURPLE,
+ }
+ );
+
+ //Shop btn
+ pos_btn.y += MENU_BUTTON_SPACING + LARGE_OVERLAY_SIZE.y;
+ button.create(
+ *this,
+ ButtonSubScene::Data {
+ .text = "SHOP",
+ .text_offset = {-20, 0},
+ .text_width = 110,
+ .icon_offset = {60, 0},
+ .icon_type = ButtonSubScene::IconSelect::SHOP,
+ .position = pos_btn,
+ .script_type = ButtonSubScene::ScriptSelect::SHOP,
+ .btn_side_color = ButtonSubScene::ButtonSideColor::ORANGE,
+ }
+ );
+
+ //Credits btn
+ pos_btn.y += MENU_BUTTON_SPACING + LARGE_OVERLAY_SIZE.y;
+ button.create(
+ *this,
+ ButtonSubScene::Data {
+ .text = "CREDITS",
+ .text_offset = {0, 0},
+ .text_width = 200,
+ .position = pos_btn,
+ .script_type = ButtonSubScene::ScriptSelect::CREDITS_SHOW,
+ .btn_side_color = ButtonSubScene::ButtonSideColor::BLUE,
+ }
+ );
+
+ //Start of map
+ StartSubScene start;
+ HallwaySubScene hallway;
+ float begin_x = start.create(*this, STARTMAP_OFFSET);
+ begin_x = hallway.create(*this, begin_x, 1, Color::YELLOW);
+
+ //INFO menu
+ GameObject menu_info
+ = this->new_object("MENU_INFO_BACKGROUND", MENU_INFO_TAG, MENU_OFFSET);
+ menu_info.add_component<Sprite>(
+ Asset("asset/ui/itemsButtonBlankDark.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 0,
+ .size = {250, 80},
+ .position_offset = MENU_OFFSET_INFO,
+ .world_space = false,
+ }
+ );
+ SaveManager & savemgr = this->get_save_manager();
+ string number = std::to_string(savemgr.get<int>(TOTAL_COINS_GAME, 0).get());
+ float amount_number = static_cast<float>(number.size());
+ // savemgr.set(COIN_GAME_AMOUNT, amount);
+ button.create(
+ *this,
+ ButtonSubScene::Data {
+ .text = number,
+ .text_offset = {-10 - (amount_number - 1) * 10, 0},
+ .text_width = amount_number * 20,
+ .icon_offset = {60, 0},
+ .icon_type = ButtonSubScene::IconSelect::COINS,
+ .position = MENU_OFFSET_INFO,
+ .script_type = ButtonSubScene::ScriptSelect::SHOP,
+ .scale = 0.6,
+ .worldspace = false,
+ .color_side = false,
+ .tag = MENU_INFO_TAG,
+ }
+ );
+
+ CreditsSubScene creditscene;
+ creditscene.create(*this);
+}
+
+string MainMenuScene::get_name() const { return MAINMENU_SCENE; }
diff --git a/game/menus/mainmenu/MainMenuScene.h b/game/menus/mainmenu/MainMenuScene.h
new file mode 100644
index 0000000..1eea90e
--- /dev/null
+++ b/game/menus/mainmenu/MainMenuScene.h
@@ -0,0 +1,10 @@
+#pragma once
+
+#include <crepe/api/Scene.h>
+#include <string>
+
+class MainMenuScene : public crepe::Scene {
+public:
+ void load_scene();
+ std::string get_name() const;
+};
diff --git a/game/menus/mainmenu/TransitionStartSubScript.cpp b/game/menus/mainmenu/TransitionStartSubScript.cpp
new file mode 100644
index 0000000..f737f7f
--- /dev/null
+++ b/game/menus/mainmenu/TransitionStartSubScript.cpp
@@ -0,0 +1,16 @@
+#include "TransitionStartSubScript.h"
+
+#include "../MenusConfig.h"
+
+using namespace crepe;
+using namespace std;
+
+void TransitionStartSubScript::fixed_update(crepe::duration_t dt) {
+ if (this->get_key_state(Keycode::SPACE) && this->transition == false)
+ this->transition = true;
+}
+
+const char * TransitionStartSubScript::get_scene_name() const {
+ // Provide the next scene defined in MainMenuConfig
+ return START_SCENE;
+}
diff --git a/game/menus/mainmenu/TransitionStartSubScript.h b/game/menus/mainmenu/TransitionStartSubScript.h
new file mode 100644
index 0000000..f9862ea
--- /dev/null
+++ b/game/menus/mainmenu/TransitionStartSubScript.h
@@ -0,0 +1,9 @@
+#pragma once
+
+#include "ITransitionScript.h"
+
+class TransitionStartSubScript : public ITransitionScript {
+public:
+ void fixed_update(crepe::duration_t dt) override;
+ const char * get_scene_name() const override;
+};
diff --git a/game/menus/shop/ButtonBuySelectBubbleScript.cpp b/game/menus/shop/ButtonBuySelectBubbleScript.cpp
new file mode 100644
index 0000000..741afde
--- /dev/null
+++ b/game/menus/shop/ButtonBuySelectBubbleScript.cpp
@@ -0,0 +1,34 @@
+#include "ButtonBuySelectBubbleScript.h"
+#include "../MenusConfig.h"
+#include "Config.h"
+#include "ValueBroker.h"
+#include "manager/SaveManager.h"
+#include "menus/shop/Shopconfig.h"
+
+using namespace crepe;
+using namespace std;
+
+void ButtonBuySelectBubbleScript::init() {
+ IButtonScript::init();
+ this->subscribe<ButtonPressEvent>([this](const ButtonPressEvent & e) {
+ return this->on_button_press(e);
+ });
+}
+
+bool ButtonBuySelectBubbleScript::on_button_press(const ButtonPressEvent & e) {
+ SaveManager & save = this->get_save_manager();
+ ValueBroker<int> buy_bullet = save.get<int>(BUY_BUBBLE_SAVE, 0);
+ if (!buy_bullet.get()) {
+ ValueBroker<int> coins = save.get<int>(TOTAL_COINS_GAME, 0);
+ if (coins.get() >= 1000) {
+ int coin = coins.get();
+ coin -= 1000;
+ save.set(TOTAL_COINS_GAME, coin);
+ save.set(BUY_BUBBLE_SAVE, 1);
+ }
+ } else {
+ save.set(JETPACK_PARTICLES, 1);
+ }
+ this->trigger_event<ShopUpdate>();
+ return false;
+}
diff --git a/game/menus/shop/ButtonBuySelectBubbleScript.h b/game/menus/shop/ButtonBuySelectBubbleScript.h
new file mode 100644
index 0000000..ce276ef
--- /dev/null
+++ b/game/menus/shop/ButtonBuySelectBubbleScript.h
@@ -0,0 +1,14 @@
+#pragma once
+
+#include "../IButtonScript.h"
+
+#include <crepe/api/Script.h>
+
+class ButtonBuySelectBubbleScript : public IButtonScript {
+public:
+ void init() override;
+ bool on_button_press(const crepe::ButtonPressEvent & e);
+
+protected:
+ bool transition = false;
+};
diff --git a/game/menus/shop/ButtonBuySelectBulletScript.cpp b/game/menus/shop/ButtonBuySelectBulletScript.cpp
new file mode 100644
index 0000000..d30849c
--- /dev/null
+++ b/game/menus/shop/ButtonBuySelectBulletScript.cpp
@@ -0,0 +1,34 @@
+#include "ButtonBuySelectBulletScript.h"
+#include "../MenusConfig.h"
+#include "Config.h"
+#include "ValueBroker.h"
+#include "manager/SaveManager.h"
+#include "menus/shop/Shopconfig.h"
+
+using namespace crepe;
+using namespace std;
+
+void ButtonBuySelectBulletScript::init() {
+ IButtonScript::init();
+ this->subscribe<ButtonPressEvent>([this](const ButtonPressEvent & e) {
+ return this->on_button_press(e);
+ });
+}
+
+bool ButtonBuySelectBulletScript::on_button_press(const ButtonPressEvent & e) {
+ SaveManager & save = this->get_save_manager();
+ ValueBroker<int> buy_bullet = save.get<int>(BUY_BULLET_SAVE, 0);
+ if (!buy_bullet.get()) {
+ ValueBroker<int> coins = save.get<int>(TOTAL_COINS_GAME, 0);
+ if (coins.get() >= 0) {
+ int coin = coins.get();
+ coin -= 0;
+ save.set(TOTAL_COINS_GAME, coin);
+ save.set(BUY_BULLET_SAVE, 1);
+ }
+ } else {
+ save.set(JETPACK_PARTICLES, 0);
+ }
+ this->trigger_event<ShopUpdate>();
+ return false;
+}
diff --git a/game/menus/shop/ButtonBuySelectBulletScript.h b/game/menus/shop/ButtonBuySelectBulletScript.h
new file mode 100644
index 0000000..86de0ac
--- /dev/null
+++ b/game/menus/shop/ButtonBuySelectBulletScript.h
@@ -0,0 +1,14 @@
+#pragma once
+
+#include "../IButtonScript.h"
+
+#include <crepe/api/Script.h>
+
+class ButtonBuySelectBulletScript : public IButtonScript {
+public:
+ void init() override;
+ bool on_button_press(const crepe::ButtonPressEvent & e);
+
+protected:
+ bool transition = false;
+};
diff --git a/game/menus/shop/ShopLoadScript.cpp b/game/menus/shop/ShopLoadScript.cpp
new file mode 100644
index 0000000..a545fe2
--- /dev/null
+++ b/game/menus/shop/ShopLoadScript.cpp
@@ -0,0 +1,126 @@
+#include "ShopLoadScript.h"
+#include "Shopconfig.h"
+#include "api/Button.h"
+#include "api/Sprite.h"
+#include "api/Text.h"
+#include "manager/SaveManager.h"
+#include <crepe/ValueBroker.h>
+
+using namespace crepe;
+using namespace std;
+
+void ShopLoadScript::init() {
+ this->update();
+ this->subscribe<ShopUpdate>([this](const ShopUpdate e) { return this->update(); });
+}
+
+bool ShopLoadScript::update() {
+ SaveManager & save = this->get_save_manager();
+ ValueBroker<int> buy_bullet = save.get<int>(BUY_BULLET_SAVE, 0);
+ ValueBroker<int> buy_bubble = save.get<int>(BUY_BUBBLE_SAVE, 0);
+
+ if (buy_bullet.get()) {
+ auto sprites = this->get_components_by_tag<Sprite>(BUY_BULLET);
+ for (auto sprite : sprites) {
+ sprite.get().active = false;
+ }
+ auto buttons = this->get_components_by_tag<Button>(BUY_BULLET);
+ for (auto btn : buttons) {
+ btn.get().active = false;
+ }
+ auto texts = this->get_components_by_tag<Text>(BUY_BULLET);
+ for (auto txt : texts) {
+ txt.get().active = false;
+ }
+ auto sprites1 = this->get_components_by_tag<Sprite>(SELECT_BULLET);
+ for (auto sprite : sprites1) {
+ sprite.get().active = true;
+ }
+ auto buttons1 = this->get_components_by_tag<Button>(SELECT_BULLET);
+ for (auto btn : buttons1) {
+ btn.get().active = true;
+ }
+ auto texts1 = this->get_components_by_tag<Text>(SELECT_BULLET);
+ for (auto txt : texts1) {
+ txt.get().active = true;
+ }
+ } else {
+ auto sprites = this->get_components_by_tag<Sprite>(SELECT_BULLET);
+ for (auto sprite : sprites) {
+ sprite.get().active = false;
+ }
+ auto buttons = this->get_components_by_tag<Button>(SELECT_BULLET);
+ for (auto btn : buttons) {
+ btn.get().active = false;
+ }
+ auto texts = this->get_components_by_tag<Text>(SELECT_BULLET);
+ for (auto txt : texts) {
+ txt.get().active = false;
+ }
+ auto sprites1 = this->get_components_by_tag<Sprite>(BUY_BULLET);
+ for (auto sprite : sprites1) {
+ sprite.get().active = true;
+ }
+ auto buttons1 = this->get_components_by_tag<Button>(BUY_BULLET);
+ for (auto btn : buttons1) {
+ btn.get().active = true;
+ }
+ auto texts1 = this->get_components_by_tag<Text>(BUY_BULLET);
+ for (auto txt : texts1) {
+ txt.get().active = true;
+ }
+ }
+
+ if (buy_bubble.get()) {
+ auto sprites = this->get_components_by_tag<Sprite>(BUY_BUBBLE);
+ for (auto sprite : sprites) {
+ sprite.get().active = false;
+ }
+ auto buttons = this->get_components_by_tag<Button>(BUY_BUBBLE);
+ for (auto btn : buttons) {
+ btn.get().active = false;
+ }
+ auto texts = this->get_components_by_tag<Text>(BUY_BUBBLE);
+ for (auto txt : texts) {
+ txt.get().active = false;
+ }
+ auto sprites1 = this->get_components_by_tag<Sprite>(SELECT_BUBBLE);
+ for (auto sprite : sprites1) {
+ sprite.get().active = true;
+ }
+ auto buttons1 = this->get_components_by_tag<Button>(SELECT_BUBBLE);
+ for (auto btn : buttons1) {
+ btn.get().active = true;
+ }
+ auto texts1 = this->get_components_by_tag<Text>(SELECT_BUBBLE);
+ for (auto txt : texts1) {
+ txt.get().active = true;
+ }
+ } else {
+ auto sprites = this->get_components_by_tag<Sprite>(SELECT_BUBBLE);
+ for (auto sprite : sprites) {
+ sprite.get().active = false;
+ }
+ auto buttons = this->get_components_by_tag<Button>(SELECT_BUBBLE);
+ for (auto btn : buttons) {
+ btn.get().active = false;
+ }
+ auto texts = this->get_components_by_tag<Text>(SELECT_BUBBLE);
+ for (auto txt : texts) {
+ txt.get().active = false;
+ }
+ auto sprites1 = this->get_components_by_tag<Sprite>(BUY_BUBBLE);
+ for (auto sprite : sprites1) {
+ sprite.get().active = true;
+ }
+ auto buttons1 = this->get_components_by_tag<Button>(BUY_BUBBLE);
+ for (auto btn : buttons1) {
+ btn.get().active = true;
+ }
+ auto texts1 = this->get_components_by_tag<Text>(BUY_BUBBLE);
+ for (auto txt : texts1) {
+ txt.get().active = true;
+ }
+ }
+ return false;
+}
diff --git a/game/menus/shop/ShopLoadScript.h b/game/menus/shop/ShopLoadScript.h
new file mode 100644
index 0000000..b17bf1c
--- /dev/null
+++ b/game/menus/shop/ShopLoadScript.h
@@ -0,0 +1,10 @@
+#pragma once
+
+#include <crepe/api/Script.h>
+#include <crepe/manager/SaveManager.h>
+
+class ShopLoadScript : public crepe::Script {
+public:
+ void init() override;
+ bool update();
+};
diff --git a/game/menus/shop/ShopMenuScene.cpp b/game/menus/shop/ShopMenuScene.cpp
new file mode 100644
index 0000000..4975a95
--- /dev/null
+++ b/game/menus/shop/ShopMenuScene.cpp
@@ -0,0 +1,326 @@
+
+#include "ShopMenuScene.h"
+
+#include "../../Config.h"
+#include "../ButtonSubScene.h"
+#include "../MenusConfig.h"
+#include "api/BehaviorScript.h"
+#include "menus/BannerSubScene.h"
+#include "menus/shop/ShopLoadScript.h"
+#include "types.h"
+
+#include "Shopconfig.h"
+#include <crepe/api/Camera.h>
+#include <crepe/api/Sprite.h>
+#include <crepe/api/Text.h>
+
+using namespace crepe;
+using namespace std;
+
+void ShopMenuScene::load_scene() {
+ GameObject camera_object = this->new_object(CAMERA_NAME);
+ camera_object.add_component<Camera>(
+ ivec2(990, 720), vec2(1100, 800),
+ Camera::Data {
+ .bg_color = Color::RED,
+ }
+ );
+ BannerSubScene banner;
+ banner.create(
+ *this,
+ {
+ .banner_title = "SHOP",
+ .banner_title_width = 200,
+ .banner_title_offset = {0, 65},
+ }
+ );
+ GameObject menu_background = this->new_object("menu_background");
+ menu_background.add_component<Sprite>(
+ Asset("asset/ui/background.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 0,
+ .size = {1100, 860},
+ .position_offset {0},
+ }
+ );
+ menu_background.add_component<BehaviorScript>().set_script<ShopLoadScript>();
+
+ ButtonSubScene button;
+ button.create(
+ *this,
+ ButtonSubScene::Data {
+ .text = "BACK",
+ .text_width = 115,
+ .position = {-400, -350},
+ .script_type = ButtonSubScene::ScriptSelect::MAINMENU,
+ .button_type = ButtonSubScene::ButtonSelect::BACK,
+ .scale = 0.8,
+ .sorting_layer_offset = 1,
+ }
+ );
+
+ const float CHAR_SIZE = 16;
+ const float CHAR_SIZE_COIN = 16;
+ crepe::vec2 size;
+
+ GameObject shop_item_bullet = this->new_object("bullet", "shop_item", vec2(-100, 0));
+ shop_item_bullet.add_component<Sprite>(
+ Asset("asset/other_effects/effect_rocketmgshell_TVOS.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 1,
+ .size = {0, 20},
+ .angle_offset = 30,
+ .position_offset = {0, 0},
+ }
+ );
+ shop_item_bullet.add_component<Sprite>(
+ Asset("asset/other_effects/effect_rocketmgshell_TVOS.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 1,
+ .size = {0, 20},
+ .angle_offset = 10,
+ .position_offset = {-10, -30},
+ }
+ );
+ shop_item_bullet.add_component<Sprite>(
+ Asset("asset/other_effects/effect_rocketmgshell_TVOS.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 1,
+ .size = {0, 20},
+ .angle_offset = -10,
+ .position_offset = {-40, 30},
+ }
+ );
+ shop_item_bullet.add_component<Sprite>(
+ Asset("asset/other_effects/effect_rocketmgshell_TVOS.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 1,
+ .size = {0, 20},
+ .angle_offset = 0,
+ .position_offset = {10, 15},
+ }
+ );
+ shop_item_bullet.add_component<Sprite>(
+ Asset("asset/other_effects/effect_rocketmgshell_TVOS.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 1,
+ .size = {0, 20},
+ .angle_offset = -5,
+ .position_offset = {45, -5},
+ }
+ );
+
+ const string BULLETS_STRING = "BULLETS";
+ size
+ = {CHAR_SIZE * BULLETS_STRING.size(),
+ (CHAR_SIZE * BULLETS_STRING.size() / BULLETS_STRING.size()) * 2};
+
+ shop_item_bullet.add_component<Text>(
+ size, FONT,
+ Text::Data {
+ .world_space = true,
+ .text_color = Color::WHITE,
+ },
+ vec2 {0, -75}, BULLETS_STRING
+ );
+ shop_item_bullet.add_component<Sprite>(
+ Asset("asset/ui/buttonCoinsSmall.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 1,
+ .size = {0, 45},
+ .position_offset = {25, 75},
+ }
+ );
+
+ const string BULLETS_GOLD_STRING = "0";
+ size
+ = {CHAR_SIZE_COIN * BULLETS_GOLD_STRING.size(),
+ (CHAR_SIZE_COIN * BULLETS_GOLD_STRING.size() / BULLETS_GOLD_STRING.size()) * 2};
+ shop_item_bullet.add_component<Text>(
+ size, FONT,
+ Text::Data {
+ .world_space = true,
+ .text_color = Color::GOLD,
+ },
+ vec2 {-5, 75}, BULLETS_GOLD_STRING
+ );
+
+ GameObject shop_item_bubble = this->new_object("bubble", "shop_item", vec2(100, 0));
+ shop_item_bubble.add_component<Sprite>(
+ Asset("asset/background/aquarium/bubble.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 1,
+ .size = {0, 10},
+ .position_offset = {0, 0},
+ }
+ );
+ shop_item_bubble.add_component<Sprite>(
+ Asset("asset/background/aquarium/bubble.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 1,
+ .size = {0, 10},
+ .position_offset = {-50, -20},
+ }
+ );
+ shop_item_bubble.add_component<Sprite>(
+ Asset("asset/background/aquarium/bubble.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 1,
+ .size = {0, 20},
+ .position_offset = {45, -40},
+ }
+ );
+ shop_item_bubble.add_component<Sprite>(
+ Asset("asset/background/aquarium/bubble.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 1,
+ .size = {0, 20},
+ .position_offset = {-20, 40},
+ }
+ );
+ shop_item_bubble.add_component<Sprite>(
+ Asset("asset/background/aquarium/bubble.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 1,
+ .size = {0, 15},
+ .position_offset = {15, -25},
+ }
+ );
+ shop_item_bubble.add_component<Sprite>(
+ Asset("asset/background/aquarium/bubble.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 1,
+ .size = {0, 10},
+ .position_offset = {10, 5},
+ }
+ );
+ shop_item_bubble.add_component<Sprite>(
+ Asset("asset/background/aquarium/bubble.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 1,
+ .size = {0, 10},
+ .position_offset = {-5, -20},
+ }
+ );
+ shop_item_bubble.add_component<Sprite>(
+ Asset("asset/background/aquarium/bubble.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 1,
+ .size = {0, 20},
+ .position_offset = {15, -40},
+ }
+ );
+ shop_item_bubble.add_component<Sprite>(
+ Asset("asset/background/aquarium/bubble.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 1,
+ .size = {0, 20},
+ .position_offset = {-20, 10},
+ }
+ );
+ shop_item_bubble.add_component<Sprite>(
+ Asset("asset/background/aquarium/bubble.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 1,
+ .size = {0, 15},
+ .position_offset = {30, -25},
+ }
+ );
+
+ const string BUBBLE_STRING = "BUBBLE";
+ size
+ = {CHAR_SIZE * BUBBLE_STRING.size(),
+ (CHAR_SIZE * BUBBLE_STRING.size() / BUBBLE_STRING.size()) * 2};
+ shop_item_bubble.add_component<Text>(
+ size, FONT,
+ Text::Data {
+ .world_space = true,
+ .text_color = Color::WHITE,
+ },
+ vec2 {0, -75}, BUBBLE_STRING
+ );
+ shop_item_bubble.add_component<Sprite>(
+ Asset("asset/ui/buttonCoinsSmall.png"),
+ Sprite::Data {
+ .sorting_in_layer = STARTING_SORTING_IN_LAYER + 1,
+ .size = {0, 45},
+ .position_offset = {45, 75},
+ }
+ );
+
+ const string BUBBLE_GOLD_STRING = "1000";
+ size
+ = {CHAR_SIZE_COIN * BUBBLE_GOLD_STRING.size(),
+ (CHAR_SIZE_COIN * BUBBLE_GOLD_STRING.size() / BUBBLE_GOLD_STRING.size()) * 2};
+ shop_item_bubble.add_component<Text>(
+ size, FONT,
+ Text::Data {
+ .world_space = true,
+ .text_color = Color::GOLD,
+ },
+ vec2 {-10, 75}, BUBBLE_GOLD_STRING
+ );
+
+ button.create(
+ *this,
+ ButtonSubScene::Data {
+ .text = "BUY",
+ .text_width = 100,
+ .position = vec2(-100, 120),
+ .script_type = ButtonSubScene::ScriptSelect::SHOP_BULLET,
+ .button_type = ButtonSubScene::ButtonSelect::LARGE,
+ .scale = 0.4,
+ .tag = BUY_BULLET,
+ .sorting_layer_offset = 20,
+ .btn_side_color = ButtonSubScene::ButtonSideColor::PURPLE
+
+ }
+ );
+
+ button.create(
+ *this,
+ ButtonSubScene::Data {
+ .text = "BUY",
+ .text_width = 100,
+ .position = vec2(100, 120),
+ .script_type = ButtonSubScene::ScriptSelect::SHOP_BUBBLE,
+ .button_type = ButtonSubScene::ButtonSelect::LARGE,
+ .scale = 0.4,
+ .tag = BUY_BUBBLE,
+ .sorting_layer_offset = 20,
+ .btn_side_color = ButtonSubScene::ButtonSideColor::PURPLE
+ }
+ );
+
+ button.create(
+ *this,
+ ButtonSubScene::Data {
+ .text = "SELECT",
+ .text_width = 100,
+ .position = vec2(-100, 120),
+ .script_type = ButtonSubScene::ScriptSelect::SHOP_BULLET,
+ .button_type = ButtonSubScene::ButtonSelect::LARGE,
+ .scale = 0.4,
+ .tag = SELECT_BULLET,
+ .sorting_layer_offset = 20,
+ .btn_side_color = ButtonSubScene::ButtonSideColor::PURPLE
+ }
+ );
+
+ button.create(
+ *this,
+ ButtonSubScene::Data {
+ .text = "SELECT",
+ .text_width = 100,
+ .position = vec2(100, 120),
+ .script_type = ButtonSubScene::ScriptSelect::SHOP_BUBBLE,
+ .button_type = ButtonSubScene::ButtonSelect::LARGE,
+ .scale = 0.4,
+ .tag = SELECT_BUBBLE,
+ .sorting_layer_offset = 20,
+ .btn_side_color = ButtonSubScene::ButtonSideColor::PURPLE
+ }
+ );
+}
+
+string ShopMenuScene::get_name() const { return SHOP_SCENE; }
diff --git a/game/menus/shop/ShopMenuScene.h b/game/menus/shop/ShopMenuScene.h
new file mode 100644
index 0000000..34c05ff
--- /dev/null
+++ b/game/menus/shop/ShopMenuScene.h
@@ -0,0 +1,12 @@
+#pragma once
+
+#include <string>
+
+#include <crepe/api/Scene.h>
+
+class ShopMenuScene : public crepe::Scene {
+public:
+ void load_scene();
+
+ std::string get_name() const;
+};
diff --git a/game/menus/shop/Shopconfig.h b/game/menus/shop/Shopconfig.h
new file mode 100644
index 0000000..a686242
--- /dev/null
+++ b/game/menus/shop/Shopconfig.h
@@ -0,0 +1,14 @@
+#pragma once
+#include "api/Event.h"
+
+//tags
+static constexpr const char * BUY_BULLET = "BUY_BULLET";
+static constexpr const char * SELECT_BULLET = "SELECT_BULLET";
+static constexpr const char * BUY_BUBBLE = "BUY_BUBBLE";
+static constexpr const char * SELECT_BUBBLE = "SELECT_BUBBLE";
+
+//save_data
+static constexpr const char * BUY_BULLET_SAVE = "BUY_BULLET_SAVE";
+static constexpr const char * BUY_BUBBLE_SAVE = "BUY_BUBBLE_SAVE";
+
+struct ShopUpdate : public crepe::Event {};
diff --git a/game/missile/AlertScript.cpp b/game/missile/AlertScript.cpp
new file mode 100644
index 0000000..0e6f5c5
--- /dev/null
+++ b/game/missile/AlertScript.cpp
@@ -0,0 +1,38 @@
+#include "AlertScript.h"
+#include "../Config.h"
+
+#include <crepe/api/CircleCollider.h>
+#include <crepe/api/Sprite.h>
+#include <crepe/api/Transform.h>
+
+using namespace crepe;
+
+void AlertScript::fixed_update(crepe::duration_t dt) {
+ const auto & cam = this->get_components_by_name<Transform>("camera").front().get();
+ //missile transform
+ const auto & this_transform = this->get_component<Transform>();
+ auto missile_transforms = this->get_components_by_name<Transform>("missile");
+ const auto & this_collider = this->get_component<CircleCollider>();
+
+ auto alert_sprites = this->get_components_by_name<Sprite>("missile_alert");
+ auto alert_transforms = this->get_components_by_name<Transform>("missile_alert");
+
+ int idx = 0;
+ for (int i = 0; i < missile_transforms.size(); ++i) {
+ const auto & missile_transform = missile_transforms[i].get();
+ if (this_transform.game_object_id == missile_transform.game_object_id) {
+ idx = i;
+ break;
+ }
+ }
+
+ auto & alert_transform = alert_transforms[idx].get();
+ alert_transform.position.x = cam.position.x + (VIEWPORT_X / 2 - 100);
+ alert_transform.position.y = this_transform.position.y;
+
+ // check if transform is in camera view
+ if (this_transform.position.x > cam.position.x - (VIEWPORT_X / 2)
+ && this_transform.position.x < cam.position.x + (VIEWPORT_X / 2)) {
+ alert_sprites[idx].get().active = false;
+ }
+}
diff --git a/game/missile/AlertScript.h b/game/missile/AlertScript.h
new file mode 100644
index 0000000..5b0b3a1
--- /dev/null
+++ b/game/missile/AlertScript.h
@@ -0,0 +1,11 @@
+#pragma once
+
+#include <crepe/api/Script.h>
+
+class AlertScript : public crepe::Script {
+private:
+ bool has_alert = false;
+
+public:
+ void fixed_update(crepe::duration_t dt);
+};
diff --git a/game/missile/AlertSubScene.cpp b/game/missile/AlertSubScene.cpp
new file mode 100644
index 0000000..5bce5ac
--- /dev/null
+++ b/game/missile/AlertSubScene.cpp
@@ -0,0 +1,33 @@
+#include "AlertSubScene.h"
+#include "../Config.h"
+
+#include <crepe/api/Animator.h>
+#include <crepe/api/Scene.h>
+#include <crepe/api/Sprite.h>
+
+using namespace crepe;
+
+MissileAlert::MissileAlert(Scene & scn) {
+ GameObject alert = scn.new_object("missile_alert", "missile_alert", {0, 0}, 0, 1);
+
+ Asset missile_alert_ss {"asset/obstacles/missile/missileAlert.png"};
+
+ auto & missile_alert_sprite = alert.add_component<Sprite>(
+ missile_alert_ss,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_OBSTACLES,
+ .size = {0, 100},
+ }
+ );
+
+ auto & missile_alert_anim = alert.add_component<Animator>(
+ missile_alert_sprite, ivec2 {64, 64}, uvec2 {4, 2},
+ Animator::Data {
+ .fps = 15,
+ .looping = true,
+ }
+ );
+
+ missile_alert_anim.set_anim(1);
+ missile_alert_sprite.active = false;
+}
diff --git a/game/missile/AlertSubScene.h b/game/missile/AlertSubScene.h
new file mode 100644
index 0000000..8ea288f
--- /dev/null
+++ b/game/missile/AlertSubScene.h
@@ -0,0 +1,8 @@
+#pragma once
+
+#include <crepe/api/Scene.h>
+
+class MissileAlert {
+public:
+ MissileAlert(crepe::Scene & scn);
+};
diff --git a/game/missile/MissilePool.cpp b/game/missile/MissilePool.cpp
new file mode 100644
index 0000000..23f03c9
--- /dev/null
+++ b/game/missile/MissilePool.cpp
@@ -0,0 +1,18 @@
+#include "MissilePool.h"
+#include "MissileSubScene.h"
+#include "missile/AlertSubScene.h"
+
+#include <crepe/api/Scene.h>
+
+using namespace std;
+using namespace crepe;
+
+MissilePool::MissilePool(Scene & scn) {
+ int amount = 0;
+ MissileSubScene missile;
+ while (amount < this->MAX_MISSILE_COUNT) {
+ MissileAlert alert(scn);
+ missile.create(scn);
+ amount++;
+ }
+}
diff --git a/game/missile/MissilePool.h b/game/missile/MissilePool.h
new file mode 100644
index 0000000..296701e
--- /dev/null
+++ b/game/missile/MissilePool.h
@@ -0,0 +1,11 @@
+#pragma once
+
+#include <crepe/api/Scene.h>
+
+class MissilePool {
+public:
+ MissilePool(crepe::Scene & scn);
+
+private:
+ static constexpr unsigned int MAX_MISSILE_COUNT = 5;
+};
diff --git a/game/missile/MissileScript.cpp b/game/missile/MissileScript.cpp
new file mode 100644
index 0000000..3aa4eb6
--- /dev/null
+++ b/game/missile/MissileScript.cpp
@@ -0,0 +1,106 @@
+#include "MissileScript.h"
+#include "../Config.h"
+#include <cmath>
+
+#include <crepe/api/AI.h>
+#include <crepe/api/Animator.h>
+#include <crepe/api/AudioSource.h>
+#include <crepe/api/BehaviorScript.h>
+#include <crepe/api/CircleCollider.h>
+#include <crepe/api/KeyCodes.h>
+#include <crepe/api/Rigidbody.h>
+#include <crepe/api/Sprite.h>
+#include <crepe/api/Transform.h>
+#include <crepe/system/CollisionSystem.h>
+#include <crepe/types.h>
+
+using namespace std;
+using namespace crepe;
+
+void MissileScript::init() {
+ subscribe<CollisionEvent>([this](const CollisionEvent & ev) -> bool {
+ return this->on_collision(ev);
+ });
+ this->seeking_disabled = false;
+}
+void MissileScript::kill_missile() {
+ auto animations = this->get_components<Animator>();
+ auto sprites = this->get_components<Sprite>();
+ auto collider = this->get_component<CircleCollider>();
+ auto & this_script = this->get_components<BehaviorScript>().front().get();
+
+ animations[0].get().active = false;
+ animations[1].get().active = false;
+ animations[2].get().active = true;
+ sprites[0].get().active = false;
+ sprites[1].get().active = false;
+ sprites[2].get().active = true;
+
+ collider.active = false;
+ this_script.active = false;
+ this->seeking_disabled = false;
+}
+void MissileScript::activate() {
+ auto anim = this->get_components<Animator>();
+ auto sprites = this->get_components<Sprite>();
+
+ sprites[0].get().active = true;
+ sprites[1].get().active = true;
+ sprites[2].get().active = false;
+
+ anim[0].get().active = true;
+ anim[1].get().active = true;
+ anim[2].get().stop();
+}
+bool MissileScript::on_collision(const CollisionEvent & ev) {
+ auto & explosion_sound = this->get_components<AudioSource>().back().get();
+
+ this->kill_missile();
+ explosion_sound.play();
+
+ return false;
+}
+
+bool MissileScript::is_in_x_range(const Transform & missile, const Transform & player) {
+ return fabs(missile.position.x - player.position.x) <= this->X_RANGE;
+}
+
+void MissileScript::fixed_update(crepe::duration_t dt) {
+ auto & explosion_anim = this->get_components<Animator>().back().get();
+ auto & missile = this->get_component<Transform>();
+ auto & m_ai = this->get_component<AI>();
+
+ const auto & player = this->get_components_by_name<Transform>("player").front().get();
+ const auto & cam = this->get_components_by_name<Transform>("camera").front().get();
+ const auto & velocity = this->get_component<Rigidbody>().data.linear_velocity;
+
+ // check if animation is at the end
+ if (explosion_anim.data.row == 7) {
+ this->activate();
+ this->seeking_disabled = false;
+ return;
+ }
+
+ if (missile.position.x < (cam.position.x - VIEWPORT_X / 1.8)) {
+ this->kill_missile();
+ return;
+ }
+
+ if (this->seeking_disabled) {
+ m_ai.seek_off();
+ } else {
+ m_ai.seek_target = player.position;
+ m_ai.seek_on();
+
+ if (is_in_x_range(missile, player)) {
+ this->seeking_disabled = true;
+ m_ai.seek_off();
+ }
+ }
+
+ vec2 angle_pos = velocity;
+ float angle = atan2(angle_pos.y, angle_pos.x) * (180 / M_PI);
+
+ missile.rotation = angle;
+ missile.position += velocity * dt.count();
+}
diff --git a/game/missile/MissileScript.h b/game/missile/MissileScript.h
new file mode 100644
index 0000000..a492e18
--- /dev/null
+++ b/game/missile/MissileScript.h
@@ -0,0 +1,20 @@
+#pragma once
+
+#include <crepe/api/Script.h>
+
+class MissileScript : public crepe::Script {
+private:
+ bool on_collision(const crepe::CollisionEvent & ev);
+
+ bool seeking_disabled;
+
+ // will be used to calculate when ai will be stopped
+ static constexpr int X_RANGE = 90;
+ bool is_in_x_range(const crepe::Transform & missile, const crepe::Transform & player);
+ void kill_missile();
+ void activate();
+
+public:
+ void init();
+ void fixed_update(crepe::duration_t dt);
+};
diff --git a/game/missile/MissileSubScene.cpp b/game/missile/MissileSubScene.cpp
new file mode 100644
index 0000000..d325050
--- /dev/null
+++ b/game/missile/MissileSubScene.cpp
@@ -0,0 +1,101 @@
+#include "MissileSubScene.h"
+#include "../Config.h"
+#include "../missile/MissileScript.h"
+#include "Random.h"
+#include "missile/AlertScript.h"
+
+#include <crepe/api/AI.h>
+#include <crepe/api/Animator.h>
+#include <crepe/api/AudioSource.h>
+#include <crepe/api/BehaviorScript.h>
+#include <crepe/api/CircleCollider.h>
+#include <crepe/api/Scene.h>
+#include <crepe/api/Sprite.h>
+#include <crepe/types.h>
+
+using namespace crepe;
+
+void MissileSubScene::create(crepe::Scene & scn) {
+ GameObject missle = scn.new_object("missile", "missile", {0, 0}, 0, 1);
+
+ Asset missle_ss {"asset/obstacles/missile/missile.png"};
+ Asset missle_thruster_ss {"asset/obstacles/missile/missileEffects.png"};
+ Asset missile_explosion_ss {"asset/obstacles/missile/missileExplosion.png"};
+ Asset explosion_sound {"asset/sfx/rocket_explode_1.ogg"};
+ Asset missile_fire {"asset/sfx/missile_launch.ogg"};
+
+ missle.add_component<BehaviorScript>().set_script<MissileScript>().active = false;
+ missle.add_component<BehaviorScript>().set_script<AlertScript>();
+
+ auto & sound = missle.add_component<AudioSource>(missile_fire);
+ sound.volume = 0.5;
+ auto & sound2 = missle.add_component<AudioSource>(explosion_sound);
+ sound2.volume = 3;
+
+ // sprites
+ auto & missle_sprite = missle.add_component<Sprite>(
+ missle_ss,
+ Sprite::Data {
+ .flip = {true, false},
+ .sorting_in_layer = SORT_IN_LAY_OBSTACLES,
+ .size = {0, 35},
+ }
+ );
+
+ auto & missle_thruster_sprite = missle.add_component<Sprite>(
+ missle_thruster_ss,
+ Sprite::Data {
+ .flip = {true, false},
+ .sorting_in_layer = SORT_IN_LAY_OBSTACLES,
+ .size = {0, 35},
+ .position_offset = {-20, 0},
+ }
+ );
+
+ auto & missile_explosion_sprite = missle.add_component<Sprite>(
+ missile_explosion_ss,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_OBSTACLES,
+ .size = {0, 50},
+ }
+ );
+
+ // Animations
+ missle.add_component<Animator>(
+ missle_sprite, ivec2 {32, 32}, uvec2 {4, 1},
+ Animator::Data {
+ .fps = 15,
+ .looping = true,
+ }
+ );
+
+ missle.add_component<Animator>(
+ missle_thruster_sprite, ivec2 {64, 64}, uvec2 {4, 2},
+ Animator::Data {
+ .fps = 15,
+ .looping = true,
+ }
+ );
+
+ auto & explosion_anim = missle.add_component<Animator>(
+ missile_explosion_sprite, ivec2 {64, 64}, uvec2 {8, 1},
+ Animator::Data {
+ .fps = 10,
+ }
+ );
+
+ missile_explosion_sprite.active = false;
+ explosion_anim.active = false;
+
+ missle.add_component<Rigidbody>(Rigidbody::Data {
+ .body_type = Rigidbody::BodyType::KINEMATIC,
+ .max_linear_velocity = Random::f(250, 200),
+ .kinematic_collision = false,
+ .collision_layers = {COLL_LAY_PLAYER, COLL_LAY_BOT_TOP},
+ .collision_layer = COLL_LAY_MISSILE,
+ });
+
+ missle.add_component<CircleCollider>(3).active = false;
+
+ auto & missle_ai = missle.add_component<AI>(1000);
+}
diff --git a/game/missile/MissileSubScene.h b/game/missile/MissileSubScene.h
new file mode 100644
index 0000000..9ea422a
--- /dev/null
+++ b/game/missile/MissileSubScene.h
@@ -0,0 +1,12 @@
+#pragma once
+
+namespace crepe {
+class Scene;
+}
+
+class MissileSubScene {
+public:
+ MissileSubScene() = default;
+
+ void create(crepe::Scene & scn);
+};
diff --git a/game/missile/SpawnEvent.cpp b/game/missile/SpawnEvent.cpp
new file mode 100644
index 0000000..6109686
--- /dev/null
+++ b/game/missile/SpawnEvent.cpp
@@ -0,0 +1,49 @@
+#include "SpawnEvent.h"
+#include "Random.h"
+
+#include <crepe/api/Animator.h>
+#include <crepe/api/AudioSource.h>
+#include <crepe/api/BehaviorScript.h>
+#include <crepe/api/Camera.h>
+#include <crepe/api/CircleCollider.h>
+#include <crepe/api/Sprite.h>
+#include <crepe/api/Transform.h>
+
+#include <cstdlib>
+
+using namespace crepe;
+
+void MissileSpawnEventHandler::init() {
+ subscribe<MissileSpawnEvent>([this](const MissileSpawnEvent & ev) -> bool {
+ return this->on_event(ev);
+ });
+}
+
+bool MissileSpawnEventHandler::on_event(const MissileSpawnEvent & event) {
+ auto alert_sprites = this->get_components_by_name<Sprite>("missile_alert");
+
+ auto missile_transforms = this->get_components_by_name<Transform>("missile");
+ auto colliders = this->get_components_by_name<CircleCollider>("missile");
+ auto missile_behaviorscripts = this->get_components_by_name<BehaviorScript>("missile");
+ auto missile_audiosources = this->get_components_by_name<AudioSource>("missile");
+
+ auto & camera_transform = this->get_components_by_name<Transform>("camera").front().get();
+
+ for (size_t i = 0; i < missile_transforms.size(); ++i) {
+ BehaviorScript & script = missile_behaviorscripts[i * 2].get();
+ if (script.active) continue;
+ script.active = true;
+ colliders[i].get().active = true;
+ missile_audiosources[i * 2].get().play();
+
+ auto & transform = missile_transforms[i].get();
+ transform.position.x = camera_transform.position.x + this->MISSILE_OFFSET;
+ transform.position.y = Random::i(this->MAX_RANGE, this->MIN_RANGE);
+
+ auto & alert_sprite = alert_sprites[i].get();
+ alert_sprite.active = true;
+ break;
+ }
+
+ return false;
+}
diff --git a/game/missile/SpawnEvent.h b/game/missile/SpawnEvent.h
new file mode 100644
index 0000000..84b1987
--- /dev/null
+++ b/game/missile/SpawnEvent.h
@@ -0,0 +1,19 @@
+#pragma once
+
+#include <crepe/api/Event.h>
+#include <crepe/api/Script.h>
+
+#include "../Config.h"
+
+struct MissileSpawnEvent : public crepe::Event {};
+
+class MissileSpawnEventHandler : public crepe::Script {
+private:
+ static constexpr int MISSILE_OFFSET = VIEWPORT_X;
+ static constexpr int MIN_RANGE = -150;
+ static constexpr int MAX_RANGE = 150;
+
+public:
+ void init();
+ bool on_event(const MissileSpawnEvent & ev);
+};
diff --git a/game/player/PlayerAudioScript.cpp b/game/player/PlayerAudioScript.cpp
new file mode 100644
index 0000000..6b5f630
--- /dev/null
+++ b/game/player/PlayerAudioScript.cpp
@@ -0,0 +1,62 @@
+#include "PlayerAudioScript.h"
+
+#include <crepe/api/Animator.h>
+#include <crepe/api/AudioSource.h>
+
+using namespace crepe;
+using namespace std;
+
+void PlayerAudioScript::fixed_update(crepe::duration_t dt) {
+ Animator & animator = this->get_components_by_name<Animator>("player").front();
+
+ if (animator.data.col == 0) {
+ if (animator.data.row != this->last_row) {
+ if (animator.data.row == 0) {
+ // right footstep
+ if (current_footstep == 0) {
+ AudioSource & audio
+ = this->get_components_by_name<AudioSource>("player_audio").at(0);
+ audio.play();
+ } else if (current_footstep == 1) {
+ AudioSource & audio
+ = this->get_components_by_name<AudioSource>("player_audio").at(2);
+ audio.play();
+ } else if (current_footstep == 2) {
+ AudioSource & audio
+ = this->get_components_by_name<AudioSource>("player_audio").at(4);
+ audio.play();
+ } else if (current_footstep == 3) {
+ AudioSource & audio
+ = this->get_components_by_name<AudioSource>("player_audio").at(6);
+ audio.play();
+ }
+ } else if (animator.data.row == 2) {
+ // left footstep
+ if (current_footstep == 0) {
+ AudioSource & audio
+ = this->get_components_by_name<AudioSource>("player_audio").at(1);
+ audio.play();
+ current_footstep = 1;
+ } else if (current_footstep == 1) {
+ AudioSource & audio
+ = this->get_components_by_name<AudioSource>("player_audio").at(3);
+ audio.play();
+ current_footstep = 2;
+ } else if (current_footstep == 2) {
+ AudioSource & audio
+ = this->get_components_by_name<AudioSource>("player_audio").at(5);
+ audio.play();
+ current_footstep = 3;
+ } else if (current_footstep == 3) {
+ AudioSource & audio
+ = this->get_components_by_name<AudioSource>("player_audio").at(7);
+ audio.play();
+ current_footstep = 0;
+ }
+ }
+ this->last_row = animator.data.row;
+ }
+ } else {
+ this->last_row = -1;
+ }
+}
diff --git a/game/player/PlayerAudioScript.h b/game/player/PlayerAudioScript.h
new file mode 100644
index 0000000..764cb20
--- /dev/null
+++ b/game/player/PlayerAudioScript.h
@@ -0,0 +1,13 @@
+#pragma once
+
+#include <crepe/api/Event.h>
+#include <crepe/api/Script.h>
+
+class PlayerAudioScript : public crepe::Script {
+public:
+ void fixed_update(crepe::duration_t dt);
+
+private:
+ int last_row = -1;
+ int current_footstep = 0;
+};
diff --git a/game/player/PlayerBulletPool.cpp b/game/player/PlayerBulletPool.cpp
new file mode 100644
index 0000000..5285ec8
--- /dev/null
+++ b/game/player/PlayerBulletPool.cpp
@@ -0,0 +1,11 @@
+#include "PlayerBulletPool.h"
+#include "PlayerBulletSubScene.h"
+using namespace std;
+
+void PlayerBulletPool::create_bullets(crepe::Scene & scn) {
+ PlayerBulletSubScene bullet;
+ int amount = 0;
+ while (amount < this->MAXIMUM_AMOUNT) {
+ amount = bullet.create(scn, amount);
+ }
+}
diff --git a/game/player/PlayerBulletPool.h b/game/player/PlayerBulletPool.h
new file mode 100644
index 0000000..9618d54
--- /dev/null
+++ b/game/player/PlayerBulletPool.h
@@ -0,0 +1,11 @@
+#pragma once
+
+#include <crepe/api/Scene.h>
+
+class PlayerBulletPool {
+public:
+ void create_bullets(crepe::Scene & scn);
+
+private:
+ static constexpr int MAXIMUM_AMOUNT = 20;
+};
diff --git a/game/player/PlayerBulletScript.cpp b/game/player/PlayerBulletScript.cpp
new file mode 100644
index 0000000..a823375
--- /dev/null
+++ b/game/player/PlayerBulletScript.cpp
@@ -0,0 +1,41 @@
+
+#include <crepe/api/Camera.h>
+#include <crepe/api/Metadata.h>
+#include <crepe/api/Rigidbody.h>
+
+#include "PlayerBulletScript.h"
+
+using namespace crepe;
+using namespace std;
+void PlayerBulletScript::init() {
+ this->subscribe<CollisionEvent>([this](const CollisionEvent & e) -> bool {
+ return this->on_collide(e);
+ });
+}
+void PlayerBulletScript::fixed_update(crepe::duration_t dt) {
+ Transform & transform = this->get_component<Transform>();
+ Camera & camera = this->get_components_by_name<Camera>("camera").front();
+ Transform & cam_transform = this->get_components_by_name<Transform>("camera").front();
+ Rigidbody & bullet_body = this->get_component<Rigidbody>();
+ transform.rotation += bullet_body.data.angular_velocity * dt.count();
+ transform.position += bullet_body.data.linear_velocity * dt.count();
+ vec2 half_screen = camera.viewport_size / 2;
+ float despawn_location = cam_transform.position.x + half_screen.x + 50;
+ if (transform.position.x > despawn_location) {
+ this->despawn_bullet();
+ }
+}
+
+void PlayerBulletScript::despawn_bullet() {
+ Transform & transform = this->get_component<Transform>();
+ Rigidbody & bullet_body = this->get_component<Rigidbody>();
+ bullet_body.active = false;
+ BehaviorScript & bullet_script = this->get_component<BehaviorScript>();
+ bullet_script.active = false;
+ transform.position = {0, -850};
+}
+
+bool PlayerBulletScript::on_collide(const CollisionEvent & e) {
+ this->despawn_bullet();
+ return false;
+}
diff --git a/game/player/PlayerBulletScript.h b/game/player/PlayerBulletScript.h
new file mode 100644
index 0000000..0637790
--- /dev/null
+++ b/game/player/PlayerBulletScript.h
@@ -0,0 +1,11 @@
+#pragma once
+#include <crepe/api/BehaviorScript.h>
+#include <crepe/api/Script.h>
+
+class PlayerBulletScript : public crepe::Script {
+public:
+ void init() override;
+ void fixed_update(crepe::duration_t dt) override;
+ bool on_collide(const crepe::CollisionEvent & e);
+ void despawn_bullet();
+};
diff --git a/game/player/PlayerBulletSubScene.cpp b/game/player/PlayerBulletSubScene.cpp
new file mode 100644
index 0000000..82ce4a9
--- /dev/null
+++ b/game/player/PlayerBulletSubScene.cpp
@@ -0,0 +1,52 @@
+#include <string>
+
+#include "../Config.h"
+#include <crepe/api/AI.h>
+#include <crepe/api/Animator.h>
+#include <crepe/api/BehaviorScript.h>
+#include <crepe/api/BoxCollider.h>
+#include <crepe/api/CircleCollider.h>
+#include <crepe/api/GameObject.h>
+#include <crepe/api/Rigidbody.h>
+#include <crepe/api/Scene.h>
+#include <crepe/api/Sprite.h>
+
+#include "PlayerBulletScript.h"
+#include "PlayerBulletSubScene.h"
+#include "PlayerScript.h"
+using namespace crepe;
+using namespace std;
+int PlayerBulletSubScene::create(Scene & scn, int counter) {
+ string unique_name = "player_bullet_" + to_string(counter++);
+ GameObject player_bullet
+ = scn.new_object(unique_name.c_str(), "player_bullet", vec2 {0, -850}, 0, 1);
+
+ Rigidbody & player_bullet_body = player_bullet.add_component<Rigidbody>(Rigidbody::Data {
+ .gravity_scale = 0,
+ .body_type = Rigidbody::BodyType::KINEMATIC,
+ .linear_velocity = vec2 {450, 0},
+ .angular_velocity = 300,
+ .kinematic_collision = false,
+ .collision_layers = {COLL_LAY_ENEMY, COLL_LAY_ZAPPER},
+
+ .collision_layer = COLL_LAY_PLAYER_BULLET,
+
+ });
+ player_bullet_body.active = false;
+ BoxCollider & player_bullet_collider
+ = player_bullet.add_component<BoxCollider>(vec2(30, 30));
+
+ Asset player_bullet_asset {"asset/other_effects/crepe.png"};
+ Sprite & player_bullet_sprite = player_bullet.add_component<Sprite>(
+ player_bullet_asset,
+ Sprite::Data {
+ .flip = {true, false},
+ .sorting_in_layer = SORT_IN_LAY_OBSTACLES,
+ .order_in_layer = 1,
+ .size = vec2(30, 0),
+ }
+ );
+ player_bullet.add_component<BehaviorScript>().set_script<PlayerBulletScript>().active
+ = false;
+ return counter;
+}
diff --git a/game/player/PlayerBulletSubScene.h b/game/player/PlayerBulletSubScene.h
new file mode 100644
index 0000000..72eda62
--- /dev/null
+++ b/game/player/PlayerBulletSubScene.h
@@ -0,0 +1,10 @@
+#pragma once
+
+namespace crepe {
+class Scene;
+}
+
+class PlayerBulletSubScene {
+public:
+ int create(crepe::Scene & scn, int counter);
+};
diff --git a/game/player/PlayerEndScript.cpp b/game/player/PlayerEndScript.cpp
new file mode 100644
index 0000000..62fd350
--- /dev/null
+++ b/game/player/PlayerEndScript.cpp
@@ -0,0 +1,104 @@
+#include "PlayerEndScript.h"
+
+#include "../Config.h"
+#include "../Events.h"
+#include "manager/LoopTimerManager.h"
+
+#include <crepe/api/Animator.h>
+#include <crepe/api/BoxCollider.h>
+#include <crepe/api/CircleCollider.h>
+#include <crepe/api/Rigidbody.h>
+#include <crepe/types.h>
+
+using namespace crepe;
+using namespace std;
+
+void PlayerEndScript::init() {
+ Rigidbody & rb_player = this->get_components_by_name<Rigidbody>("player").front();
+ rb_player.data.elasticity_coefficient = 0.7;
+
+ subscribe<CollisionEvent>([this](const CollisionEvent & ev) -> bool {
+ return this->on_collision(ev);
+ });
+}
+
+bool PlayerEndScript::on_collision(const crepe::CollisionEvent & ev) {
+ if (ev.info.other.metadata.name == "floor") {
+ Transform & transform_player
+ = this->get_components_by_name<Transform>("player").front();
+ RefVector<Animator> anim_player = this->get_components_by_name<Animator>("player");
+ Rigidbody & rb_player = this->get_components_by_name<Rigidbody>("player").front();
+ Rigidbody & rb_camera = this->get_components_by_name<Rigidbody>("camera").front();
+
+ float dt = this->get_loop_timer().get_fixed_delta_time().count();
+
+ if (jump == 0) {
+ int random_number = rand() % 4;
+ for (Animator & anim : anim_player) {
+ anim.active = false;
+ anim.set_anim(6);
+ for (int i = 0; i < random_number; i++) {
+ anim.next_anim();
+ }
+ }
+ } else if (jump == 1) {
+ for (Animator & anim : anim_player) {
+ anim.next_anim();
+ }
+ }
+
+ if (jump == 0) {
+ rb_player.data.angular_velocity = 16000 * dt;
+ rb_player.data.angular_velocity_coefficient = 0.7;
+ jump++;
+ } else if (jump == 1) {
+ jump++;
+ } else if (jump == 2) {
+ RefVector<Rigidbody> rb_back_forest
+ = this->get_components_by_tag<Rigidbody>("forest_background");
+ for (Rigidbody & rb : rb_back_forest) {
+ rb.data.linear_velocity_coefficient = vec2(0.5, 0.5);
+ }
+
+ rb_player.data.angular_velocity = 0;
+ rb_player.data.elasticity_coefficient = 0;
+ if (rb_player.data.linear_velocity.x != 0) {
+ rb_player.data.linear_velocity = vec2(PLAYER_SPEED * dt, 0);
+ }
+ rb_player.data.linear_velocity_coefficient = vec2(0.5, 0.5);
+ rb_camera.data.linear_velocity_coefficient = vec2(0.5, 0.5);
+ for (Animator & anim : anim_player) {
+ anim.active = false;
+ anim.set_anim(7);
+ }
+ if (transform_player.rotation > 0 && transform_player.rotation < 90) {
+ // Do not call next_anim()
+ } else if (transform_player.rotation > 90 && transform_player.rotation < 180) {
+ for (Animator & anim : anim_player) {
+ anim.next_anim();
+ }
+ } else if (transform_player.rotation > 180 && transform_player.rotation < 270) {
+ for (Animator & anim : anim_player) {
+ anim.next_anim();
+ anim.next_anim();
+ }
+ } else {
+ for (Animator & anim : anim_player) {
+ anim.next_anim();
+ anim.next_anim();
+ anim.next_anim();
+ }
+ }
+ jump++;
+ }
+
+ if (rb_player.data.linear_velocity.x < 5 && jump == 3) {
+ this->trigger_event<EndGameEvent>();
+ jump++;
+ }
+
+ return false;
+ }
+
+ return false;
+}
diff --git a/game/player/PlayerEndScript.h b/game/player/PlayerEndScript.h
new file mode 100644
index 0000000..03ea8a9
--- /dev/null
+++ b/game/player/PlayerEndScript.h
@@ -0,0 +1,14 @@
+#pragma once
+
+#include <crepe/api/Script.h>
+
+class PlayerEndScript : public crepe::Script {
+public:
+ void init();
+
+private:
+ bool on_collision(const crepe::CollisionEvent & ev);
+
+private:
+ int jump = 0;
+};
diff --git a/game/player/PlayerScript.cpp b/game/player/PlayerScript.cpp
new file mode 100644
index 0000000..8cbe8dc
--- /dev/null
+++ b/game/player/PlayerScript.cpp
@@ -0,0 +1,214 @@
+#include "PlayerScript.h"
+
+#include "../Config.h"
+#include "../enemy/BattleScript.h"
+
+#include <crepe/api/Animator.h>
+#include <crepe/api/AudioSource.h>
+#include <crepe/api/BoxCollider.h>
+#include <crepe/api/ParticleEmitter.h>
+#include <crepe/api/Rigidbody.h>
+#include <crepe/api/Transform.h>
+#include <crepe/types.h>
+
+using namespace crepe;
+using namespace std;
+
+void PlayerScript::init() {
+ subscribe<CollisionEvent>([this](const CollisionEvent & ev) -> bool {
+ return this->on_collision(ev);
+ });
+ subscribe<KeyPressEvent>([this](const KeyPressEvent & ev) -> bool {
+ if (ev.repeat) return false;
+ return this->on_key_down(ev);
+ });
+ subscribe<KeyReleaseEvent>([this](const KeyReleaseEvent & ev) -> bool {
+ return this->on_key_up(ev);
+ });
+ this->last_fired = std::chrono::steady_clock::now();
+ this->body = get_component<Rigidbody>();
+}
+
+bool PlayerScript::on_key_down(const KeyPressEvent & ev) {
+ if (ev.key == Keycode::SPACE) {
+ const vec2 UP = {0, -1};
+ this->help_kick(UP);
+ }
+ return false;
+}
+
+bool PlayerScript::on_key_up(const KeyReleaseEvent & ev) {
+ if (ev.key == Keycode::SPACE) {
+ const vec2 DOWN = {0, 1};
+ this->help_kick(DOWN);
+ }
+ return false;
+}
+
+void PlayerScript::help_kick(const vec2 & direction) {
+ // softly "kick" the player (at start/end of flight)
+ vec2 & velocity = this->body->data.linear_velocity;
+ float kick_amount = std::min(
+ velocity.length() * PLAYER_HELP_KICK_SCALE, engine_gravity * PLAYER_HELP_KICK_MAX
+ );
+ velocity += direction * kick_amount;
+}
+
+bool PlayerScript::on_collision(const CollisionEvent & ev) {
+ BehaviorScript & play_scr = this->get_components_by_name<BehaviorScript>("player").front();
+ BehaviorScript & end_scr = this->get_components_by_name<BehaviorScript>("player").back();
+ RefVector<Animator> animators = this->get_components_by_name<Animator>("player");
+ RefVector<ParticleEmitter> emitters
+ = this->get_components_by_name<ParticleEmitter>("player");
+
+ if (ev.info.other.metadata.tag == "zapper") {
+ for (Animator & anim : animators) {
+ anim.active = true;
+ anim.set_anim(4);
+ anim.data.looping = true;
+ prev_anim = 0;
+ }
+ for (ParticleEmitter & emitter : emitters) {
+ emitter.data.emission_rate = 0;
+ }
+ play_scr.active = false;
+ end_scr.active = true;
+
+ AudioSource & audio = this->get_components_by_name<AudioSource>("player").at(0);
+ audio.play();
+
+ return false;
+ } else if (ev.info.other.metadata.tag == "laser") {
+ for (Animator & anim : animators) {
+ anim.active = true;
+ anim.set_anim(4);
+ anim.data.looping = true;
+ prev_anim = 0;
+ }
+ for (ParticleEmitter & emitter : emitters) {
+ emitter.data.emission_rate = 0;
+ }
+ play_scr.active = false;
+ end_scr.active = true;
+
+ AudioSource & audio = this->get_components_by_name<AudioSource>("player").at(1);
+ audio.play();
+
+ return false;
+ } else if (ev.info.other.metadata.tag == "missile"
+ || ev.info.other.metadata.tag == "enemy_bullet") {
+ for (Animator & anim : animators) {
+ anim.active = true;
+ anim.set_anim(5);
+ anim.data.looping = true;
+ prev_anim = 0;
+ }
+ for (ParticleEmitter & emitter : emitters) {
+ emitter.data.emission_rate = 0;
+ }
+ play_scr.active = false;
+ end_scr.active = true;
+
+ AudioSource & audio = this->get_components_by_name<AudioSource>("player").at(2);
+ audio.play();
+
+ return false;
+ }
+
+ return false;
+}
+
+void PlayerScript::fixed_update(crepe::duration_t dt) {
+ RefVector<Animator> animators = this->get_components_by_name<Animator>("player");
+ RefVector<ParticleEmitter> emitters
+ = this->get_components_by_name<ParticleEmitter>("player");
+ Transform & transform = this->get_components_by_name<Transform>("player").front();
+
+ for (ParticleEmitter & emitter : emitters) {
+ emitter.data.boundary.offset = vec2(0, -transform.position.y);
+ }
+
+ Rigidbody & rb = this->body;
+ if (this->get_key_state(Keycode::ENTER)) {
+
+ auto now = std::chrono::steady_clock::now();
+ std::chrono::duration<float> elapsed = now - last_fired;
+ if (elapsed > shot_delay) {
+ this->shoot(transform.position, 0);
+ last_fired = now;
+ }
+ }
+ if (this->get_key_state(Keycode::SPACE)) {
+ rb.add_force_linear(
+ vec2(0, -1) * (engine_gravity * PLAYER_GRAVITY_SCALE * dt.count())
+ );
+
+ if (prev_anim != 1) {
+ for (Animator & anim : animators) {
+ anim.active = true;
+ anim.set_anim(1);
+ anim.data.looping = true;
+ prev_anim = 1;
+ }
+ for (ParticleEmitter & emitter : emitters) {
+ emitter.data.emission_rate = 30;
+ }
+ }
+
+ AudioSource & audio = this->get_components_by_name<AudioSource>("player").at(
+ 3 + current_jetpack_sound
+ );
+ audio.play();
+ current_jetpack_sound++;
+ if (current_jetpack_sound > 7) {
+ current_jetpack_sound = 0;
+ }
+ } else if (transform.position.y == 200) {
+ Rigidbody & rb = this->body;
+ if (prev_anim != 0 && rb.data.linear_velocity.x != 0) {
+ for (Animator & anim : animators) {
+ anim.active = true;
+ anim.set_anim(0);
+ anim.data.looping = true;
+ prev_anim = 0;
+ }
+ for (ParticleEmitter & emitter : emitters) {
+ emitter.data.emission_rate = 0;
+ }
+ }
+ } else {
+ if (prev_anim != 2) {
+ for (Animator & anim : animators) {
+ anim.set_anim(2);
+ anim.data.looping = false;
+ prev_anim = 2;
+ }
+ for (ParticleEmitter & emitter : emitters) {
+ emitter.data.emission_rate = 0;
+ }
+ }
+ }
+}
+
+void PlayerScript::shoot(const vec2 & location, float angle) {
+ RefVector<Transform> bullet_transforms
+ = this->get_components_by_tag<Transform>("player_bullet");
+
+ for (Transform & bullet_pos : bullet_transforms) {
+ if (bullet_pos.position.x == 0 && bullet_pos.position.y == -850) {
+
+ bullet_pos.position = location;
+ bullet_pos.position.x += 20;
+ Rigidbody & bullet_body
+ = this->get_components_by_id<Rigidbody>(bullet_pos.game_object_id).front();
+ BoxCollider bullet_collider
+ = this->get_components_by_id<BoxCollider>(bullet_pos.game_object_id).front();
+ bullet_body.active = true;
+ BehaviorScript & bullet_script
+ = this->get_components_by_id<BehaviorScript>(bullet_pos.game_object_id)
+ .front();
+ bullet_script.active = true;
+ return;
+ }
+ }
+}
diff --git a/game/player/PlayerScript.h b/game/player/PlayerScript.h
new file mode 100644
index 0000000..6a7dedb
--- /dev/null
+++ b/game/player/PlayerScript.h
@@ -0,0 +1,32 @@
+#pragma once
+
+#include "util/OptionalRef.h"
+#include <chrono>
+#include <crepe/api/Config.h>
+#include <crepe/api/Event.h>
+#include <crepe/api/Script.h>
+
+class PlayerScript : public crepe::Script {
+public:
+ void init();
+ void fixed_update(crepe::duration_t dt);
+
+private:
+ bool on_collision(const crepe::CollisionEvent & ev);
+ bool on_key_down(const crepe::KeyPressEvent & ev);
+ bool on_key_up(const crepe::KeyReleaseEvent & ev);
+ // bool on_key_up(const crepe::KeyReleaseEvent& ev);
+ void shoot(const crepe::vec2 & location, float angle);
+ void help_kick(const crepe::vec2 & direction);
+
+private:
+ int prev_anim = 0;
+ std::chrono::time_point<std::chrono::steady_clock> last_fired;
+ std::chrono::time_point<std::chrono::steady_clock> last_switched;
+ std::chrono::duration<float> shot_delay = std::chrono::duration<float>(0.5);
+ std::chrono::duration<float> switch_delay = std::chrono::duration<float>(0.01);
+ int current_jetpack_sound = 0;
+
+ float & engine_gravity = crepe::Config::get_instance().physics.gravity;
+ crepe::OptionalRef<crepe::Rigidbody> body;
+};
diff --git a/game/player/PlayerSubScene.cpp b/game/player/PlayerSubScene.cpp
new file mode 100644
index 0000000..d0142e0
--- /dev/null
+++ b/game/player/PlayerSubScene.cpp
@@ -0,0 +1,222 @@
+#include "PlayerSubScene.h"
+#include "PlayerAudioScript.h"
+#include "PlayerEndScript.h"
+#include "PlayerScript.h"
+
+#include "../Config.h"
+#include "../coins/CoinScript.h"
+
+#include <crepe/ValueBroker.h>
+#include <crepe/api/Animator.h>
+#include <crepe/api/AudioSource.h>
+#include <crepe/api/BoxCollider.h>
+#include <crepe/api/CircleCollider.h>
+#include <crepe/api/GameObject.h>
+#include <crepe/api/ParticleEmitter.h>
+#include <crepe/api/Scene.h>
+#include <crepe/api/Script.h>
+#include <crepe/api/Sprite.h>
+#include <crepe/manager/SaveManager.h>
+#include <crepe/types.h>
+
+using namespace crepe;
+using namespace std;
+
+PlayerSubScene::PlayerSubScene(Scene & scn) {
+ GameObject player = scn.new_object("player", "player", vec2(-100, 200));
+
+ SaveManager & save = scn.get_save_manager();
+ ValueBroker<int> particle_type = save.get<int>(JETPACK_PARTICLES, 0);
+
+ string player_bullet_string;
+ string player_bullet_x2_string;
+ string player_shell_string;
+ if (particle_type.get() == 0) {
+ player_bullet_string = "asset/other_effects/effect_smgbullet.png";
+ player_bullet_x2_string = "asset/other_effects/effect_smgbullet_x2.png";
+ player_shell_string = "asset/other_effects/effect_rocketmgshell_TVOS.png";
+ } else {
+ player_bullet_string = "asset/background/aquarium/bubble.png";
+ player_bullet_x2_string = "asset/background/aquarium/bubble.png";
+ player_shell_string = "asset/background/aquarium/bubble.png";
+ }
+
+ Asset player_bullet {player_bullet_string};
+ Sprite & player_bullet_sprite = player.add_component<Sprite>(
+ player_bullet,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_PLAYER,
+ .order_in_layer = 3,
+ .size = vec2(0, 6),
+ }
+ );
+ player.add_component<ParticleEmitter>(player_bullet_sprite, ParticleEmitter::Data{
+ .offset = vec2(-15, 15),
+ .emission_rate = 0,
+ .min_speed = 300,
+ .max_speed = 500,
+ .min_angle = 85,
+ .max_angle = 100,
+ .boundary = ParticleEmitter::Boundary {
+ .height = 400,
+ .reset_on_exit = true,
+ },
+ });
+ Asset player_bullet_x2 {player_bullet_x2_string};
+ Sprite & player_bullet_x2_sprite = player.add_component<Sprite>(
+ player_bullet_x2,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_PLAYER,
+ .order_in_layer = 3,
+ .size = vec2(0, 12),
+ }
+ );
+ player.add_component<ParticleEmitter>(player_bullet_x2_sprite, ParticleEmitter::Data{
+ .offset = vec2(-15, 15),
+ .emission_rate = 0,
+ .min_speed = 300,
+ .max_speed = 500,
+ .min_angle = 85,
+ .max_angle = 100,
+ .boundary = ParticleEmitter::Boundary {
+ .height = 400,
+ .reset_on_exit = true,
+ },
+ });
+ Asset player_shell {player_shell_string};
+ Sprite & player_shell_sprite = player.add_component<Sprite>(
+ player_shell,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_PLAYER,
+ .order_in_layer = 3,
+ .size = vec2(0, 12),
+ .angle_offset = 90,
+ }
+ );
+ player.add_component<ParticleEmitter>(player_shell_sprite, ParticleEmitter::Data{
+ .offset = vec2(-15, 15),
+ .emission_rate = 0,
+ .min_speed = 200,
+ .max_speed = 500,
+ .min_angle = 110,
+ .max_angle = 120,
+ .force_over_time = vec2(0, 1000),
+ .boundary = ParticleEmitter::Boundary {
+ .height = 400,
+ .reset_on_exit = true,
+ },
+ });
+
+ Asset player_body_asset {"asset/barry/defaultBody.png"};
+ Sprite & player_body_sprite = player.add_component<Sprite>(
+ player_body_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_PLAYER,
+ .order_in_layer = 0,
+ .size = vec2(0, 50),
+ }
+ );
+ player.add_component<Animator>(
+ player_body_sprite, ivec2(32, 32), uvec2(4, 8),
+ Animator::Data {
+ .fps = 5,
+ .looping = true,
+ }
+ );
+ player.add_component<BoxCollider>(vec2(50, 35));
+ Asset player_head_asset {"asset/barry/defaultHead.png"};
+ Sprite & player_head_sprite = player.add_component<Sprite>(
+ player_head_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_PLAYER,
+ .order_in_layer = 1,
+ .size = vec2(0, 50),
+ .position_offset = vec2(0, -20),
+ }
+ );
+ player.add_component<Animator>(
+ player_head_sprite, ivec2(32, 32), uvec2(4, 8),
+ Animator::Data {
+ .fps = 5,
+ .looping = true,
+ }
+ );
+ player.add_component<CircleCollider>(25, vec2(0, -20));
+ Asset player_jetpack_asset {"asset/barry/jetpackDefault.png"};
+ Sprite & player_jetpack_sprite = player.add_component<Sprite>(
+ player_jetpack_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_PLAYER,
+ .order_in_layer = 2,
+ .size = vec2(0, 60),
+ .position_offset = vec2(-20, 0),
+ }
+ );
+ player_jetpack_sprite.active = false;
+ player.add_component<Animator>(
+ player_jetpack_sprite, ivec2(32, 44), uvec2(4, 4),
+ Animator::Data {
+ .fps = 5,
+ .looping = true,
+ }
+ );
+ player.add_component<BoxCollider>(vec2(40, 50), vec2(-20, 0));
+ player.add_component<Rigidbody>(Rigidbody::Data {
+ .gravity_scale = 1.0,
+ .body_type = Rigidbody::BodyType::DYNAMIC,
+ .linear_velocity = vec2(PLAYER_SPEED * 0.02, 0),
+ .collision_layers
+ = {COLL_LAY_BOT_TOP, COLL_LAY_ZAPPER, COLL_LAY_LASER, COLL_LAY_MISSILE, COLL_LAY_BULLET
+ },
+ .collision_layer = COLL_LAY_PLAYER,
+ });
+
+ player.add_component<BehaviorScript>().set_script<PlayerScript>().active = false;
+ player.add_component<BehaviorScript>().set_script<CoinScript>();
+ player.add_component<BehaviorScript>().set_script<PlayerEndScript>().active = false;
+
+ player.add_component<AudioSource>(Asset("asset/sfx/dud_zapper_lp.ogg"));
+ player.add_component<AudioSource>(Asset("asset/sfx/dud_zapper_pop.ogg"));
+ player.add_component<AudioSource>(Asset("asset/sfx/dud_fire.ogg"));
+ player.add_component<AudioSource>(Asset("asset/sfx/jetpack_firecracker_lp_01.ogg")).volume
+ = 0.1;
+ player.add_component<AudioSource>(Asset("asset/sfx/jetpack_firecracker_lp_02.ogg")).volume
+ = 0.1;
+ player.add_component<AudioSource>(Asset("asset/sfx/jetpack_firecracker_lp_03.ogg")).volume
+ = 0.1;
+ player.add_component<AudioSource>(Asset("asset/sfx/jetpack_firecracker_lp_04.ogg")).volume
+ = 0.1;
+ player.add_component<AudioSource>(Asset("asset/sfx/jetpack_firecracker_lp_05.ogg")).volume
+ = 0.1;
+ player.add_component<AudioSource>(Asset("asset/sfx/jetpack_firecracker_lp_06.ogg")).volume
+ = 0.1;
+ player.add_component<AudioSource>(Asset("asset/sfx/jetpack_firecracker_lp_07.ogg")).volume
+ = 0.1;
+ player.add_component<AudioSource>(Asset("asset/sfx/jetpack_firecracker_lp_08.ogg")).volume
+ = 0.1;
+
+ GameObject player_audio = scn.new_object("player_audio", "player_audio", vec2(0, 0));
+ player_audio.add_component<AudioSource>(Asset("asset/sfx/barefoot_step_left_1.ogg")).volume
+ = 3.0;
+ player_audio.add_component<AudioSource>(Asset("asset/sfx/barefoot_step_right_1.ogg"))
+ .volume
+ = 3.0;
+ player_audio.add_component<AudioSource>(Asset("asset/sfx/barefoot_step_left_2.ogg")).volume
+ = 3.0;
+ player_audio.add_component<AudioSource>(Asset("asset/sfx/barefoot_step_right_2.ogg"))
+ .volume
+ = 3.0;
+ player_audio.add_component<AudioSource>(Asset("asset/sfx/barefoot_step_left_3.ogg")).volume
+ = 3.0;
+ player_audio.add_component<AudioSource>(Asset("asset/sfx/barefoot_step_right_3.ogg"))
+ .volume
+ = 3.0;
+ player_audio.add_component<AudioSource>(Asset("asset/sfx/barefoot_step_left_4.ogg")).volume
+ = 3.0;
+ player_audio.add_component<AudioSource>(Asset("asset/sfx/barefoot_step_right_4.ogg"))
+ .volume
+ = 3.0;
+
+ player_audio.add_component<BehaviorScript>().set_script<PlayerAudioScript>().active
+ = false;
+}
diff --git a/game/player/PlayerSubScene.h b/game/player/PlayerSubScene.h
new file mode 100644
index 0000000..bf94c32
--- /dev/null
+++ b/game/player/PlayerSubScene.h
@@ -0,0 +1,10 @@
+#pragma once
+
+namespace crepe {
+class Scene;
+}
+
+class PlayerSubScene {
+public:
+ PlayerSubScene(crepe::Scene & scn);
+};
diff --git a/game/prefab/CMakeLists.txt b/game/prefab/CMakeLists.txt
new file mode 100644
index 0000000..6c36ef2
--- /dev/null
+++ b/game/prefab/CMakeLists.txt
@@ -0,0 +1,6 @@
+target_sources(main PUBLIC
+ ZapperObject.cpp
+ ZapperPoolSubScene.cpp
+ ZapperPoolScript.cpp
+)
+
diff --git a/game/prefab/ZapperObject.cpp b/game/prefab/ZapperObject.cpp
new file mode 100644
index 0000000..24bbbd2
--- /dev/null
+++ b/game/prefab/ZapperObject.cpp
@@ -0,0 +1,118 @@
+#include <crepe/api/Transform.h>
+
+#include "Config.h"
+#include "ZapperObject.h"
+
+using namespace crepe;
+
+ZapperObject::ZapperObject(crepe::GameObject && base)
+ : GameObject(std::move(base)),
+ sprite {
+ .orb_start = add_component<Sprite>(
+ Asset {"asset/obstacles/zapper/orbAnim.png"},
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_OBSTACLES,
+ .order_in_layer = 1,
+ .size = vec2(0, 50) * SCALE,
+ }
+ ),
+ .orb_end = add_component<Sprite>(
+ sprite.orb_start.source,
+ Sprite::Data {
+ .flip = {true, true},
+ .sorting_in_layer = SORT_IN_LAY_OBSTACLES,
+ .order_in_layer = 1,
+ .size = vec2(0, 50) * SCALE,
+ }
+ ),
+ .glow_start = add_component<Sprite>(
+ Asset {"asset/obstacles/zapper/regular_zappers/glow.png"},
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_OBSTACLES,
+ .order_in_layer = -1,
+ .size = vec2(128, 128) * SCALE,
+ }
+ ),
+ .glow_end = add_component<Sprite>(
+ sprite.glow_start.source,
+ Sprite::Data {
+ .flip = {true, true},
+ .sorting_in_layer = SORT_IN_LAY_OBSTACLES,
+ .order_in_layer = -1,
+ .size = vec2(128, 128) * SCALE,
+ }
+ ),
+ .beam = add_component<Sprite>(
+ Asset {"asset/obstacles/zapper/regular_zappers/zapEffect.png"},
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_OBSTACLES,
+ .order_in_layer = 0,
+ .size = vec2(0, 40 * SCALE),
+ .angle_offset = 90,
+ }
+ ),
+ },
+ animator {
+ .orb_start = add_component<Animator>(
+ sprite.orb_start, ivec2(62, 42), uvec2(4, 1),
+ Animator::Data {
+ .fps = 10,
+ .looping = true,
+ }
+ ),
+ .orb_end = add_component<Animator>(
+ sprite.orb_end, ivec2(62, 42), uvec2(4, 1), animator.orb_start.data
+ ),
+ .glow_start = add_component<Animator>(
+ sprite.glow_start, ivec2(128, 128), uvec2(16, 1),
+ Animator::Data {
+ .fps = 30,
+ .looping = true,
+ }
+ ),
+ .glow_end = add_component<Animator>(
+ sprite.glow_end, ivec2(128, 128), uvec2(16, 1), animator.glow_start.data
+ ),
+ },
+ body {add_component<Rigidbody>(Rigidbody::Data {
+ .body_type = Rigidbody::BodyType::KINEMATIC,
+ .kinematic_collision = false,
+ .collision_layer = COLL_LAY_ZAPPER,
+ })},
+ collider {add_component<BoxCollider>(vec2(0, 0))} {
+ this->set_active(false);
+}
+
+void ZapperObject::place(const crepe::vec2 & position, float rotation, float length) {
+ this->transform.position = position;
+ this->transform.rotation = rotation;
+
+ vec2 offset = vec2(0, 1) * length / 2;
+
+ this->sprite.orb_start.data.position_offset = offset;
+ this->sprite.glow_start.data.position_offset = offset;
+ this->sprite.orb_end.data.position_offset = -offset;
+ this->sprite.glow_end.data.position_offset = -offset;
+
+ this->sprite.beam.data.size.x = length;
+
+ this->collider.dimensions = offset.rotate(rotation) * 2 + vec2(30, 30) * SCALE;
+}
+
+void ZapperObject::set_active(bool active) {
+ this->sprite.orb_start.active = active;
+ this->sprite.orb_end.active = active;
+ this->sprite.glow_start.active = active;
+ this->sprite.glow_end.active = active;
+ this->sprite.beam.active = active;
+
+ this->animator.orb_start.active = active;
+ this->animator.orb_end.active = active;
+ this->animator.glow_start.active = active;
+ this->animator.glow_end.active = active;
+
+ this->body.active = active;
+ this->collider.active = active;
+
+ this->active = active;
+}
diff --git a/game/prefab/ZapperObject.h b/game/prefab/ZapperObject.h
new file mode 100644
index 0000000..42edc51
--- /dev/null
+++ b/game/prefab/ZapperObject.h
@@ -0,0 +1,40 @@
+#pragma once
+
+#include <crepe/api/Animator.h>
+#include <crepe/api/BoxCollider.h>
+#include <crepe/api/GameObject.h>
+#include <crepe/api/Rigidbody.h>
+#include <crepe/api/Sprite.h>
+
+class ZapperObject : public crepe::GameObject {
+public:
+ ZapperObject(crepe::GameObject &&);
+
+public:
+ bool active = true;
+
+ struct {
+ crepe::Sprite & orb_start;
+ crepe::Sprite & orb_end;
+ crepe::Sprite & glow_start;
+ crepe::Sprite & glow_end;
+ crepe::Sprite & beam;
+ } sprite;
+
+ struct {
+ crepe::Animator & orb_start;
+ crepe::Animator & orb_end;
+ crepe::Animator & glow_start;
+ crepe::Animator & glow_end;
+ } animator;
+
+ crepe::Rigidbody & body;
+ crepe::BoxCollider & collider;
+
+private:
+ static constexpr float SCALE = 0.8;
+
+public:
+ void place(const crepe::vec2 & position, float rotation, float length);
+ void set_active(bool active);
+};
diff --git a/game/prefab/ZapperPoolScript.cpp b/game/prefab/ZapperPoolScript.cpp
new file mode 100644
index 0000000..b832ddd
--- /dev/null
+++ b/game/prefab/ZapperPoolScript.cpp
@@ -0,0 +1,70 @@
+#include <crepe/api/Camera.h>
+
+#include "../Config.h"
+#include "../Random.h"
+
+#include "ZapperPoolScript.h"
+#include "ZapperPoolSubScene.h"
+#include "util/OptionalRef.h"
+
+using namespace crepe;
+using namespace std;
+
+ZapperPoolScript::ZapperPoolScript(std::vector<ZapperObject> && pool) : pool(pool) {}
+
+void ZapperPoolScript::init() {
+ subscribe<CreateZapperEvent>([this](const CreateZapperEvent &) {
+ this->spawn_random();
+ return true;
+ });
+
+ camera_transform = get_components_by_name<Transform>(CAMERA_NAME).back();
+ camera_camera = get_components_by_name<Camera>(CAMERA_NAME).back();
+}
+
+void ZapperPoolScript::fixed_update(crepe::duration_t) {
+ float threshold
+ = camera_transform->position.x - camera_camera->viewport_size.x / 2 - OFFSCREEN_MARGIN;
+ for (ZapperObject & zapper : this->pool) {
+ if (!zapper.active) continue;
+
+ if (zapper.transform.position.x < threshold) zapper.set_active(false);
+ }
+}
+
+void ZapperPoolScript::spawn_random() {
+ OptionalRef<ZapperObject> zapper = this->get_next_zapper();
+ if (!zapper) return; // pool exhausted
+
+ bool horizontal = Random::b();
+ vec2 pos;
+ float rotation, length;
+ pos.x
+ = camera_transform->position.x + camera_camera->viewport_size.x / 2 + OFFSCREEN_MARGIN;
+
+ if (horizontal) {
+ rotation = 90;
+ length = Random::f(400, 200);
+ pos.y = Random::f(0.5, -0.5) * HALLWAY_HEIGHT;
+ // align zapper to right edge of camera
+ pos.x -= MAX_LENGTH - (length / 2);
+ } else {
+ rotation = 0;
+ length = Random::f(200, 75);
+ // ensure zapper doesn't crash into ceiling or floor
+ pos.y = Random::f(0.5, -0.5) * (HALLWAY_HEIGHT - length);
+ // align zapper to right edge of camera
+ pos.x -= MAX_LENGTH;
+ }
+
+ zapper->place(pos, rotation, length);
+ zapper->set_active(true);
+}
+
+OptionalRef<ZapperObject> ZapperPoolScript::get_next_zapper() {
+ for (ZapperObject & zapper : this->pool) {
+ if (zapper.active) continue;
+ return zapper;
+ }
+ return {};
+}
diff --git a/game/prefab/ZapperPoolScript.h b/game/prefab/ZapperPoolScript.h
new file mode 100644
index 0000000..2208c80
--- /dev/null
+++ b/game/prefab/ZapperPoolScript.h
@@ -0,0 +1,33 @@
+#pragma once
+
+#include <crepe/api/Script.h>
+
+#include "ZapperObject.h"
+#include "util/OptionalRef.h"
+
+class ZapperPoolSubScene;
+
+class ZapperPoolScript : public crepe::Script {
+public:
+ ZapperPoolScript(std::vector<ZapperObject> && pool);
+
+ void init();
+ void fixed_update(crepe::duration_t);
+
+ unsigned i = 0;
+
+private:
+ std::vector<ZapperObject> pool;
+
+private:
+ crepe::OptionalRef<crepe::Transform> camera_transform;
+ crepe::OptionalRef<crepe::Camera> camera_camera;
+ crepe::OptionalRef<ZapperObject> get_next_zapper();
+
+private:
+ void spawn_random();
+
+private:
+ static constexpr float MAX_LENGTH = 400;
+ static constexpr float OFFSCREEN_MARGIN = 50 + MAX_LENGTH;
+};
diff --git a/game/prefab/ZapperPoolSubScene.cpp b/game/prefab/ZapperPoolSubScene.cpp
new file mode 100644
index 0000000..a52aa75
--- /dev/null
+++ b/game/prefab/ZapperPoolSubScene.cpp
@@ -0,0 +1,17 @@
+#include <crepe/api/BehaviorScript.h>
+
+#include "ZapperPoolScript.h"
+#include "ZapperPoolSubScene.h"
+
+using namespace crepe;
+using namespace std;
+
+ZapperPoolSubScene::ZapperPoolSubScene(Scene & scene)
+ : controller {scene.new_object("controller")} {
+
+ vector<ZapperObject> pool;
+ for (size_t i = 0; i < this->POOL_SIZE; i++)
+ pool.emplace_back(scene.new_object("zapper", "zapper"));
+ BehaviorScript & behavior = this->controller.add_component<BehaviorScript>();
+ behavior.set_script<ZapperPoolScript>(std::move(pool));
+}
diff --git a/game/prefab/ZapperPoolSubScene.h b/game/prefab/ZapperPoolSubScene.h
new file mode 100644
index 0000000..6f6e297
--- /dev/null
+++ b/game/prefab/ZapperPoolSubScene.h
@@ -0,0 +1,19 @@
+#pragma once
+
+#include <crepe/api/Event.h>
+#include <crepe/api/GameObject.h>
+#include <crepe/api/Scene.h>
+#include <crepe/util/OptionalRef.h>
+
+class CreateZapperEvent : public crepe::Event {};
+
+class ZapperPoolSubScene {
+public:
+ ZapperPoolSubScene(crepe::Scene & scene);
+
+private:
+ crepe::GameObject controller;
+
+private:
+ static constexpr size_t POOL_SIZE = 4;
+};
diff --git a/game/preview/NpcScript.cpp b/game/preview/NpcScript.cpp
new file mode 100644
index 0000000..5a93c2b
--- /dev/null
+++ b/game/preview/NpcScript.cpp
@@ -0,0 +1,27 @@
+#include "NpcScript.h"
+
+#include <crepe/api/Sprite.h>
+#include <crepe/api/Transform.h>
+#include <crepe/manager/SaveManager.h>
+
+using namespace std;
+using namespace crepe;
+
+void NpcScript::fixed_update(duration_t dt) {
+ auto & rb = this->get_component<Rigidbody>();
+ auto & npc = this->get_component<Sprite>();
+ auto & transform = this->get_component<Transform>();
+
+ if (transform.position.x < 200) {
+ rb.data.linear_velocity.x *= -1;
+ }
+ if (transform.position.x > 700) {
+ rb.data.linear_velocity.x *= -1;
+ }
+
+ if (rb.data.linear_velocity.x < 0) {
+ npc.data.flip = {true, false};
+ } else {
+ npc.data.flip = {false, false};
+ }
+}
diff --git a/game/preview/NpcScript.h b/game/preview/NpcScript.h
new file mode 100644
index 0000000..d278f83
--- /dev/null
+++ b/game/preview/NpcScript.h
@@ -0,0 +1,8 @@
+#pragma once
+
+#include <crepe/api/Script.h>
+
+class NpcScript : public crepe::Script {
+public:
+ void fixed_update(crepe::duration_t dt);
+};
diff --git a/game/preview/NpcSubScene.cpp b/game/preview/NpcSubScene.cpp
new file mode 100644
index 0000000..c9ab5b6
--- /dev/null
+++ b/game/preview/NpcSubScene.cpp
@@ -0,0 +1,65 @@
+
+
+#include "NpcSubScene.h"
+
+#include "../Config.h"
+#include "NpcScript.h"
+
+#include <crepe/ValueBroker.h>
+#include <crepe/api/Animator.h>
+#include <crepe/api/BehaviorScript.h>
+#include <crepe/api/Scene.h>
+#include <crepe/api/Sprite.h>
+#include <crepe/manager/SaveManager.h>
+
+using namespace crepe;
+
+NpcSubScene::NpcSubScene(Scene & scn) {
+ GameObject npc = scn.new_object("npc", "npc_tag", vec2 {500, 0}, 0, 1);
+ Asset npc_body {"asset/workers/worker1Body.png"};
+ Asset npc_head {"asset/workers/worker1Head.png"};
+
+ auto & npc_body_sprite = npc.add_component<Sprite>(
+ npc_body,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_PLAYER,
+ .size = {0, 50},
+ }
+ );
+ auto & npc_head_sprite = npc.add_component<Sprite>(
+ npc_head,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_PLAYER,
+ .size = {0, 50},
+ .position_offset = {0, -20},
+ }
+ );
+
+ npc.add_component<Animator>(
+ npc_body_sprite, ivec2 {32, 32}, uvec2 {4, 8},
+ Animator::Data {
+ .fps = 5,
+ .looping = true,
+ }
+ );
+ npc.add_component<Animator>(
+ npc_head_sprite, ivec2 {32, 32}, uvec2 {4, 8},
+ Animator::Data {
+ .fps = 5,
+ .looping = true,
+ }
+ );
+ npc.add_component<BoxCollider>(vec2 {50, 50});
+
+ npc.add_component<Rigidbody>(Rigidbody::Data {
+ .mass = 10,
+ .gravity_scale = 1,
+ .body_type = Rigidbody::BodyType::DYNAMIC,
+ .linear_velocity = {-50, 0},
+ //.max_linear_velocity = 40,
+ .collision_layers = {COLL_LAY_BOT_TOP, COLL_LAY_PLAYER, 100},
+ .collision_layer = COLL_LAY_PLAYER,
+ });
+
+ npc.add_component<BehaviorScript>().set_script<NpcScript>();
+}
diff --git a/game/preview/NpcSubScene.h b/game/preview/NpcSubScene.h
new file mode 100644
index 0000000..a226195
--- /dev/null
+++ b/game/preview/NpcSubScene.h
@@ -0,0 +1,10 @@
+#pragma once
+
+namespace crepe {
+class Scene;
+}
+
+class NpcSubScene {
+public:
+ NpcSubScene(crepe::Scene & scn);
+};
diff --git a/game/preview/PrevPlayerScript.cpp b/game/preview/PrevPlayerScript.cpp
new file mode 100644
index 0000000..fce5c5a
--- /dev/null
+++ b/game/preview/PrevPlayerScript.cpp
@@ -0,0 +1,126 @@
+#include "PrevPlayerScript.h"
+
+#include "../missile/SpawnEvent.h"
+
+#include <crepe/api/AudioSource.h>
+#include <crepe/api/Camera.h>
+#include <crepe/api/Transform.h>
+#include <crepe/manager/SaveManager.h>
+
+using namespace crepe;
+
+bool PrevPlayerScript::key_pressed(const KeyPressEvent & ev) {
+ switch (ev.key) {
+ case Keycode::A:
+ this->get_component<Rigidbody>().data.linear_velocity.x = -move_speed;
+ this->body->data.flip = {true, false};
+ this->head->data.flip = {true, false};
+ break;
+ case Keycode::D:
+ this->get_component<Rigidbody>().data.linear_velocity.x = move_speed;
+ this->body->data.flip = {false, false};
+ this->head->data.flip = {false, false};
+ break;
+
+ case Keycode::SPACE:
+ this->get_component<Rigidbody>().data.linear_velocity.y = -move_speed;
+ break;
+ case Keycode::D0:
+ this->body_anim->set_anim(0);
+ this->head_anim->set_anim(0);
+ break;
+ case Keycode::D1:
+ this->body_anim->set_anim(1);
+ this->head_anim->set_anim(1);
+ break;
+ case Keycode::D2:
+ this->body_anim->set_anim(2);
+ this->head_anim->set_anim(2);
+ break;
+ case Keycode::D3:
+ this->body_anim->set_anim(3);
+ this->head_anim->set_anim(3);
+ break;
+ case Keycode::D4:
+ this->body_anim->set_anim(4);
+ this->head_anim->set_anim(4);
+ break;
+ case Keycode::D5:
+ this->body_anim->set_anim(5);
+ this->head_anim->set_anim(5);
+ break;
+ case Keycode::D6:
+ this->body_anim->set_anim(6);
+ this->head_anim->set_anim(6);
+ break;
+ case Keycode::D7:
+ this->body_anim->set_anim(7);
+ this->head_anim->set_anim(7);
+ break;
+ case Keycode::LEFT:
+ this->get_component<Transform>().rotation += 10;
+ break;
+ case Keycode::RIGHT:
+ this->get_component<Transform>().rotation -= 10;
+ break;
+ case Keycode::UP:
+ this->head->data.position_offset += 10;
+ break;
+ case Keycode::DOWN:
+ this->head->data.position_offset -= 10;
+ break;
+ case Keycode::P:
+ this->get_component<AudioSource>().play();
+ break;
+ case Keycode::Q:
+ this->get_components_by_name<Camera>("camera").front().get().data.zoom -= 0.01;
+ break;
+ case Keycode::E:
+ this->get_components_by_name<Camera>("camera").front().get().data.zoom += 0.01;
+ break;
+ case Keycode::J:
+ this->get_components_by_name<Transform>("camera").front().get().position.x
+ -= move_speed;
+ break;
+ case Keycode::K:
+ this->get_components_by_name<Transform>("camera").front().get().position.y
+ -= move_speed;
+ break;
+ case Keycode::L:
+ this->get_components_by_name<Transform>("camera").front().get().position.x
+ += move_speed;
+ break;
+ case Keycode::I:
+ this->get_components_by_name<Transform>("camera").front().get().position.y
+ += move_speed;
+ break;
+ case Keycode::M:
+ trigger_event<MissileSpawnEvent>(MissileSpawnEvent {});
+ break;
+ default:
+ break;
+ }
+ return false;
+}
+
+void PrevPlayerScript::init() {
+ auto animations = this->get_components<Animator>();
+ body_anim = animations[0];
+ head_anim = animations[1];
+
+ auto sprites = this->get_components<Sprite>();
+ body = sprites[0];
+ head = sprites[1];
+
+ subscribe<KeyPressEvent>([this](const KeyPressEvent & ev) -> bool {
+ return this->key_pressed(ev);
+ });
+};
+
+void PrevPlayerScript::fixed_update(crepe::duration_t dt) {
+ auto & savemgr = this->get_save_manager();
+ const auto & pos = this->get_component<Transform>().position;
+
+ savemgr.set("player_x", pos.x);
+ savemgr.set("player_y", pos.y);
+};
diff --git a/game/preview/PrevPlayerScript.h b/game/preview/PrevPlayerScript.h
new file mode 100644
index 0000000..cc3184e
--- /dev/null
+++ b/game/preview/PrevPlayerScript.h
@@ -0,0 +1,23 @@
+
+#include <crepe/api/Event.h>
+#include <crepe/api/Script.h>
+#include <crepe/util/OptionalRef.h>
+
+#include <crepe/api/Animator.h>
+#include <crepe/api/Sprite.h>
+
+class PrevPlayerScript : public crepe::Script {
+private:
+ crepe::OptionalRef<crepe::Animator> head_anim;
+ crepe::OptionalRef<crepe::Animator> body_anim;
+ crepe::OptionalRef<crepe::Sprite> head;
+ crepe::OptionalRef<crepe::Sprite> body;
+
+private:
+ float move_speed = 100;
+
+private:
+ void init();
+ void fixed_update(crepe::duration_t dt);
+ bool key_pressed(const crepe::KeyPressEvent & ev);
+};
diff --git a/game/preview/PrevPlayerSubScene.cpp b/game/preview/PrevPlayerSubScene.cpp
new file mode 100644
index 0000000..9ff111c
--- /dev/null
+++ b/game/preview/PrevPlayerSubScene.cpp
@@ -0,0 +1,81 @@
+
+#include "PrevPlayerSubScene.h"
+
+#include "../Config.h"
+#include "PrevPlayerScript.h"
+
+#include <crepe/api/Animator.h>
+#include <crepe/api/AudioSource.h>
+#include <crepe/api/BehaviorScript.h>
+#include <crepe/api/Scene.h>
+#include <crepe/api/Sprite.h>
+
+#include <crepe/ValueBroker.h>
+#include <crepe/manager/SaveManager.h>
+
+using namespace crepe;
+
+PrevPlayerSubScene::PrevPlayerSubScene(Scene & scn) {
+
+ GameObject player = scn.new_object("player", "player", vec2 {800, -100}, 0, 1);
+ Asset player_body_asset {"asset/barry/defaultBody.png"};
+ Sprite & player_body_sprite = player.add_component<Sprite>(
+ player_body_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_PLAYER,
+ .order_in_layer = 0,
+ .size = vec2(0, 50),
+ }
+ );
+ player.add_component<Animator>(
+ player_body_sprite, ivec2(32, 32), uvec2(4, 8),
+ Animator::Data {
+ .fps = 5,
+ .looping = true,
+ }
+ );
+ Asset player_head_asset {"asset/barry/defaultHead.png"};
+ Sprite & player_head_sprite = player.add_component<Sprite>(
+ player_head_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_PLAYER,
+ .order_in_layer = 1,
+ .size = vec2(0, 50),
+ .position_offset = vec2(0, -20),
+ }
+ );
+ player.add_component<Animator>(
+ player_head_sprite, ivec2(32, 32), uvec2(4, 8),
+ Animator::Data {
+ .fps = 5,
+ .looping = true,
+ }
+ );
+ Asset player_jetpack_asset {"asset/barry/jetpackDefault.png"};
+ Sprite & player_jetpack_sprite = player.add_component<Sprite>(
+ player_jetpack_asset,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_PLAYER,
+ .order_in_layer = 2,
+ .size = vec2(0, 60),
+ .position_offset = vec2(-20, 0),
+ }
+ );
+ player_jetpack_sprite.active = false;
+ player.add_component<Animator>(
+ player_jetpack_sprite, ivec2(32, 44), uvec2(4, 4),
+ Animator::Data {
+ .fps = 5,
+ .looping = true,
+ }
+ );
+ player.add_component<Rigidbody>(Rigidbody::Data {
+ .gravity_scale = 1,
+ .body_type = Rigidbody::BodyType::DYNAMIC,
+ .linear_velocity = vec2(100, 0),
+ .collision_layers = {COLL_LAY_BOT_TOP, 100},
+ .collision_layer = COLL_LAY_PLAYER,
+ });
+ player.add_component<BoxCollider>(vec2(50, 50));
+ player.add_component<BehaviorScript>().set_script<PrevPlayerScript>();
+}
diff --git a/game/preview/PrevPlayerSubScene.h b/game/preview/PrevPlayerSubScene.h
new file mode 100644
index 0000000..a61f341
--- /dev/null
+++ b/game/preview/PrevPlayerSubScene.h
@@ -0,0 +1,10 @@
+#pragma once
+
+namespace crepe {
+class Scene;
+}
+
+class PrevPlayerSubScene {
+public:
+ PrevPlayerSubScene(crepe::Scene & scn);
+};
diff --git a/game/preview/PreviewReplaySubScript.cpp b/game/preview/PreviewReplaySubScript.cpp
new file mode 100644
index 0000000..580a608
--- /dev/null
+++ b/game/preview/PreviewReplaySubScript.cpp
@@ -0,0 +1,55 @@
+#include "PreviewReplaySubScript.h"
+#include "Config.h"
+#include "menus/ButtonReplaySubScript.h"
+#include <crepe/api/AudioSource.h>
+#include <crepe/types.h>
+
+using namespace crepe;
+using namespace std;
+
+void PreviewReplaySubScript::init() {
+ IButtonScript::init();
+ this->subscribe<ButtonPressEvent>([this](const ButtonPressEvent & e) {
+ return this->on_button_press(e);
+ });
+ this->subscribe<StopPreviewRecording>([this](const StopPreviewRecording & e) {
+ return this->stop_recording();
+ });
+ this->subscribe<DeleteRecordingEvent>([this](const DeleteRecordingEvent & e) {
+ return this->delete_recording();
+ });
+ this->subscribe<StartPreviewRecording>([this](const StartPreviewRecording & e) {
+ return this->start_recording();
+ });
+}
+
+bool PreviewReplaySubScript::on_button_press(const ButtonPressEvent & e) {
+ if (DISABLE_REPLAY) return false;
+ replay.play(this->recording);
+ return false;
+}
+bool PreviewReplaySubScript::start_recording() {
+ if (DISABLE_REPLAY) return false;
+ if (record_saved) {
+ this->stop_recording();
+ this->delete_recording();
+ }
+ replay.record_start();
+ this->record_started = true;
+ return false;
+}
+
+bool PreviewReplaySubScript::stop_recording() {
+ if (DISABLE_REPLAY) return false;
+ if (this->record_started) this->recording = replay.record_end();
+ this->record_saved = true;
+ return false;
+}
+
+bool PreviewReplaySubScript::delete_recording() {
+ if (DISABLE_REPLAY) return false;
+ if (this->record_started) this->stop_recording();
+ if (this->record_saved) replay.release(this->recording);
+ this->record_saved = false;
+ return false;
+}
diff --git a/game/preview/PreviewReplaySubScript.h b/game/preview/PreviewReplaySubScript.h
new file mode 100644
index 0000000..59b78c3
--- /dev/null
+++ b/game/preview/PreviewReplaySubScript.h
@@ -0,0 +1,24 @@
+#pragma once
+
+#include "menus/IButtonScript.h"
+
+#include <crepe/api/Script.h>
+
+struct StartPreviewRecording : public crepe::Event {};
+struct StopPreviewRecording : public crepe::Event {};
+
+class PreviewReplaySubScript : public IButtonScript {
+public:
+ void init() override;
+ bool on_button_press(const crepe::ButtonPressEvent & e);
+
+private:
+ crepe::recording_t recording = 0;
+ bool start_recording();
+ bool stop_recording();
+ bool delete_recording();
+
+private:
+ bool record_saved = false;
+ bool record_started = false;
+};
diff --git a/game/preview/PreviewStartRecSubScript.cpp b/game/preview/PreviewStartRecSubScript.cpp
new file mode 100644
index 0000000..8a2f54c
--- /dev/null
+++ b/game/preview/PreviewStartRecSubScript.cpp
@@ -0,0 +1,20 @@
+#include "PreviewStartRecSubScript.h"
+#include "PreviewReplaySubScript.h"
+
+#include <crepe/api/AudioSource.h>
+#include <crepe/types.h>
+
+using namespace crepe;
+using namespace std;
+
+void PreviewStartRecSubScript::init() {
+ IButtonScript::init();
+ this->subscribe<ButtonPressEvent>([this](const ButtonPressEvent & e) {
+ return this->on_button_press(e);
+ });
+}
+
+bool PreviewStartRecSubScript::on_button_press(const ButtonPressEvent & e) {
+ this->trigger_event<StartPreviewRecording>();
+ return false;
+}
diff --git a/game/preview/PreviewStartRecSubScript.h b/game/preview/PreviewStartRecSubScript.h
new file mode 100644
index 0000000..a54a085
--- /dev/null
+++ b/game/preview/PreviewStartRecSubScript.h
@@ -0,0 +1,11 @@
+#pragma once
+
+#include "menus/IButtonScript.h"
+
+#include <crepe/api/Script.h>
+
+class PreviewStartRecSubScript : public IButtonScript {
+public:
+ void init() override;
+ bool on_button_press(const crepe::ButtonPressEvent & e);
+};
diff --git a/game/preview/PreviewStopRecSubScript.cpp b/game/preview/PreviewStopRecSubScript.cpp
new file mode 100644
index 0000000..a229da8
--- /dev/null
+++ b/game/preview/PreviewStopRecSubScript.cpp
@@ -0,0 +1,20 @@
+#include "PreviewStopRecSubScript.h"
+#include "PreviewReplaySubScript.h"
+
+#include <crepe/api/AudioSource.h>
+#include <crepe/types.h>
+
+using namespace crepe;
+using namespace std;
+
+void PreviewStopRecSubScript::init() {
+ IButtonScript::init();
+ this->subscribe<ButtonPressEvent>([this](const ButtonPressEvent & e) {
+ return this->on_button_press(e);
+ });
+}
+
+bool PreviewStopRecSubScript::on_button_press(const ButtonPressEvent & e) {
+ this->trigger_event<StopPreviewRecording>();
+ return false;
+}
diff --git a/game/preview/PreviewStopRecSubScript.h b/game/preview/PreviewStopRecSubScript.h
new file mode 100644
index 0000000..b2dd73b
--- /dev/null
+++ b/game/preview/PreviewStopRecSubScript.h
@@ -0,0 +1,11 @@
+#pragma once
+
+#include "menus/IButtonScript.h"
+
+#include <crepe/api/Script.h>
+
+class PreviewStopRecSubScript : public IButtonScript {
+public:
+ void init() override;
+ bool on_button_press(const crepe::ButtonPressEvent & e);
+};
diff --git a/game/preview/SmokeSubScene.cpp b/game/preview/SmokeSubScene.cpp
new file mode 100644
index 0000000..e363f95
--- /dev/null
+++ b/game/preview/SmokeSubScene.cpp
@@ -0,0 +1,37 @@
+
+#include "SmokeSubScene.h"
+
+#include "../Config.h"
+
+#include <crepe/api/ParticleEmitter.h>
+#include <crepe/api/Scene.h>
+#include <crepe/api/Sprite.h>
+
+using namespace crepe;
+
+SmokeSubScene::SmokeSubScene(Scene & scn) {
+ GameObject smoke = scn.new_object("smoke_particle", "TAG", vec2 {500, -210}, 0, 1);
+
+ Asset smoke_ss {"asset/particles/smoke.png"};
+
+ auto & smoke_sprite = smoke.add_component<Sprite>(
+ smoke_ss,
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_PARTICLES_FOREGROUND,
+ .size = {0, 30},
+ }
+ );
+
+ smoke.add_component<ParticleEmitter>(
+ smoke_sprite,
+ ParticleEmitter::Data {
+ .offset = {0, -60},
+ .max_particles = 10,
+ .emission_rate = 25,
+ .min_angle = 60,
+ .max_angle = 120,
+ .begin_lifespan = 1,
+ .end_lifespan = 2,
+ }
+ );
+}
diff --git a/game/preview/SmokeSubScene.h b/game/preview/SmokeSubScene.h
new file mode 100644
index 0000000..93d8a2d
--- /dev/null
+++ b/game/preview/SmokeSubScene.h
@@ -0,0 +1,10 @@
+#pragma once
+
+namespace crepe {
+class Scene;
+}
+
+class SmokeSubScene {
+public:
+ SmokeSubScene(crepe::Scene & scn);
+};
diff --git a/game/scheduler/ObjectsScheduler.cpp b/game/scheduler/ObjectsScheduler.cpp
new file mode 100644
index 0000000..7f58c79
--- /dev/null
+++ b/game/scheduler/ObjectsScheduler.cpp
@@ -0,0 +1,120 @@
+
+
+#include "ObjectsScheduler.h"
+
+#include "../Config.h"
+#include "../Random.h"
+#include "../enemy/EnemyScript.h"
+#include "../missile/SpawnEvent.h"
+#include "api/Rigidbody.h"
+#include "api/Transform.h"
+#include "enemy/BattleScript.h"
+#include "prefab/ZapperPoolSubScene.h"
+
+using namespace crepe;
+
+void ObjectsScheduler::preset_0() {
+ for (int i = 0; i < this->amount_of_boss_fights; i++) {
+ this->trigger_event<MissileSpawnEvent>(MissileSpawnEvent {});
+ }
+ if (this->amount_of_boss_fights >= 1) {
+ this->trigger_event<BattleStartEvent>(BattleStartEvent {
+ .num_enemies = Random::i(this->amount_of_boss_fights, 0),
+ .battle = false,
+ });
+ }
+}
+
+void ObjectsScheduler::preset_1() {
+ trigger_event<MissileSpawnEvent>(MissileSpawnEvent {});
+ if (this->amount_of_boss_fights >= 3) {
+ this->trigger_event<BattleStartEvent>(BattleStartEvent {
+ .num_enemies = Random::i(1, 0),
+ .battle = false,
+ });
+ }
+}
+
+void ObjectsScheduler::preset_2() {
+ trigger_event<CreateZapperEvent>(CreateZapperEvent {});
+ if (this->amount_of_boss_fights >= 2) {
+ this->trigger_event<BattleStartEvent>(BattleStartEvent {
+ .num_enemies = Random::i(2, 1),
+ .battle = false,
+ });
+ }
+}
+
+void ObjectsScheduler::preset_3() { trigger_event<CreateZapperEvent>(CreateZapperEvent {}); }
+
+void ObjectsScheduler::preset_4() {}
+
+void ObjectsScheduler::boss_fight_1() {
+ this->get_components_by_name<Rigidbody>("camera").front().get().data.linear_velocity.x = 0;
+ this->get_components_by_name<Rigidbody>("player").front().get().data.linear_velocity.x = 0;
+
+ this->amount_of_boss_fights++;
+ this->trigger_event<BattleStartEvent>(BattleStartEvent {
+ .num_enemies = amount_of_boss_fights,
+ .battle = true,
+ });
+
+ RefVector<Rigidbody> rb_back_forest
+ = this->get_components_by_tag<Rigidbody>("forest_background");
+ for (Rigidbody & rb : rb_back_forest) {
+ rb.data.linear_velocity.x = 0;
+ }
+}
+
+bool ObjectsScheduler::boss_fight_1_event() {
+ this->get_components_by_name<Rigidbody>("camera").front().get().data.linear_velocity.x
+ = PLAYER_SPEED * 0.02;
+ this->get_components_by_name<Rigidbody>("player").front().get().data.linear_velocity.x
+ = PLAYER_SPEED * 0.02;
+
+ bool first = true;
+ RefVector<Rigidbody> rb_back_forest
+ = this->get_components_by_tag<Rigidbody>("forest_background");
+ for (Rigidbody & rb : rb_back_forest) {
+ if (first == true) {
+ rb.data.linear_velocity.x = 30;
+ first = false;
+ } else {
+ rb.data.linear_velocity.x = 40;
+ first = true;
+ }
+ }
+
+ return false;
+}
+
+void ObjectsScheduler::init() {
+ this->obstacles.push_back([this]() { preset_0(); });
+ this->obstacles.push_back([this]() { preset_1(); });
+ this->obstacles.push_back([this]() { preset_2(); });
+ this->obstacles.push_back([this]() { preset_3(); });
+ this->obstacles.push_back([this]() { preset_4(); });
+
+ this->obstacles.push_back([this]() { boss_fight_1(); });
+
+ // subscribe to battlewonevent
+ this->subscribe<BattleWonEvent>([this](const BattleWonEvent & ev) -> bool {
+ return this->boss_fight_1_event();
+ });
+}
+
+void ObjectsScheduler::fixed_update(duration_t dt) {
+ int pos_x
+ = (int) this->get_components_by_name<Transform>("camera").front().get().position.x;
+
+ int boss_check = (pos_x - this->start_offset) / this->boss_fight_interval;
+ if (boss_check > this->last_boss_check) {
+ this->obstacles.back()();
+ this->last_boss_check = boss_check;
+ }
+ int obstacle_check = (pos_x - this->start_offset) / this->obstacle_interval;
+ if (obstacle_check > this->last_obstacle_check) {
+ this->obstacles[Random::i(this->obstacles.size() - 1, 0)]();
+ this->last_obstacle_check = obstacle_check;
+ }
+}
diff --git a/game/scheduler/ObjectsScheduler.h b/game/scheduler/ObjectsScheduler.h
new file mode 100644
index 0000000..7ada8e1
--- /dev/null
+++ b/game/scheduler/ObjectsScheduler.h
@@ -0,0 +1,34 @@
+#pragma once
+
+#include "api/Script.h"
+#include <functional>
+#include <vector>
+
+class ObjectsScheduler : public crepe::Script {
+
+private:
+ std::vector<std::function<void()>> obstacles;
+
+ int last_boss_check = 0;
+ int last_obstacle_check = 0;
+
+ int boss_fight_interval = 5000;
+ int obstacle_interval = 350;
+ int start_offset = 1300;
+
+ int amount_of_boss_fights = 0;
+
+private:
+ void preset_0();
+ void preset_1();
+ void preset_2();
+ void preset_3();
+ void preset_4();
+ void boss_fight_1();
+
+ bool boss_fight_1_event();
+
+public:
+ void init();
+ void fixed_update(crepe::duration_t dt);
+};
diff --git a/game/util/scrollgen b/game/util/scrollgen
new file mode 100755
index 0000000..0389107
--- /dev/null
+++ b/game/util/scrollgen
@@ -0,0 +1,36 @@
+#!/bin/sh
+INPUT="$1"
+FRAMES="$2"
+
+die() {
+ echo "$@"
+ exit 1
+}
+check_command() {
+ cmd="$1"
+ command -v "$cmd" > /dev/null || die "command '$cmd' not found"
+}
+
+check_command magick
+check_command identify
+[ "$#" -eq 2 ] || die "usage: $0 <input image> <frame count>"
+[ -e "$INPUT" ] || die "file not found: $INPUT"
+[ "$FRAMES" -gt 0 ] || die "invalid frame count: $FRAMES"
+
+tile_width=$(identify -format "%w" "$INPUT")
+tile_height=$(identify -format "%h" "$INPUT")
+
+OUTPUT="$INPUT.scroll.png"
+magick -size "${tile_width}x$(( $tile_height * $FRAMES ))" 'xc:#ff00ff00' "$OUTPUT"
+
+for i in $(seq 0 $(( $FRAMES - 1 ))); do
+ offset_x=$(( $tile_width * $i / $FRAMES ))
+ offset_y=$(( i * $tile_height ))
+
+ magick "$OUTPUT" "$INPUT" -geometry "+${offset_x}+${offset_y}" -composite "$OUTPUT"
+ magick "$OUTPUT" "$INPUT" -geometry "+$(( $offset_x - $tile_width ))+${offset_y}" -composite "$OUTPUT"
+ echo "+${offset_x}+${offset_y}"
+done
+
+# magick -size ${total_width}x${sprite_height} xc:none canvas.png
+
diff --git a/game/workers/CollisionScript.cpp b/game/workers/CollisionScript.cpp
new file mode 100644
index 0000000..372bfec
--- /dev/null
+++ b/game/workers/CollisionScript.cpp
@@ -0,0 +1,69 @@
+#include "CollisionScript.h"
+
+#include <crepe/api/Animator.h>
+#include <crepe/api/BehaviorScript.h>
+#include <crepe/api/Rigidbody.h>
+#include <crepe/api/Sprite.h>
+#include <crepe/api/Transform.h>
+#include <crepe/types.h>
+
+using namespace crepe;
+using namespace std;
+
+void CollisionScript::init() {
+ subscribe<CollisionEvent>([this](const CollisionEvent & ev) -> bool {
+ return this->on_collision(ev);
+ });
+}
+
+bool CollisionScript::on_collision(const CollisionEvent & ev) {
+ RefVector<Animator> animators = this->get_components<Animator>();
+ RefVector<Sprite> sprites = this->get_components<Sprite>();
+ Rigidbody & rb = this->get_component<Rigidbody>();
+ Transform & tr = this->get_component<Transform>();
+ BehaviorScript & bs_panic = this->get_components<BehaviorScript>().front();
+
+ if (ev.info.other.metadata.tag == "zapper") {
+ for (Animator & anim : animators) {
+ anim.active = false;
+ anim.set_anim(3);
+ }
+ for (Sprite & sprite : sprites) {
+ sprite.data.position_offset.x = 15;
+ }
+ rb.data.linear_velocity_coefficient = {0.5, 0.5};
+ tr.rotation = 90;
+ bs_panic.active = false;
+
+ return false;
+ } else if (ev.info.other.metadata.tag == "laser") {
+ for (Animator & anim : animators) {
+ anim.active = false;
+ anim.set_anim(3);
+ }
+ for (Sprite & sprite : sprites) {
+ sprite.data.position_offset.x = 15;
+ }
+ rb.data.linear_velocity_coefficient = {0.5, 0.5};
+ tr.rotation = 90;
+ bs_panic.active = false;
+
+ return false;
+ } else if (ev.info.other.metadata.tag == "missile"
+ || ev.info.other.metadata.tag == "enemy_bullet") {
+ for (Animator & anim : animators) {
+ anim.active = false;
+ anim.set_anim(3);
+ }
+ for (Sprite & sprite : sprites) {
+ sprite.data.position_offset.x = 15;
+ }
+ rb.data.linear_velocity_coefficient = {0.5, 0.5};
+ tr.rotation = 90;
+ bs_panic.active = false;
+
+ return false;
+ }
+
+ return false;
+}
diff --git a/game/workers/CollisionScript.h b/game/workers/CollisionScript.h
new file mode 100644
index 0000000..70c5fe1
--- /dev/null
+++ b/game/workers/CollisionScript.h
@@ -0,0 +1,12 @@
+#pragma once
+
+#include <crepe/api/Event.h>
+#include <crepe/api/Script.h>
+
+class CollisionScript : public crepe::Script {
+public:
+ void init();
+
+private:
+ bool on_collision(const crepe::CollisionEvent & ev);
+};
diff --git a/game/workers/PanicFromPlayerScript.cpp b/game/workers/PanicFromPlayerScript.cpp
new file mode 100644
index 0000000..baa48df
--- /dev/null
+++ b/game/workers/PanicFromPlayerScript.cpp
@@ -0,0 +1,55 @@
+#include "PanicFromPlayerScript.h"
+
+#include <crepe/api/Animator.h>
+#include <crepe/api/Rigidbody.h>
+#include <crepe/api/Sprite.h>
+#include <crepe/api/Transform.h>
+#include <crepe/types.h>
+
+using namespace crepe;
+using namespace std;
+
+void PanicFromPlayerScript::fixed_update(duration_t dt) {
+ Animator & anim_player = this->get_components_by_name<Animator>("player").front();
+
+ if (anim_player.data.col == 1) {
+ Transform & trans_player = this->get_components_by_name<Transform>("player").back();
+ Transform & trans_worker = this->get_components<Transform>().back();
+
+ float result_x = trans_player.position.x - trans_worker.position.x;
+
+ if (result_x < 100 && result_x > -20) {
+ RefVector<Animator> anim_worker = this->get_components<Animator>();
+ RefVector<Sprite> sprite_worker = this->get_components<Sprite>();
+ Rigidbody & rb_worker = this->get_components<Rigidbody>().back();
+
+ if (anim_worker.front().get().data.col != 1) {
+ anim_worker.front().get().set_anim(1);
+ anim_worker.back().get().set_anim(1);
+
+ anim_worker.front().get().data.fps = 10;
+ anim_worker.back().get().data.fps = 10;
+ }
+
+ if (result_x < 0) {
+ float min_value = 8000;
+ float max_value = 10000;
+ float value = min_value
+ + static_cast<float>(rand())
+ / (static_cast<float>(RAND_MAX / (max_value - min_value)));
+ rb_worker.data.linear_velocity.x = value * dt.count();
+ sprite_worker.front().get().data.flip.flip_x = false;
+ sprite_worker.back().get().data.flip.flip_x = false;
+ } else {
+ float min_value = -4000;
+ float max_value = -5000;
+ float value = min_value
+ + static_cast<float>(rand())
+ / (static_cast<float>(RAND_MAX / (max_value - min_value)));
+ rb_worker.data.linear_velocity.x = value * dt.count();
+ sprite_worker.front().get().data.flip.flip_x = true;
+ sprite_worker.back().get().data.flip.flip_x = true;
+ }
+ }
+ }
+};
diff --git a/game/workers/PanicFromPlayerScript.h b/game/workers/PanicFromPlayerScript.h
new file mode 100644
index 0000000..d173e89
--- /dev/null
+++ b/game/workers/PanicFromPlayerScript.h
@@ -0,0 +1,8 @@
+#pragma once
+
+#include <crepe/api/Script.h>
+
+class PanicFromPlayerScript : public crepe::Script {
+public:
+ void fixed_update(crepe::duration_t dt);
+};
diff --git a/game/workers/WorkerScript.cpp b/game/workers/WorkerScript.cpp
new file mode 100644
index 0000000..b0bfc4e
--- /dev/null
+++ b/game/workers/WorkerScript.cpp
@@ -0,0 +1,145 @@
+#include "WorkerScript.h"
+
+#include "../Config.h"
+#include "api/BehaviorScript.h"
+
+#include <crepe/api/Animator.h>
+#include <crepe/api/Rigidbody.h>
+#include <crepe/api/Sprite.h>
+#include <crepe/api/Transform.h>
+#include <crepe/types.h>
+#include <cstdlib>
+
+using namespace crepe;
+using namespace std;
+
+void WorkerScript::fixed_update(duration_t dt) {
+ RefVector<Rigidbody> rb_workers = this->get_components_by_tag<Rigidbody>("worker");
+ RefVector<Transform> trans_workers = this->get_components_by_tag<Transform>("worker");
+
+ Rigidbody & rb_cam = this->get_components_by_name<Rigidbody>("camera").back();
+ Transform & trans_cam = this->get_components_by_name<Transform>("camera").back();
+
+ int counter = 0;
+ for (Rigidbody & rb_worker : rb_workers) {
+ Transform & trans_worker = trans_workers.at(counter);
+
+ float result_x = rb_cam.data.linear_velocity.x - rb_worker.data.linear_velocity.x;
+
+ if (result_x > 0) {
+ float left_cam_pos_x = trans_cam.position.x - VIEWPORT_X / 2;
+ if (trans_worker.position.x < left_cam_pos_x - 1000) {
+ trans_worker.position.x = left_cam_pos_x + VIEWPORT_X + 1000;
+
+ do {
+ float min_value = -2500 * dt.count();
+ float max_value = 2500 * dt.count();
+ rb_worker.data.linear_velocity.x
+ = min_value
+ + static_cast<float>(rand())
+ / (static_cast<float>(RAND_MAX / (max_value - min_value)));
+ } while (rb_worker.data.linear_velocity.x < 500 * dt.count()
+ && rb_worker.data.linear_velocity.x > -500 * dt.count());
+
+ RefVector<Sprite> sprite_worker
+ = this->get_components_by_id<Sprite>(trans_worker.game_object_id);
+ RefVector<Animator> animator_worker
+ = this->get_components_by_id<Animator>(trans_worker.game_object_id);
+ BehaviorScript & bs_panic
+ = this->get_components_by_id<BehaviorScript>(trans_worker.game_object_id)
+ .front();
+
+ if (rb_worker.data.linear_velocity.x < 0) {
+ sprite_worker.front().get().data.flip.flip_x = true;
+ sprite_worker.back().get().data.flip.flip_x = true;
+
+ animator_worker.front().get().data.fps
+ = -rb_worker.data.linear_velocity.x / 5;
+ animator_worker.back().get().data.fps
+ = -rb_worker.data.linear_velocity.x / 5;
+ animator_worker.front().get().set_anim(0);
+ animator_worker.back().get().set_anim(0);
+ animator_worker.front().get().active = true;
+ animator_worker.back().get().active = true;
+ } else {
+ sprite_worker.front().get().data.flip.flip_x = false;
+ sprite_worker.back().get().data.flip.flip_x = false;
+
+ animator_worker.front().get().data.fps
+ = rb_worker.data.linear_velocity.x / 5;
+ animator_worker.back().get().data.fps
+ = rb_worker.data.linear_velocity.x / 5;
+ animator_worker.front().get().set_anim(0);
+ animator_worker.back().get().set_anim(0);
+ animator_worker.front().get().active = true;
+ animator_worker.back().get().active = true;
+ }
+
+ trans_worker.rotation = 0;
+ bs_panic.active = true;
+ rb_worker.data.linear_velocity_coefficient = {1, 1};
+ for (Sprite & sprite : sprite_worker) {
+ sprite.data.position_offset.x = 0;
+ }
+ }
+ } else {
+ float right_cam_pos_x = trans_cam.position.x + VIEWPORT_X / 2;
+ if (trans_worker.position.x > right_cam_pos_x + 1000) {
+ do {
+ float min_value = -2500 * dt.count();
+ float max_value = 2500 * dt.count();
+ rb_worker.data.linear_velocity.x
+ = min_value
+ + static_cast<float>(rand())
+ / (static_cast<float>(RAND_MAX / (max_value - min_value)));
+ } while (rb_worker.data.linear_velocity.x < 500 * dt.count()
+ && rb_worker.data.linear_velocity.x > -500 * dt.count());
+
+ RefVector<Sprite> sprite_worker
+ = this->get_components_by_id<Sprite>(trans_worker.game_object_id);
+ RefVector<Animator> animator_worker
+ = this->get_components_by_id<Animator>(trans_worker.game_object_id);
+ BehaviorScript & bs_panic
+ = this->get_components_by_id<BehaviorScript>(trans_worker.game_object_id)
+ .front();
+
+ if (rb_worker.data.linear_velocity.x < 0) {
+ sprite_worker.front().get().data.flip.flip_x = true;
+ sprite_worker.back().get().data.flip.flip_x = true;
+
+ animator_worker.front().get().data.fps
+ = -rb_worker.data.linear_velocity.x / 5;
+ animator_worker.back().get().data.fps
+ = -rb_worker.data.linear_velocity.x / 5;
+
+ animator_worker.front().get().set_anim(0);
+ animator_worker.back().get().set_anim(0);
+ animator_worker.front().get().active = true;
+ animator_worker.back().get().active = true;
+ } else {
+ sprite_worker.front().get().data.flip.flip_x = false;
+ sprite_worker.back().get().data.flip.flip_x = false;
+
+ animator_worker.front().get().data.fps
+ = rb_worker.data.linear_velocity.x / 5;
+ animator_worker.back().get().data.fps
+ = rb_worker.data.linear_velocity.x / 5;
+
+ animator_worker.front().get().set_anim(0);
+ animator_worker.back().get().set_anim(0);
+ animator_worker.front().get().active = true;
+ animator_worker.back().get().active = true;
+ }
+
+ trans_worker.rotation = 0;
+ bs_panic.active = true;
+ rb_worker.data.linear_velocity_coefficient = {1, 1};
+ for (Sprite & sprite : sprite_worker) {
+ sprite.data.position_offset.x = 0;
+ }
+ }
+ }
+
+ counter++;
+ }
+}
diff --git a/game/workers/WorkerScript.h b/game/workers/WorkerScript.h
new file mode 100644
index 0000000..ae4a6c7
--- /dev/null
+++ b/game/workers/WorkerScript.h
@@ -0,0 +1,8 @@
+#pragma once
+
+#include <crepe/api/Script.h>
+
+class WorkerScript : public crepe::Script {
+public:
+ void fixed_update(crepe::duration_t dt);
+};
diff --git a/game/workers/WorkersSubScene.cpp b/game/workers/WorkersSubScene.cpp
new file mode 100644
index 0000000..54996d1
--- /dev/null
+++ b/game/workers/WorkersSubScene.cpp
@@ -0,0 +1,423 @@
+#include "WorkersSubScene.h"
+#include "CollisionScript.h"
+#include "PanicFromPlayerScript.h"
+#include "WorkerScript.h"
+
+#include "../Config.h"
+#include "api/GameObject.h"
+
+#include <crepe/api/Animator.h>
+#include <crepe/api/BehaviorScript.h>
+#include <crepe/api/BoxCollider.h>
+#include <crepe/api/Rigidbody.h>
+#include <crepe/api/Scene.h>
+#include <crepe/api/Sprite.h>
+
+using namespace crepe;
+using namespace std;
+
+WorkersSubScene::WorkersSubScene(Scene & scn) {
+ this->worker1(scn, 1200, -50);
+ this->worker2(scn, 1300, 20);
+ this->worker3(scn, 1400, -40);
+ this->worker4(scn, 7250, 50);
+ this->worker5(scn, 3400, -20);
+ this->worker6(scn, 2000, 30);
+ this->worker7(scn, 3725, 35);
+ this->worker8(scn, 2200, -15);
+
+ GameObject script = scn.new_object("workers_script");
+ script.add_component<BehaviorScript>().set_script<WorkerScript>();
+}
+
+void WorkersSubScene::worker1(crepe::Scene & scn, float start_x, float init_speed) {
+ GameObject worker_1 = scn.new_object("worker_1", "worker", vec2(start_x, 200));
+ Sprite & worker_1_body_sprite = worker_1.add_component<Sprite>(
+ Asset {"asset/workers/worker1Body.png"},
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_WORKERS_BACK,
+ .order_in_layer = 0,
+ .size = vec2(0, 50),
+ }
+ );
+ worker_1.add_component<Animator>(
+ worker_1_body_sprite, ivec2(32, 32), uvec2(4, 8),
+ Animator::Data {
+ .fps = static_cast<unsigned int>(abs(init_speed) / 5),
+ .looping = true,
+ }
+ );
+ Sprite & worker_1_head_sprite = worker_1.add_component<Sprite>(
+ Asset {"asset/workers/worker1Head.png"},
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_WORKERS_BACK,
+ .order_in_layer = 1,
+ .size = vec2(0, 50),
+ .position_offset = vec2(0, -20),
+ }
+ );
+ worker_1.add_component<Animator>(
+ worker_1_head_sprite, ivec2(32, 32), uvec2(4, 8),
+ Animator::Data {
+ .fps = static_cast<unsigned int>(abs(init_speed) / 5),
+ .looping = true,
+ }
+ );
+ worker_1.add_component<BoxCollider>(vec2(50, 50));
+ worker_1.add_component<Rigidbody>(Rigidbody::Data {
+ .gravity_scale = 20,
+ .body_type = Rigidbody::BodyType::DYNAMIC,
+ .linear_velocity = vec2(init_speed, 0),
+ .collision_layers = {COLL_LAY_BOT_TOP},
+ });
+ worker_1.add_component<BehaviorScript>().set_script<PanicFromPlayerScript>();
+ worker_1.add_component<BehaviorScript>().set_script<CollisionScript>();
+
+ if (init_speed < 0) {
+ worker_1_body_sprite.data.flip = Sprite::FlipSettings {true, false};
+ worker_1_head_sprite.data.flip = Sprite::FlipSettings {true, false};
+ }
+}
+
+void WorkersSubScene::worker2(crepe::Scene & scn, float start_x, float init_speed) {
+ GameObject worker_2 = scn.new_object("worker_2", "worker", vec2(start_x, 200));
+ Sprite & worker_2_body_sprite = worker_2.add_component<Sprite>(
+ Asset {"asset/workers/worker2Body.png"},
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_WORKERS_FRONT,
+ .order_in_layer = 2,
+ .size = vec2(0, 50),
+ }
+ );
+ worker_2.add_component<Animator>(
+ worker_2_body_sprite, ivec2(32, 32), uvec2(4, 8),
+ Animator::Data {
+ .fps = static_cast<unsigned int>(abs(init_speed) / 5),
+ .looping = true,
+ }
+ );
+ Sprite & worker_2_head_sprite = worker_2.add_component<Sprite>(
+ Asset {"asset/workers/worker1Head.png"},
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_WORKERS_FRONT,
+ .order_in_layer = 3,
+ .size = vec2(0, 50),
+ .position_offset = vec2(0, -20),
+ }
+ );
+ worker_2.add_component<Animator>(
+ worker_2_head_sprite, ivec2(32, 32), uvec2(4, 8),
+ Animator::Data {
+ .fps = static_cast<unsigned int>(abs(init_speed) / 5),
+ .looping = true,
+ }
+ );
+ worker_2.add_component<BoxCollider>(vec2(50, 50));
+ worker_2.add_component<Rigidbody>(Rigidbody::Data {
+ .gravity_scale = 20,
+ .body_type = Rigidbody::BodyType::DYNAMIC,
+ .linear_velocity = vec2(init_speed, 0),
+ .collision_layers = {COLL_LAY_BOT_TOP},
+ });
+ worker_2.add_component<BehaviorScript>().set_script<PanicFromPlayerScript>();
+ worker_2.add_component<BehaviorScript>().set_script<CollisionScript>();
+
+ if (init_speed < 0) {
+ worker_2_body_sprite.data.flip = Sprite::FlipSettings {true, false};
+ worker_2_head_sprite.data.flip = Sprite::FlipSettings {true, false};
+ }
+}
+
+void WorkersSubScene::worker3(crepe::Scene & scn, float start_x, float init_speed) {
+ GameObject worker_3 = scn.new_object("worker_3", "worker", vec2(start_x, 200));
+ Sprite & worker_3_body_sprite = worker_3.add_component<Sprite>(
+ Asset {"asset/workers/worker1Body.png"},
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_WORKERS_BACK,
+ .order_in_layer = 4,
+ .size = vec2(0, 50),
+ }
+ );
+ worker_3.add_component<Animator>(
+ worker_3_body_sprite, ivec2(32, 32), uvec2(4, 8),
+ Animator::Data {
+ .fps = static_cast<unsigned int>(abs(init_speed) / 5),
+ .looping = true,
+ }
+ );
+ Sprite & worker_3_head_sprite = worker_3.add_component<Sprite>(
+ Asset {"asset/workers/worker2Head.png"},
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_WORKERS_BACK,
+ .order_in_layer = 5,
+ .size = vec2(0, 50),
+ .position_offset = vec2(0, -20),
+ }
+ );
+ worker_3.add_component<Animator>(
+ worker_3_head_sprite, ivec2(32, 32), uvec2(4, 8),
+ Animator::Data {
+ .fps = static_cast<unsigned int>(abs(init_speed) / 5),
+ .looping = true,
+ }
+ );
+ worker_3.add_component<BoxCollider>(vec2(50, 50));
+ worker_3.add_component<Rigidbody>(Rigidbody::Data {
+ .gravity_scale = 20,
+ .body_type = Rigidbody::BodyType::DYNAMIC,
+ .linear_velocity = vec2(init_speed, 0),
+ .collision_layers = {COLL_LAY_BOT_TOP},
+ });
+ worker_3.add_component<BehaviorScript>().set_script<PanicFromPlayerScript>();
+ worker_3.add_component<BehaviorScript>().set_script<CollisionScript>();
+
+ if (init_speed < 0) {
+ worker_3_body_sprite.data.flip = Sprite::FlipSettings {true, false};
+ worker_3_head_sprite.data.flip = Sprite::FlipSettings {true, false};
+ }
+}
+
+void WorkersSubScene::worker4(crepe::Scene & scn, float start_x, float init_speed) {
+ GameObject worker_4 = scn.new_object("worker_4", "worker", vec2(start_x, 200));
+ Sprite & worker_4_body_sprite = worker_4.add_component<Sprite>(
+ Asset {"asset/workers/worker2Body.png"},
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_WORKERS_FRONT,
+ .order_in_layer = 6,
+ .size = vec2(0, 50),
+ }
+ );
+ worker_4.add_component<Animator>(
+ worker_4_body_sprite, ivec2(32, 32), uvec2(4, 8),
+ Animator::Data {
+ .fps = static_cast<unsigned int>(abs(init_speed) / 5),
+ .looping = true,
+ }
+ );
+ Sprite & worker_4_head_sprite = worker_4.add_component<Sprite>(
+ Asset {"asset/workers/worker2Head.png"},
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_WORKERS_FRONT,
+ .order_in_layer = 7,
+ .size = vec2(0, 50),
+ .position_offset = vec2(0, -20),
+ }
+ );
+ worker_4.add_component<Animator>(
+ worker_4_head_sprite, ivec2(32, 32), uvec2(4, 8),
+ Animator::Data {
+ .fps = static_cast<unsigned int>(abs(init_speed) / 5),
+ .looping = true,
+ }
+ );
+ worker_4.add_component<BoxCollider>(vec2(50, 50));
+ worker_4.add_component<Rigidbody>(Rigidbody::Data {
+ .gravity_scale = 20,
+ .body_type = Rigidbody::BodyType::DYNAMIC,
+ .linear_velocity = vec2(init_speed, 0),
+ .collision_layers = {COLL_LAY_BOT_HIGH},
+ });
+ worker_4.add_component<BehaviorScript>().set_script<PanicFromPlayerScript>();
+ worker_4.add_component<BehaviorScript>().set_script<CollisionScript>();
+
+ if (init_speed < 0) {
+ worker_4_body_sprite.data.flip = Sprite::FlipSettings {true, false};
+ worker_4_head_sprite.data.flip = Sprite::FlipSettings {true, false};
+ }
+}
+
+void WorkersSubScene::worker5(crepe::Scene & scn, float start_x, float init_speed) {
+ GameObject worker_5 = scn.new_object("worker_5", "worker", vec2(start_x, 200));
+ Sprite & worker_5_body_sprite = worker_5.add_component<Sprite>(
+ Asset {"asset/workers/workerFatBody.png"},
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_WORKERS_BACK,
+ .order_in_layer = 8,
+ .size = vec2(0, 50),
+ }
+ );
+ worker_5.add_component<Animator>(
+ worker_5_body_sprite, ivec2(32, 32), uvec2(4, 8),
+ Animator::Data {
+ .fps = static_cast<unsigned int>(abs(init_speed) / 5),
+ .looping = true,
+ }
+ );
+ Sprite & worker_5_head_sprite = worker_5.add_component<Sprite>(
+ Asset {"asset/workers/worker1Head.png"},
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_WORKERS_BACK,
+ .order_in_layer = 9,
+ .size = vec2(0, 50),
+ .position_offset = vec2(0, -20),
+ }
+ );
+ worker_5.add_component<Animator>(
+ worker_5_head_sprite, ivec2(32, 32), uvec2(4, 8),
+ Animator::Data {
+ .fps = static_cast<unsigned int>(abs(init_speed) / 5),
+ .looping = true,
+ }
+ );
+ worker_5.add_component<BoxCollider>(vec2(50, 50));
+ worker_5.add_component<Rigidbody>(Rigidbody::Data {
+ .gravity_scale = 20,
+ .body_type = Rigidbody::BodyType::DYNAMIC,
+ .linear_velocity = vec2(init_speed, 0),
+ .collision_layers = {COLL_LAY_BOT_HIGH},
+ });
+ worker_5.add_component<BehaviorScript>().set_script<PanicFromPlayerScript>();
+ worker_5.add_component<BehaviorScript>().set_script<CollisionScript>();
+
+ if (init_speed < 0) {
+ worker_5_body_sprite.data.flip = Sprite::FlipSettings {true, false};
+ worker_5_head_sprite.data.flip = Sprite::FlipSettings {true, false};
+ }
+}
+
+void WorkersSubScene::worker6(crepe::Scene & scn, float start_x, float init_speed) {
+ GameObject worker_6 = scn.new_object("worker_6", "worker", vec2(start_x, 200));
+ Sprite & worker_6_body_sprite = worker_6.add_component<Sprite>(
+ Asset {"asset/workers/workerFatBody.png"},
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_WORKERS_FRONT,
+ .order_in_layer = 10,
+ .size = vec2(0, 50),
+ }
+ );
+ worker_6.add_component<Animator>(
+ worker_6_body_sprite, ivec2(32, 32), uvec2(4, 8),
+ Animator::Data {
+ .fps = static_cast<unsigned int>(abs(init_speed) / 5),
+ .looping = true,
+ }
+ );
+ Sprite & worker_6_head_sprite = worker_6.add_component<Sprite>(
+ Asset {"asset/workers/worker2Head.png"},
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_WORKERS_FRONT,
+ .order_in_layer = 11,
+ .size = vec2(0, 50),
+ .position_offset = vec2(0, -20),
+ }
+ );
+ worker_6.add_component<Animator>(
+ worker_6_head_sprite, ivec2(32, 32), uvec2(4, 8),
+ Animator::Data {
+ .fps = static_cast<unsigned int>(abs(init_speed) / 5),
+ .looping = true,
+ }
+ );
+ worker_6.add_component<BoxCollider>(vec2(50, 50));
+ worker_6.add_component<Rigidbody>(Rigidbody::Data {
+ .gravity_scale = 20,
+ .body_type = Rigidbody::BodyType::DYNAMIC,
+ .linear_velocity = vec2(init_speed, 0),
+ .collision_layers = {COLL_LAY_BOT_LOW},
+ });
+ worker_6.add_component<BehaviorScript>().set_script<PanicFromPlayerScript>();
+ worker_6.add_component<BehaviorScript>().set_script<CollisionScript>();
+
+ if (init_speed < 0) {
+ worker_6_body_sprite.data.flip = Sprite::FlipSettings {true, false};
+ worker_6_head_sprite.data.flip = Sprite::FlipSettings {true, false};
+ }
+}
+
+void WorkersSubScene::worker7(crepe::Scene & scn, float start_x, float init_speed) {
+ GameObject worker_7 = scn.new_object("worker_7", "worker", vec2(start_x, 200));
+ Sprite & worker_7_body_sprite = worker_7.add_component<Sprite>(
+ Asset {"asset/workers/workerTallBody.png"},
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_WORKERS_BACK,
+ .order_in_layer = 12,
+ .size = vec2(0, 50),
+ }
+ );
+ worker_7.add_component<Animator>(
+ worker_7_body_sprite, ivec2(32, 32), uvec2(4, 8),
+ Animator::Data {
+ .fps = static_cast<unsigned int>(abs(init_speed) / 5),
+ .looping = true,
+ }
+ );
+ Sprite & worker_7_head_sprite = worker_7.add_component<Sprite>(
+ Asset {"asset/workers/worker1Head.png"},
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_WORKERS_BACK,
+ .order_in_layer = 13,
+ .size = vec2(0, 50),
+ .position_offset = vec2(0, -20),
+ }
+ );
+ worker_7.add_component<Animator>(
+ worker_7_head_sprite, ivec2(32, 32), uvec2(4, 8),
+ Animator::Data {
+ .fps = static_cast<unsigned int>(abs(init_speed) / 5),
+ .looping = true,
+ }
+ );
+ worker_7.add_component<BoxCollider>(vec2(50, 50));
+ worker_7.add_component<Rigidbody>(Rigidbody::Data {
+ .gravity_scale = 20,
+ .body_type = Rigidbody::BodyType::DYNAMIC,
+ .linear_velocity = vec2(init_speed, 0),
+ .collision_layers = {COLL_LAY_BOT_LOW},
+ });
+ worker_7.add_component<BehaviorScript>().set_script<PanicFromPlayerScript>();
+ worker_7.add_component<BehaviorScript>().set_script<CollisionScript>();
+
+ if (init_speed < 0) {
+ worker_7_body_sprite.data.flip = Sprite::FlipSettings {true, false};
+ worker_7_head_sprite.data.flip = Sprite::FlipSettings {true, false};
+ }
+}
+
+void WorkersSubScene::worker8(crepe::Scene & scn, float start_x, float init_speed) {
+ GameObject worker_8 = scn.new_object("worker_8", "worker", vec2(start_x, 200));
+ Sprite & worker_8_body_sprite = worker_8.add_component<Sprite>(
+ Asset {"asset/workers/workerTallBody.png"},
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_WORKERS_FRONT,
+ .order_in_layer = 14,
+ .size = vec2(0, 50),
+ }
+ );
+ worker_8.add_component<Animator>(
+ worker_8_body_sprite, ivec2(32, 32), uvec2(4, 8),
+ Animator::Data {
+ .fps = static_cast<unsigned int>(abs(init_speed) / 5),
+ .looping = true,
+ }
+ );
+ Sprite & worker_8_head_sprite = worker_8.add_component<Sprite>(
+ Asset {"asset/workers/worker2Head.png"},
+ Sprite::Data {
+ .sorting_in_layer = SORT_IN_LAY_WORKERS_FRONT,
+ .order_in_layer = 15,
+ .size = vec2(0, 50),
+ .position_offset = vec2(0, -20),
+ }
+ );
+ worker_8.add_component<Animator>(
+ worker_8_head_sprite, ivec2(32, 32), uvec2(4, 8),
+ Animator::Data {
+ .fps = static_cast<unsigned int>(abs(init_speed) / 5),
+ .looping = true,
+ }
+ );
+ worker_8.add_component<BoxCollider>(vec2(50, 50));
+ worker_8.add_component<Rigidbody>(Rigidbody::Data {
+ .gravity_scale = 20,
+ .body_type = Rigidbody::BodyType::DYNAMIC,
+ .linear_velocity = vec2(init_speed, 0),
+ .collision_layers = {COLL_LAY_BOT_LOW},
+ });
+ worker_8.add_component<BehaviorScript>().set_script<PanicFromPlayerScript>();
+ worker_8.add_component<BehaviorScript>().set_script<CollisionScript>();
+
+ if (init_speed < 0) {
+ worker_8_body_sprite.data.flip = Sprite::FlipSettings {true, false};
+ worker_8_head_sprite.data.flip = Sprite::FlipSettings {true, false};
+ }
+}
diff --git a/game/workers/WorkersSubScene.h b/game/workers/WorkersSubScene.h
new file mode 100644
index 0000000..6692d87
--- /dev/null
+++ b/game/workers/WorkersSubScene.h
@@ -0,0 +1,20 @@
+#pragma once
+
+namespace crepe {
+class Scene;
+}
+
+class WorkersSubScene {
+public:
+ WorkersSubScene(crepe::Scene & scn);
+
+private:
+ void worker1(crepe::Scene & scn, float start_x, float init_speed);
+ void worker2(crepe::Scene & scn, float start_x, float init_speed);
+ void worker3(crepe::Scene & scn, float start_x, float init_speed);
+ void worker4(crepe::Scene & scn, float start_x, float init_speed);
+ void worker5(crepe::Scene & scn, float start_x, float init_speed);
+ void worker6(crepe::Scene & scn, float start_x, float init_speed);
+ void worker7(crepe::Scene & scn, float start_x, float init_speed);
+ void worker8(crepe::Scene & scn, float start_x, float init_speed);
+};