diff --git a/source/core/assets/configs/tower.json b/source/core/assets/configs/tower.json index 4d1daa23c..907105301 100644 --- a/source/core/assets/configs/tower.json +++ b/source/core/assets/configs/tower.json @@ -2,51 +2,51 @@ "income": { "health": 10, "baseAttack": 0, - "cost": 10 + "cost": 100 }, "wall": { "health": 300, "baseAttack": 0, - "cost": 45 + "cost": 200 }, "TNTTower": { "health": 10, "baseAttack": 5, - "cost": 30 + "cost": 50 }, "weapon": { "health": 75, "baseAttack": 15, - "cost": 15 + "cost": 50 }, "DroidTower": { "health": 100, "baseAttack": 25, - "cost": 45 + "cost": 300 }, "fireTower": { "health": 100, "baseAttack": 25, - "cost": 45 + "cost": 300 }, "stunTower": { "health": 100, "baseAttack": 25, - "cost": 45 + "cost": 500 }, "fireworksTower": { "health": 125, "baseAttack": 45, - "cost": 70 + "cost": 500 }, "pierceTower": { "health": 125, "baseAttack": 45, - "cost": 70 + "cost": 500 }, "ricochetTower": { "health": 125, "baseAttack": 45, - "cost": 70 + "cost": 500 } } \ No newline at end of file diff --git a/source/core/assets/images/HelpScreen/Demonbosshelp.png b/source/core/assets/images/HelpScreen/Demonbosshelp.png new file mode 100644 index 000000000..502a41977 Binary files /dev/null and b/source/core/assets/images/HelpScreen/Demonbosshelp.png differ diff --git a/source/core/assets/images/HelpScreen/Dragonknighthelp.png b/source/core/assets/images/HelpScreen/Dragonknighthelp.png new file mode 100644 index 000000000..92594af03 Binary files /dev/null and b/source/core/assets/images/HelpScreen/Dragonknighthelp.png differ diff --git a/source/core/assets/images/HelpScreen/Econ.png b/source/core/assets/images/HelpScreen/Econ.png new file mode 100644 index 000000000..299bc7c22 Binary files /dev/null and b/source/core/assets/images/HelpScreen/Econ.png differ diff --git a/source/core/assets/images/HelpScreen/Engineer.png b/source/core/assets/images/HelpScreen/Engineer.png new file mode 100644 index 000000000..7bcb61922 Binary files /dev/null and b/source/core/assets/images/HelpScreen/Engineer.png differ diff --git a/source/core/assets/images/HelpScreen/Firewormhelp.png b/source/core/assets/images/HelpScreen/Firewormhelp.png new file mode 100644 index 000000000..1b1fb1be9 Binary files /dev/null and b/source/core/assets/images/HelpScreen/Firewormhelp.png differ diff --git a/source/core/assets/images/HelpScreen/HTP.png b/source/core/assets/images/HelpScreen/HTP.png new file mode 100644 index 000000000..fafc6b262 Binary files /dev/null and b/source/core/assets/images/HelpScreen/HTP.png differ diff --git a/source/core/assets/images/HelpScreen/IceBosshelp.png b/source/core/assets/images/HelpScreen/IceBosshelp.png new file mode 100644 index 000000000..af09d4c92 Binary files /dev/null and b/source/core/assets/images/HelpScreen/IceBosshelp.png differ diff --git a/source/core/assets/images/HelpScreen/Mob.png b/source/core/assets/images/HelpScreen/Mob.png new file mode 100644 index 000000000..a0dfa6fda Binary files /dev/null and b/source/core/assets/images/HelpScreen/Mob.png differ diff --git a/source/core/assets/images/HelpScreen/Patrickbosshelp.png b/source/core/assets/images/HelpScreen/Patrickbosshelp.png new file mode 100644 index 000000000..d7d342e78 Binary files /dev/null and b/source/core/assets/images/HelpScreen/Patrickbosshelp.png differ diff --git a/source/core/assets/images/HelpScreen/Skeletonhelp.png b/source/core/assets/images/HelpScreen/Skeletonhelp.png new file mode 100644 index 000000000..258c6f730 Binary files /dev/null and b/source/core/assets/images/HelpScreen/Skeletonhelp.png differ diff --git a/source/core/assets/images/HelpScreen/Turret.png b/source/core/assets/images/HelpScreen/Turret.png new file mode 100644 index 000000000..f8be39f9b Binary files /dev/null and b/source/core/assets/images/HelpScreen/Turret.png differ diff --git a/source/core/assets/images/HelpScreen/WaterQueenhelp.png b/source/core/assets/images/HelpScreen/WaterQueenhelp.png new file mode 100644 index 000000000..1481b12c0 Binary files /dev/null and b/source/core/assets/images/HelpScreen/WaterQueenhelp.png differ diff --git a/source/core/assets/images/HelpScreen/Waterslimehelp.png b/source/core/assets/images/HelpScreen/Waterslimehelp.png new file mode 100644 index 000000000..53884cbac Binary files /dev/null and b/source/core/assets/images/HelpScreen/Waterslimehelp.png differ diff --git a/source/core/assets/images/HelpScreen/Wizardhelp.png b/source/core/assets/images/HelpScreen/Wizardhelp.png new file mode 100644 index 000000000..ab5be4a71 Binary files /dev/null and b/source/core/assets/images/HelpScreen/Wizardhelp.png differ diff --git a/source/core/assets/images/HelpScreen/hs.jpg b/source/core/assets/images/HelpScreen/hs.jpg new file mode 100644 index 000000000..44eb11456 Binary files /dev/null and b/source/core/assets/images/HelpScreen/hs.jpg differ diff --git a/source/core/assets/images/LoadingScreen.png b/source/core/assets/images/LoadingScreen.png new file mode 100644 index 000000000..bfae39510 Binary files /dev/null and b/source/core/assets/images/LoadingScreen.png differ diff --git a/source/core/assets/images/background/main_menu/main_menu_bg.png b/source/core/assets/images/background/main_menu/main_menu_bg.png index 8fc9e4556..54e011d2a 100644 Binary files a/source/core/assets/images/background/main_menu/main_menu_bg.png and b/source/core/assets/images/background/main_menu/main_menu_bg.png differ diff --git a/source/core/assets/images/background/settings/settings_bg.png b/source/core/assets/images/background/settings/settings_bg.png index f0a0c5064..febd8eb6f 100644 Binary files a/source/core/assets/images/background/settings/settings_bg.png and b/source/core/assets/images/background/settings/settings_bg.png differ diff --git a/source/core/assets/images/mobs/skeleton.atlas b/source/core/assets/images/mobs/skeleton.atlas index f0117d879..9e1c78e17 100644 --- a/source/core/assets/images/mobs/skeleton.atlas +++ b/source/core/assets/images/mobs/skeleton.atlas @@ -69,7 +69,7 @@ skeleton_attack index: -1 skeleton_attack rotate: false - xy: 514, 2 + xy: 578, 52 size: 62, 48 orig: 62, 48 offset: 0, 0 @@ -83,7 +83,7 @@ skeleton_attack index: -1 skeleton_attack rotate: false - xy: 770, 2 + xy: 834, 52 size: 62, 48 orig: 62, 48 offset: 0, 0 @@ -95,41 +95,6 @@ skeleton_attack orig: 62, 48 offset: 0, 0 index: -1 -skeleton_death - rotate: false - xy: 1026, 2 - size: 62, 48 - orig: 62, 48 - offset: 0, 0 - index: -1 -skeleton_death - rotate: false - xy: 66, 52 - size: 62, 48 - orig: 62, 48 - offset: 0, 0 - index: -1 -skeleton_death - rotate: false - xy: 834, 52 - size: 62, 48 - orig: 62, 48 - offset: 0, 0 - index: -1 -skeleton_death - rotate: false - xy: 1154, 55 - size: 62, 45 - orig: 62, 45 - offset: 0, 0 - index: -1 -skeleton_death - rotate: false - xy: 578, 2 - size: 62, 48 - orig: 62, 48 - offset: 0, 0 - index: -1 skeleton_death rotate: false xy: 194, 52 @@ -188,7 +153,7 @@ skeleton_walk index: -1 skeleton_walk rotate: false - xy: 514, 52 + xy: 514, 2 size: 62, 48 orig: 62, 48 offset: 0, 0 @@ -202,7 +167,7 @@ skeleton_walk index: -1 skeleton_walk rotate: false - xy: 770, 52 + xy: 770, 2 size: 62, 48 orig: 62, 48 offset: 0, 0 @@ -237,21 +202,21 @@ skeleton_walk index: -1 skeleton_walk rotate: false - xy: 578, 52 + xy: 514, 52 size: 62, 48 orig: 62, 48 offset: 0, 0 index: -1 skeleton_walk rotate: false - xy: 130, 2 + xy: 66, 52 size: 62, 48 orig: 62, 48 offset: 0, 0 index: -1 skeleton_walk rotate: false - xy: 834, 2 + xy: 770, 52 size: 62, 48 orig: 62, 48 offset: 0, 0 diff --git a/source/core/assets/images/mobs/skeleton.png b/source/core/assets/images/mobs/skeleton.png index cbcbd1ed1..9d75bb066 100644 Binary files a/source/core/assets/images/mobs/skeleton.png and b/source/core/assets/images/mobs/skeleton.png differ diff --git a/source/core/assets/images/ui/Screen/Losing.png b/source/core/assets/images/ui/Screen/Losing.png new file mode 100644 index 000000000..133a500f8 Binary files /dev/null and b/source/core/assets/images/ui/Screen/Losing.png differ diff --git a/source/core/assets/images/ui/Screen/Nextlevel.png b/source/core/assets/images/ui/Screen/Nextlevel.png new file mode 100644 index 000000000..424016fa4 Binary files /dev/null and b/source/core/assets/images/ui/Screen/Nextlevel.png differ diff --git a/source/core/assets/images/ui/Screen/WinningScreen.png b/source/core/assets/images/ui/Screen/WinningScreen.png new file mode 100644 index 000000000..bd977c7a8 Binary files /dev/null and b/source/core/assets/images/ui/Screen/WinningScreen.png differ diff --git a/source/core/assets/images/ui/Sprites/UI_Glass_Arrow_Large_01a - Copy.png b/source/core/assets/images/ui/Sprites/UI_Glass_Arrow_Large_01a - Copy.png new file mode 100644 index 000000000..6f8775172 Binary files /dev/null and b/source/core/assets/images/ui/Sprites/UI_Glass_Arrow_Large_01a - Copy.png differ diff --git a/source/core/assets/images/ui/game screen/1 earth before.png b/source/core/assets/images/ui/game screen/1 earth before.png index c4233d125..1312aeb97 100644 Binary files a/source/core/assets/images/ui/game screen/1 earth before.png and b/source/core/assets/images/ui/game screen/1 earth before.png differ diff --git a/source/core/assets/images/ui/game screen/1.1 earth before.png b/source/core/assets/images/ui/game screen/1.1 earth before.png index 5bed55ad9..0e58a9b27 100644 Binary files a/source/core/assets/images/ui/game screen/1.1 earth before.png and b/source/core/assets/images/ui/game screen/1.1 earth before.png differ diff --git a/source/core/assets/images/ui/game screen/2.0 earth dying.png b/source/core/assets/images/ui/game screen/2.0 earth dying.png index 162af4008..86127efd1 100644 Binary files a/source/core/assets/images/ui/game screen/2.0 earth dying.png and b/source/core/assets/images/ui/game screen/2.0 earth dying.png differ diff --git a/source/core/assets/images/ui/game screen/2.1 earth dying.png b/source/core/assets/images/ui/game screen/2.1 earth dying.png index c2e54ef5c..e6f0f8328 100644 Binary files a/source/core/assets/images/ui/game screen/2.1 earth dying.png and b/source/core/assets/images/ui/game screen/2.1 earth dying.png differ diff --git a/source/core/assets/images/ui/game screen/3. meeting.png b/source/core/assets/images/ui/game screen/3. meeting.png index e91965704..a7b89cdbc 100644 Binary files a/source/core/assets/images/ui/game screen/3. meeting.png and b/source/core/assets/images/ui/game screen/3. meeting.png differ diff --git a/source/core/assets/images/ui/game screen/3.1 meeting turret.png b/source/core/assets/images/ui/game screen/3.1 meeting turret.png index 6669cedb4..94793be5d 100644 Binary files a/source/core/assets/images/ui/game screen/3.1 meeting turret.png and b/source/core/assets/images/ui/game screen/3.1 meeting turret.png differ diff --git a/source/core/assets/images/ui/game screen/4.0 spaceship built.png b/source/core/assets/images/ui/game screen/4.0 spaceship built.png index 0baad28ac..ec496ffdb 100644 Binary files a/source/core/assets/images/ui/game screen/4.0 spaceship built.png and b/source/core/assets/images/ui/game screen/4.0 spaceship built.png differ diff --git a/source/core/assets/images/ui/game screen/4.1 spaceship leaving.png b/source/core/assets/images/ui/game screen/4.1 spaceship leaving.png index 10761bfb8..190ad6288 100644 Binary files a/source/core/assets/images/ui/game screen/4.1 spaceship leaving.png and b/source/core/assets/images/ui/game screen/4.1 spaceship leaving.png differ diff --git a/source/core/assets/images/ui/game screen/5 arrival.png b/source/core/assets/images/ui/game screen/5 arrival.png index 31369d500..15cd7697c 100644 Binary files a/source/core/assets/images/ui/game screen/5 arrival.png and b/source/core/assets/images/ui/game screen/5 arrival.png differ diff --git a/source/core/assets/images/ui/game screen/5.1 arrival.png b/source/core/assets/images/ui/game screen/5.1 arrival.png index f718a06f8..ec393bc53 100644 Binary files a/source/core/assets/images/ui/game screen/5.1 arrival.png and b/source/core/assets/images/ui/game screen/5.1 arrival.png differ diff --git a/source/core/assets/images/ui/game screen/6.0 survey.png b/source/core/assets/images/ui/game screen/6.0 survey.png index ecce5abc5..bf328c7f6 100644 Binary files a/source/core/assets/images/ui/game screen/6.0 survey.png and b/source/core/assets/images/ui/game screen/6.0 survey.png differ diff --git a/source/core/assets/images/ui/game screen/6.1 survey.png b/source/core/assets/images/ui/game screen/6.1 survey.png index 22606d09e..56924bc69 100644 Binary files a/source/core/assets/images/ui/game screen/6.1 survey.png and b/source/core/assets/images/ui/game screen/6.1 survey.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/1.png b/source/core/assets/planets/Barren_or_Moon_bw/2/1.png new file mode 100644 index 000000000..83b146773 Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/1.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/10.png b/source/core/assets/planets/Barren_or_Moon_bw/2/10.png new file mode 100644 index 000000000..38c48d3f9 Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/10.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/11.png b/source/core/assets/planets/Barren_or_Moon_bw/2/11.png new file mode 100644 index 000000000..95cf8ab42 Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/11.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/12.png b/source/core/assets/planets/Barren_or_Moon_bw/2/12.png new file mode 100644 index 000000000..af7abbc1c Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/12.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/13.png b/source/core/assets/planets/Barren_or_Moon_bw/2/13.png new file mode 100644 index 000000000..69d8b3c97 Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/13.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/14.png b/source/core/assets/planets/Barren_or_Moon_bw/2/14.png new file mode 100644 index 000000000..cde8aac41 Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/14.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/15.png b/source/core/assets/planets/Barren_or_Moon_bw/2/15.png new file mode 100644 index 000000000..72688aec7 Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/15.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/16.png b/source/core/assets/planets/Barren_or_Moon_bw/2/16.png new file mode 100644 index 000000000..8d2c0c3f8 Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/16.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/17.png b/source/core/assets/planets/Barren_or_Moon_bw/2/17.png new file mode 100644 index 000000000..103f9db5d Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/17.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/18.png b/source/core/assets/planets/Barren_or_Moon_bw/2/18.png new file mode 100644 index 000000000..8042f62d4 Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/18.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/19.png b/source/core/assets/planets/Barren_or_Moon_bw/2/19.png new file mode 100644 index 000000000..e1f79e56d Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/19.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/2.png b/source/core/assets/planets/Barren_or_Moon_bw/2/2.png new file mode 100644 index 000000000..dfe9ffa08 Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/2.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/20.png b/source/core/assets/planets/Barren_or_Moon_bw/2/20.png new file mode 100644 index 000000000..ea08f6dbd Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/20.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/21.png b/source/core/assets/planets/Barren_or_Moon_bw/2/21.png new file mode 100644 index 000000000..6c0e49f7e Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/21.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/22.png b/source/core/assets/planets/Barren_or_Moon_bw/2/22.png new file mode 100644 index 000000000..8f46a8754 Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/22.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/23.png b/source/core/assets/planets/Barren_or_Moon_bw/2/23.png new file mode 100644 index 000000000..d26c3f2cb Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/23.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/24.png b/source/core/assets/planets/Barren_or_Moon_bw/2/24.png new file mode 100644 index 000000000..31a0089cf Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/24.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/25.png b/source/core/assets/planets/Barren_or_Moon_bw/2/25.png new file mode 100644 index 000000000..0f6b42833 Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/25.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/26.png b/source/core/assets/planets/Barren_or_Moon_bw/2/26.png new file mode 100644 index 000000000..ceea305fb Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/26.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/27.png b/source/core/assets/planets/Barren_or_Moon_bw/2/27.png new file mode 100644 index 000000000..409574f4b Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/27.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/28.png b/source/core/assets/planets/Barren_or_Moon_bw/2/28.png new file mode 100644 index 000000000..da0ea0619 Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/28.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/29.png b/source/core/assets/planets/Barren_or_Moon_bw/2/29.png new file mode 100644 index 000000000..5883a9c6a Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/29.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/3.png b/source/core/assets/planets/Barren_or_Moon_bw/2/3.png new file mode 100644 index 000000000..3eb5240ad Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/3.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/30.png b/source/core/assets/planets/Barren_or_Moon_bw/2/30.png new file mode 100644 index 000000000..6b9a39fda Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/30.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/31.png b/source/core/assets/planets/Barren_or_Moon_bw/2/31.png new file mode 100644 index 000000000..0ab825429 Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/31.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/32.png b/source/core/assets/planets/Barren_or_Moon_bw/2/32.png new file mode 100644 index 000000000..3f819e061 Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/32.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/33.png b/source/core/assets/planets/Barren_or_Moon_bw/2/33.png new file mode 100644 index 000000000..3b342a7ed Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/33.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/34.png b/source/core/assets/planets/Barren_or_Moon_bw/2/34.png new file mode 100644 index 000000000..7e72e313b Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/34.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/35.png b/source/core/assets/planets/Barren_or_Moon_bw/2/35.png new file mode 100644 index 000000000..8f7a6ac98 Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/35.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/36.png b/source/core/assets/planets/Barren_or_Moon_bw/2/36.png new file mode 100644 index 000000000..c97607f9d Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/36.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/37.png b/source/core/assets/planets/Barren_or_Moon_bw/2/37.png new file mode 100644 index 000000000..111369017 Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/37.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/38.png b/source/core/assets/planets/Barren_or_Moon_bw/2/38.png new file mode 100644 index 000000000..d1bcb6c63 Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/38.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/39.png b/source/core/assets/planets/Barren_or_Moon_bw/2/39.png new file mode 100644 index 000000000..a95c42274 Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/39.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/4.png b/source/core/assets/planets/Barren_or_Moon_bw/2/4.png new file mode 100644 index 000000000..99c9a00e7 Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/4.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/40.png b/source/core/assets/planets/Barren_or_Moon_bw/2/40.png new file mode 100644 index 000000000..a2d84a577 Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/40.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/41.png b/source/core/assets/planets/Barren_or_Moon_bw/2/41.png new file mode 100644 index 000000000..5a5e72514 Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/41.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/42.png b/source/core/assets/planets/Barren_or_Moon_bw/2/42.png new file mode 100644 index 000000000..184df838b Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/42.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/43.png b/source/core/assets/planets/Barren_or_Moon_bw/2/43.png new file mode 100644 index 000000000..28e4bd7b2 Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/43.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/44.png b/source/core/assets/planets/Barren_or_Moon_bw/2/44.png new file mode 100644 index 000000000..08e086483 Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/44.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/45.png b/source/core/assets/planets/Barren_or_Moon_bw/2/45.png new file mode 100644 index 000000000..8f0966920 Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/45.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/46.png b/source/core/assets/planets/Barren_or_Moon_bw/2/46.png new file mode 100644 index 000000000..2746951f5 Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/46.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/47.png b/source/core/assets/planets/Barren_or_Moon_bw/2/47.png new file mode 100644 index 000000000..f4c6864b6 Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/47.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/48.png b/source/core/assets/planets/Barren_or_Moon_bw/2/48.png new file mode 100644 index 000000000..9cc5ffc9a Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/48.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/49.png b/source/core/assets/planets/Barren_or_Moon_bw/2/49.png new file mode 100644 index 000000000..5d6713c95 Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/49.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/5.png b/source/core/assets/planets/Barren_or_Moon_bw/2/5.png new file mode 100644 index 000000000..89bc4b608 Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/5.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/50.png b/source/core/assets/planets/Barren_or_Moon_bw/2/50.png new file mode 100644 index 000000000..6f917443d Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/50.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/51.png b/source/core/assets/planets/Barren_or_Moon_bw/2/51.png new file mode 100644 index 000000000..d4dbdbec9 Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/51.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/52.png b/source/core/assets/planets/Barren_or_Moon_bw/2/52.png new file mode 100644 index 000000000..98e7cc3dc Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/52.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/53.png b/source/core/assets/planets/Barren_or_Moon_bw/2/53.png new file mode 100644 index 000000000..e63b80a40 Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/53.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/54.png b/source/core/assets/planets/Barren_or_Moon_bw/2/54.png new file mode 100644 index 000000000..440ab7aca Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/54.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/55.png b/source/core/assets/planets/Barren_or_Moon_bw/2/55.png new file mode 100644 index 000000000..fee03428e Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/55.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/56.png b/source/core/assets/planets/Barren_or_Moon_bw/2/56.png new file mode 100644 index 000000000..d7efc51e5 Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/56.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/57.png b/source/core/assets/planets/Barren_or_Moon_bw/2/57.png new file mode 100644 index 000000000..4636aa76b Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/57.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/58.png b/source/core/assets/planets/Barren_or_Moon_bw/2/58.png new file mode 100644 index 000000000..f0d2ce55b Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/58.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/59.png b/source/core/assets/planets/Barren_or_Moon_bw/2/59.png new file mode 100644 index 000000000..1dcca3581 Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/59.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/6.png b/source/core/assets/planets/Barren_or_Moon_bw/2/6.png new file mode 100644 index 000000000..2b2446eec Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/6.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/60.png b/source/core/assets/planets/Barren_or_Moon_bw/2/60.png new file mode 100644 index 000000000..ee5a8d3d6 Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/60.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/7.png b/source/core/assets/planets/Barren_or_Moon_bw/2/7.png new file mode 100644 index 000000000..aa572f74c Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/7.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/8.png b/source/core/assets/planets/Barren_or_Moon_bw/2/8.png new file mode 100644 index 000000000..cc337325f Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/8.png differ diff --git a/source/core/assets/planets/Barren_or_Moon_bw/2/9.png b/source/core/assets/planets/Barren_or_Moon_bw/2/9.png new file mode 100644 index 000000000..343949c3e Binary files /dev/null and b/source/core/assets/planets/Barren_or_Moon_bw/2/9.png differ diff --git a/source/core/assets/planets/Desert_bw/1/1.png b/source/core/assets/planets/Desert_bw/1/1.png new file mode 100644 index 000000000..f46cb689c Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/1.png differ diff --git a/source/core/assets/planets/Desert_bw/1/10.png b/source/core/assets/planets/Desert_bw/1/10.png new file mode 100644 index 000000000..c58ab448a Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/10.png differ diff --git a/source/core/assets/planets/Desert_bw/1/11.png b/source/core/assets/planets/Desert_bw/1/11.png new file mode 100644 index 000000000..9bbd5b15e Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/11.png differ diff --git a/source/core/assets/planets/Desert_bw/1/12.png b/source/core/assets/planets/Desert_bw/1/12.png new file mode 100644 index 000000000..10ab860ba Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/12.png differ diff --git a/source/core/assets/planets/Desert_bw/1/13.png b/source/core/assets/planets/Desert_bw/1/13.png new file mode 100644 index 000000000..06bef9c92 Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/13.png differ diff --git a/source/core/assets/planets/Desert_bw/1/14.png b/source/core/assets/planets/Desert_bw/1/14.png new file mode 100644 index 000000000..8190ee841 Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/14.png differ diff --git a/source/core/assets/planets/Desert_bw/1/15.png b/source/core/assets/planets/Desert_bw/1/15.png new file mode 100644 index 000000000..4e8d7bc08 Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/15.png differ diff --git a/source/core/assets/planets/Desert_bw/1/16.png b/source/core/assets/planets/Desert_bw/1/16.png new file mode 100644 index 000000000..698ee30e1 Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/16.png differ diff --git a/source/core/assets/planets/Desert_bw/1/17.png b/source/core/assets/planets/Desert_bw/1/17.png new file mode 100644 index 000000000..58a1598b7 Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/17.png differ diff --git a/source/core/assets/planets/Desert_bw/1/18.png b/source/core/assets/planets/Desert_bw/1/18.png new file mode 100644 index 000000000..986726910 Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/18.png differ diff --git a/source/core/assets/planets/Desert_bw/1/19.png b/source/core/assets/planets/Desert_bw/1/19.png new file mode 100644 index 000000000..b8eaff229 Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/19.png differ diff --git a/source/core/assets/planets/Desert_bw/1/2.png b/source/core/assets/planets/Desert_bw/1/2.png new file mode 100644 index 000000000..2781b82e5 Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/2.png differ diff --git a/source/core/assets/planets/Desert_bw/1/20.png b/source/core/assets/planets/Desert_bw/1/20.png new file mode 100644 index 000000000..b3958de59 Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/20.png differ diff --git a/source/core/assets/planets/Desert_bw/1/21.png b/source/core/assets/planets/Desert_bw/1/21.png new file mode 100644 index 000000000..a60474811 Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/21.png differ diff --git a/source/core/assets/planets/Desert_bw/1/22.png b/source/core/assets/planets/Desert_bw/1/22.png new file mode 100644 index 000000000..c601e384f Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/22.png differ diff --git a/source/core/assets/planets/Desert_bw/1/23.png b/source/core/assets/planets/Desert_bw/1/23.png new file mode 100644 index 000000000..6d85bd568 Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/23.png differ diff --git a/source/core/assets/planets/Desert_bw/1/24.png b/source/core/assets/planets/Desert_bw/1/24.png new file mode 100644 index 000000000..9a87cb24a Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/24.png differ diff --git a/source/core/assets/planets/Desert_bw/1/25.png b/source/core/assets/planets/Desert_bw/1/25.png new file mode 100644 index 000000000..ba3244ab0 Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/25.png differ diff --git a/source/core/assets/planets/Desert_bw/1/26.png b/source/core/assets/planets/Desert_bw/1/26.png new file mode 100644 index 000000000..bbf9658cd Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/26.png differ diff --git a/source/core/assets/planets/Desert_bw/1/27.png b/source/core/assets/planets/Desert_bw/1/27.png new file mode 100644 index 000000000..085b26c0c Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/27.png differ diff --git a/source/core/assets/planets/Desert_bw/1/28.png b/source/core/assets/planets/Desert_bw/1/28.png new file mode 100644 index 000000000..90e2e3973 Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/28.png differ diff --git a/source/core/assets/planets/Desert_bw/1/29.png b/source/core/assets/planets/Desert_bw/1/29.png new file mode 100644 index 000000000..0b45d77e2 Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/29.png differ diff --git a/source/core/assets/planets/Desert_bw/1/3.png b/source/core/assets/planets/Desert_bw/1/3.png new file mode 100644 index 000000000..7e4940092 Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/3.png differ diff --git a/source/core/assets/planets/Desert_bw/1/30.png b/source/core/assets/planets/Desert_bw/1/30.png new file mode 100644 index 000000000..699ef070a Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/30.png differ diff --git a/source/core/assets/planets/Desert_bw/1/31.png b/source/core/assets/planets/Desert_bw/1/31.png new file mode 100644 index 000000000..a9ae13fe9 Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/31.png differ diff --git a/source/core/assets/planets/Desert_bw/1/32.png b/source/core/assets/planets/Desert_bw/1/32.png new file mode 100644 index 000000000..8f67ae032 Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/32.png differ diff --git a/source/core/assets/planets/Desert_bw/1/33.png b/source/core/assets/planets/Desert_bw/1/33.png new file mode 100644 index 000000000..18a14ff2d Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/33.png differ diff --git a/source/core/assets/planets/Desert_bw/1/34.png b/source/core/assets/planets/Desert_bw/1/34.png new file mode 100644 index 000000000..79f2d7d4a Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/34.png differ diff --git a/source/core/assets/planets/Desert_bw/1/35.png b/source/core/assets/planets/Desert_bw/1/35.png new file mode 100644 index 000000000..e6b295841 Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/35.png differ diff --git a/source/core/assets/planets/Desert_bw/1/36.png b/source/core/assets/planets/Desert_bw/1/36.png new file mode 100644 index 000000000..a93e0e1b2 Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/36.png differ diff --git a/source/core/assets/planets/Desert_bw/1/37.png b/source/core/assets/planets/Desert_bw/1/37.png new file mode 100644 index 000000000..8fce78ac2 Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/37.png differ diff --git a/source/core/assets/planets/Desert_bw/1/38.png b/source/core/assets/planets/Desert_bw/1/38.png new file mode 100644 index 000000000..48421334b Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/38.png differ diff --git a/source/core/assets/planets/Desert_bw/1/39.png b/source/core/assets/planets/Desert_bw/1/39.png new file mode 100644 index 000000000..deb2f6036 Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/39.png differ diff --git a/source/core/assets/planets/Desert_bw/1/4.png b/source/core/assets/planets/Desert_bw/1/4.png new file mode 100644 index 000000000..ba06391a6 Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/4.png differ diff --git a/source/core/assets/planets/Desert_bw/1/40.png b/source/core/assets/planets/Desert_bw/1/40.png new file mode 100644 index 000000000..369e0c071 Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/40.png differ diff --git a/source/core/assets/planets/Desert_bw/1/41.png b/source/core/assets/planets/Desert_bw/1/41.png new file mode 100644 index 000000000..fb5c606ac Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/41.png differ diff --git a/source/core/assets/planets/Desert_bw/1/42.png b/source/core/assets/planets/Desert_bw/1/42.png new file mode 100644 index 000000000..8f19f22af Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/42.png differ diff --git a/source/core/assets/planets/Desert_bw/1/43.png b/source/core/assets/planets/Desert_bw/1/43.png new file mode 100644 index 000000000..7733662a7 Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/43.png differ diff --git a/source/core/assets/planets/Desert_bw/1/44.png b/source/core/assets/planets/Desert_bw/1/44.png new file mode 100644 index 000000000..b838b2be2 Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/44.png differ diff --git a/source/core/assets/planets/Desert_bw/1/45.png b/source/core/assets/planets/Desert_bw/1/45.png new file mode 100644 index 000000000..292b15023 Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/45.png differ diff --git a/source/core/assets/planets/Desert_bw/1/46.png b/source/core/assets/planets/Desert_bw/1/46.png new file mode 100644 index 000000000..794767e0b Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/46.png differ diff --git a/source/core/assets/planets/Desert_bw/1/47.png b/source/core/assets/planets/Desert_bw/1/47.png new file mode 100644 index 000000000..09fb1ba56 Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/47.png differ diff --git a/source/core/assets/planets/Desert_bw/1/48.png b/source/core/assets/planets/Desert_bw/1/48.png new file mode 100644 index 000000000..0841b4217 Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/48.png differ diff --git a/source/core/assets/planets/Desert_bw/1/49.png b/source/core/assets/planets/Desert_bw/1/49.png new file mode 100644 index 000000000..456f88164 Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/49.png differ diff --git a/source/core/assets/planets/Desert_bw/1/5.png b/source/core/assets/planets/Desert_bw/1/5.png new file mode 100644 index 000000000..f1f657c6b Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/5.png differ diff --git a/source/core/assets/planets/Desert_bw/1/50.png b/source/core/assets/planets/Desert_bw/1/50.png new file mode 100644 index 000000000..2dbba0084 Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/50.png differ diff --git a/source/core/assets/planets/Desert_bw/1/51.png b/source/core/assets/planets/Desert_bw/1/51.png new file mode 100644 index 000000000..525975029 Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/51.png differ diff --git a/source/core/assets/planets/Desert_bw/1/52.png b/source/core/assets/planets/Desert_bw/1/52.png new file mode 100644 index 000000000..53636a69d Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/52.png differ diff --git a/source/core/assets/planets/Desert_bw/1/53.png b/source/core/assets/planets/Desert_bw/1/53.png new file mode 100644 index 000000000..d33b7b261 Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/53.png differ diff --git a/source/core/assets/planets/Desert_bw/1/54.png b/source/core/assets/planets/Desert_bw/1/54.png new file mode 100644 index 000000000..69b2e663a Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/54.png differ diff --git a/source/core/assets/planets/Desert_bw/1/55.png b/source/core/assets/planets/Desert_bw/1/55.png new file mode 100644 index 000000000..0361ed344 Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/55.png differ diff --git a/source/core/assets/planets/Desert_bw/1/56.png b/source/core/assets/planets/Desert_bw/1/56.png new file mode 100644 index 000000000..8fa954a7c Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/56.png differ diff --git a/source/core/assets/planets/Desert_bw/1/57.png b/source/core/assets/planets/Desert_bw/1/57.png new file mode 100644 index 000000000..0046fa102 Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/57.png differ diff --git a/source/core/assets/planets/Desert_bw/1/58.png b/source/core/assets/planets/Desert_bw/1/58.png new file mode 100644 index 000000000..c5d0730ce Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/58.png differ diff --git a/source/core/assets/planets/Desert_bw/1/59.png b/source/core/assets/planets/Desert_bw/1/59.png new file mode 100644 index 000000000..d439755c9 Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/59.png differ diff --git a/source/core/assets/planets/Desert_bw/1/6.png b/source/core/assets/planets/Desert_bw/1/6.png new file mode 100644 index 000000000..4f47a7d3a Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/6.png differ diff --git a/source/core/assets/planets/Desert_bw/1/60.png b/source/core/assets/planets/Desert_bw/1/60.png new file mode 100644 index 000000000..901ec426c Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/60.png differ diff --git a/source/core/assets/planets/Desert_bw/1/7.png b/source/core/assets/planets/Desert_bw/1/7.png new file mode 100644 index 000000000..4afcc2b93 Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/7.png differ diff --git a/source/core/assets/planets/Desert_bw/1/8.png b/source/core/assets/planets/Desert_bw/1/8.png new file mode 100644 index 000000000..2467b0973 Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/8.png differ diff --git a/source/core/assets/planets/Desert_bw/1/9.png b/source/core/assets/planets/Desert_bw/1/9.png new file mode 100644 index 000000000..64c1407ae Binary files /dev/null and b/source/core/assets/planets/Desert_bw/1/9.png differ diff --git a/source/core/assets/planets/Lava_bw/1/1.png b/source/core/assets/planets/Lava_bw/1/1.png new file mode 100644 index 000000000..4870542ee Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/1.png differ diff --git a/source/core/assets/planets/Lava_bw/1/10.png b/source/core/assets/planets/Lava_bw/1/10.png new file mode 100644 index 000000000..27892603b Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/10.png differ diff --git a/source/core/assets/planets/Lava_bw/1/11.png b/source/core/assets/planets/Lava_bw/1/11.png new file mode 100644 index 000000000..8fac24b5e Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/11.png differ diff --git a/source/core/assets/planets/Lava_bw/1/12.png b/source/core/assets/planets/Lava_bw/1/12.png new file mode 100644 index 000000000..9d1506a86 Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/12.png differ diff --git a/source/core/assets/planets/Lava_bw/1/13.png b/source/core/assets/planets/Lava_bw/1/13.png new file mode 100644 index 000000000..6772fcc59 Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/13.png differ diff --git a/source/core/assets/planets/Lava_bw/1/14.png b/source/core/assets/planets/Lava_bw/1/14.png new file mode 100644 index 000000000..a840e1709 Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/14.png differ diff --git a/source/core/assets/planets/Lava_bw/1/15.png b/source/core/assets/planets/Lava_bw/1/15.png new file mode 100644 index 000000000..1ecb607ba Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/15.png differ diff --git a/source/core/assets/planets/Lava_bw/1/16.png b/source/core/assets/planets/Lava_bw/1/16.png new file mode 100644 index 000000000..cea173d97 Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/16.png differ diff --git a/source/core/assets/planets/Lava_bw/1/17.png b/source/core/assets/planets/Lava_bw/1/17.png new file mode 100644 index 000000000..96aa223a8 Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/17.png differ diff --git a/source/core/assets/planets/Lava_bw/1/18.png b/source/core/assets/planets/Lava_bw/1/18.png new file mode 100644 index 000000000..9f747091a Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/18.png differ diff --git a/source/core/assets/planets/Lava_bw/1/19.png b/source/core/assets/planets/Lava_bw/1/19.png new file mode 100644 index 000000000..26f7dd583 Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/19.png differ diff --git a/source/core/assets/planets/Lava_bw/1/2.png b/source/core/assets/planets/Lava_bw/1/2.png new file mode 100644 index 000000000..2f3256c15 Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/2.png differ diff --git a/source/core/assets/planets/Lava_bw/1/20.png b/source/core/assets/planets/Lava_bw/1/20.png new file mode 100644 index 000000000..6635e686b Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/20.png differ diff --git a/source/core/assets/planets/Lava_bw/1/21.png b/source/core/assets/planets/Lava_bw/1/21.png new file mode 100644 index 000000000..4a652d49d Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/21.png differ diff --git a/source/core/assets/planets/Lava_bw/1/22.png b/source/core/assets/planets/Lava_bw/1/22.png new file mode 100644 index 000000000..7d2e6a780 Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/22.png differ diff --git a/source/core/assets/planets/Lava_bw/1/23.png b/source/core/assets/planets/Lava_bw/1/23.png new file mode 100644 index 000000000..87d84cae8 Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/23.png differ diff --git a/source/core/assets/planets/Lava_bw/1/24.png b/source/core/assets/planets/Lava_bw/1/24.png new file mode 100644 index 000000000..0afc0c14f Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/24.png differ diff --git a/source/core/assets/planets/Lava_bw/1/25.png b/source/core/assets/planets/Lava_bw/1/25.png new file mode 100644 index 000000000..8c1f67ad8 Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/25.png differ diff --git a/source/core/assets/planets/Lava_bw/1/26.png b/source/core/assets/planets/Lava_bw/1/26.png new file mode 100644 index 000000000..3821d8c8f Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/26.png differ diff --git a/source/core/assets/planets/Lava_bw/1/27.png b/source/core/assets/planets/Lava_bw/1/27.png new file mode 100644 index 000000000..4f1278653 Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/27.png differ diff --git a/source/core/assets/planets/Lava_bw/1/28.png b/source/core/assets/planets/Lava_bw/1/28.png new file mode 100644 index 000000000..e63366a61 Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/28.png differ diff --git a/source/core/assets/planets/Lava_bw/1/29.png b/source/core/assets/planets/Lava_bw/1/29.png new file mode 100644 index 000000000..9d2f957c6 Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/29.png differ diff --git a/source/core/assets/planets/Lava_bw/1/3.png b/source/core/assets/planets/Lava_bw/1/3.png new file mode 100644 index 000000000..c1832e3f8 Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/3.png differ diff --git a/source/core/assets/planets/Lava_bw/1/30.png b/source/core/assets/planets/Lava_bw/1/30.png new file mode 100644 index 000000000..a829dcafa Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/30.png differ diff --git a/source/core/assets/planets/Lava_bw/1/31.png b/source/core/assets/planets/Lava_bw/1/31.png new file mode 100644 index 000000000..bca15d38a Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/31.png differ diff --git a/source/core/assets/planets/Lava_bw/1/32.png b/source/core/assets/planets/Lava_bw/1/32.png new file mode 100644 index 000000000..f518b17ea Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/32.png differ diff --git a/source/core/assets/planets/Lava_bw/1/33.png b/source/core/assets/planets/Lava_bw/1/33.png new file mode 100644 index 000000000..c198e15dc Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/33.png differ diff --git a/source/core/assets/planets/Lava_bw/1/34.png b/source/core/assets/planets/Lava_bw/1/34.png new file mode 100644 index 000000000..5858eed2b Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/34.png differ diff --git a/source/core/assets/planets/Lava_bw/1/35.png b/source/core/assets/planets/Lava_bw/1/35.png new file mode 100644 index 000000000..4ff7388ba Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/35.png differ diff --git a/source/core/assets/planets/Lava_bw/1/36.png b/source/core/assets/planets/Lava_bw/1/36.png new file mode 100644 index 000000000..bdbe0cbbc Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/36.png differ diff --git a/source/core/assets/planets/Lava_bw/1/37.png b/source/core/assets/planets/Lava_bw/1/37.png new file mode 100644 index 000000000..124696f87 Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/37.png differ diff --git a/source/core/assets/planets/Lava_bw/1/38.png b/source/core/assets/planets/Lava_bw/1/38.png new file mode 100644 index 000000000..b63846619 Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/38.png differ diff --git a/source/core/assets/planets/Lava_bw/1/39.png b/source/core/assets/planets/Lava_bw/1/39.png new file mode 100644 index 000000000..69067340a Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/39.png differ diff --git a/source/core/assets/planets/Lava_bw/1/4.png b/source/core/assets/planets/Lava_bw/1/4.png new file mode 100644 index 000000000..1bba4861a Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/4.png differ diff --git a/source/core/assets/planets/Lava_bw/1/40.png b/source/core/assets/planets/Lava_bw/1/40.png new file mode 100644 index 000000000..b73701d1c Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/40.png differ diff --git a/source/core/assets/planets/Lava_bw/1/41.png b/source/core/assets/planets/Lava_bw/1/41.png new file mode 100644 index 000000000..be99af527 Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/41.png differ diff --git a/source/core/assets/planets/Lava_bw/1/42.png b/source/core/assets/planets/Lava_bw/1/42.png new file mode 100644 index 000000000..0f33b33ef Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/42.png differ diff --git a/source/core/assets/planets/Lava_bw/1/43.png b/source/core/assets/planets/Lava_bw/1/43.png new file mode 100644 index 000000000..12da1adfd Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/43.png differ diff --git a/source/core/assets/planets/Lava_bw/1/44.png b/source/core/assets/planets/Lava_bw/1/44.png new file mode 100644 index 000000000..bb3861f75 Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/44.png differ diff --git a/source/core/assets/planets/Lava_bw/1/45.png b/source/core/assets/planets/Lava_bw/1/45.png new file mode 100644 index 000000000..4889df522 Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/45.png differ diff --git a/source/core/assets/planets/Lava_bw/1/46.png b/source/core/assets/planets/Lava_bw/1/46.png new file mode 100644 index 000000000..f42703873 Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/46.png differ diff --git a/source/core/assets/planets/Lava_bw/1/47.png b/source/core/assets/planets/Lava_bw/1/47.png new file mode 100644 index 000000000..fd942d677 Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/47.png differ diff --git a/source/core/assets/planets/Lava_bw/1/48.png b/source/core/assets/planets/Lava_bw/1/48.png new file mode 100644 index 000000000..a52660c5f Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/48.png differ diff --git a/source/core/assets/planets/Lava_bw/1/49.png b/source/core/assets/planets/Lava_bw/1/49.png new file mode 100644 index 000000000..66f7a08b1 Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/49.png differ diff --git a/source/core/assets/planets/Lava_bw/1/5.png b/source/core/assets/planets/Lava_bw/1/5.png new file mode 100644 index 000000000..dcab56667 Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/5.png differ diff --git a/source/core/assets/planets/Lava_bw/1/50.png b/source/core/assets/planets/Lava_bw/1/50.png new file mode 100644 index 000000000..f93a71b11 Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/50.png differ diff --git a/source/core/assets/planets/Lava_bw/1/51.png b/source/core/assets/planets/Lava_bw/1/51.png new file mode 100644 index 000000000..34c30435c Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/51.png differ diff --git a/source/core/assets/planets/Lava_bw/1/52.png b/source/core/assets/planets/Lava_bw/1/52.png new file mode 100644 index 000000000..a9d61627e Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/52.png differ diff --git a/source/core/assets/planets/Lava_bw/1/53.png b/source/core/assets/planets/Lava_bw/1/53.png new file mode 100644 index 000000000..d8f6e7acc Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/53.png differ diff --git a/source/core/assets/planets/Lava_bw/1/54.png b/source/core/assets/planets/Lava_bw/1/54.png new file mode 100644 index 000000000..1977fe824 Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/54.png differ diff --git a/source/core/assets/planets/Lava_bw/1/55.png b/source/core/assets/planets/Lava_bw/1/55.png new file mode 100644 index 000000000..41d663d35 Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/55.png differ diff --git a/source/core/assets/planets/Lava_bw/1/56.png b/source/core/assets/planets/Lava_bw/1/56.png new file mode 100644 index 000000000..58e5ce938 Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/56.png differ diff --git a/source/core/assets/planets/Lava_bw/1/57.png b/source/core/assets/planets/Lava_bw/1/57.png new file mode 100644 index 000000000..92f979cb6 Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/57.png differ diff --git a/source/core/assets/planets/Lava_bw/1/58.png b/source/core/assets/planets/Lava_bw/1/58.png new file mode 100644 index 000000000..2453d425a Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/58.png differ diff --git a/source/core/assets/planets/Lava_bw/1/59.png b/source/core/assets/planets/Lava_bw/1/59.png new file mode 100644 index 000000000..34b374eb6 Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/59.png differ diff --git a/source/core/assets/planets/Lava_bw/1/6.png b/source/core/assets/planets/Lava_bw/1/6.png new file mode 100644 index 000000000..b0ede3cc5 Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/6.png differ diff --git a/source/core/assets/planets/Lava_bw/1/60.png b/source/core/assets/planets/Lava_bw/1/60.png new file mode 100644 index 000000000..27c94b8f7 Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/60.png differ diff --git a/source/core/assets/planets/Lava_bw/1/7.png b/source/core/assets/planets/Lava_bw/1/7.png new file mode 100644 index 000000000..2d651f42c Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/7.png differ diff --git a/source/core/assets/planets/Lava_bw/1/8.png b/source/core/assets/planets/Lava_bw/1/8.png new file mode 100644 index 000000000..1d823981d Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/8.png differ diff --git a/source/core/assets/planets/Lava_bw/1/9.png b/source/core/assets/planets/Lava_bw/1/9.png new file mode 100644 index 000000000..db5fca99d Binary files /dev/null and b/source/core/assets/planets/Lava_bw/1/9.png differ diff --git a/source/core/assets/sounds/towers/12-Gauge-Pump-Action-Shotgun-Close-Gunshot.mp3 b/source/core/assets/sounds/towers/12-Gauge-Pump-Action-Shotgun-Close-Gunshot.mp3 new file mode 100644 index 000000000..61bc03e66 Binary files /dev/null and b/source/core/assets/sounds/towers/12-Gauge-Pump-Action-Shotgun-Close-Gunshot.mp3 differ diff --git a/source/core/assets/sounds/towers/5.56_single_shot.mp3 b/source/core/assets/sounds/towers/5.56_single_shot.mp3 new file mode 100644 index 000000000..0611f722a Binary files /dev/null and b/source/core/assets/sounds/towers/5.56_single_shot.mp3 differ diff --git a/source/core/assets/sounds/towers/Desert-Eagle-Far-Single-Gunshot.mp3 b/source/core/assets/sounds/towers/Desert-Eagle-Far-Single-Gunshot.mp3 new file mode 100644 index 000000000..5e39881ce Binary files /dev/null and b/source/core/assets/sounds/towers/Desert-Eagle-Far-Single-Gunshot.mp3 differ diff --git a/source/core/assets/sounds/towers/ar15_single_shot_far.mp3 b/source/core/assets/sounds/towers/ar15_single_shot_far.mp3 new file mode 100644 index 000000000..34ef353ee Binary files /dev/null and b/source/core/assets/sounds/towers/ar15_single_shot_far.mp3 differ diff --git a/source/core/assets/sounds/towers/eco_tower_ping.mp3 b/source/core/assets/sounds/towers/eco_tower_ping.mp3 new file mode 100644 index 000000000..1553f6001 Binary files /dev/null and b/source/core/assets/sounds/towers/eco_tower_ping.mp3 differ diff --git a/source/core/assets/sounds/towers/explosion.mp3 b/source/core/assets/sounds/towers/explosion.mp3 new file mode 100644 index 000000000..ab19a2947 Binary files /dev/null and b/source/core/assets/sounds/towers/explosion.mp3 differ diff --git a/source/core/src/main/com/csse3200/game/GdxGame.java b/source/core/src/main/com/csse3200/game/GdxGame.java index 903f443d3..5e3581d2a 100644 --- a/source/core/src/main/com/csse3200/game/GdxGame.java +++ b/source/core/src/main/com/csse3200/game/GdxGame.java @@ -5,6 +5,7 @@ import com.badlogic.gdx.Screen; import com.csse3200.game.files.UserSettings; import com.csse3200.game.screens.*; +import com.csse3200.game.screens.HelpScreen.*; import org.slf4j.Logger; import org.slf4j.LoggerFactory; @@ -17,7 +18,7 @@ */ public class GdxGame extends Game { private static final Logger logger = LoggerFactory.getLogger(GdxGame.class); - + public int currentLevel = 0; @Override public void create() { logger.info("Creating game"); @@ -63,30 +64,31 @@ public void dispose() { * @return new screen */ private Screen newScreen(ScreenType screenType) { - switch (screenType) { - case MAIN_MENU: - return new MainMenuScreen(this); - case MAIN_GAME: - return new MainGameScreen(this); - case SETTINGS: - return new SettingsScreen(this); - case STORY_SCREEN: - return new StoryScreen(this); - case LEVEL_SELECT: - return new LevelSelectScreen(this); - case LOSING_SCREEN: - return new LosingScreen(this); - case TURRET_SELECTION: - return new TurretSelectionScreen(this); - case HELP_SCREEN: - return new HelpScreen(this); - default: - return null; - } + return switch (screenType) { + case Next_Screen -> new NextLevelScreen(this, currentLevel); + case Win_Screen -> new WinningScreen(this); + case MAIN_MENU -> new MainMenuScreen(this); + case MAIN_GAME -> new MainGameScreen(this); + case SETTINGS -> new SettingsScreen(this); + case STORY_SCREEN -> new StoryScreen(this); + case LEVEL_SELECT -> new LevelSelectScreen(this, currentLevel); + case LOSING_SCREEN -> new LosingScreen(this); + case TURRET_SELECTION -> new TurretSelectionScreen(this); + case HELP_SCREEN -> new GameDescriptionHelpScreen(this); + case HELP_MOBS_SCREEN -> new MobsDescriptionHelpScreen(this); + case HELP_TOWER_SCREEN -> new TowerDescriptionHelpScreen(this); + case HELP_BOSS_SCREEN -> new BossDescriptionHelpScreen(this); + case LOAD_SCREEN -> new LoadingScreen(this); + case HOW_TO_PLAY -> new HowToPlay(this); + + case TUTORIAL_SCREEN-> new Tutorial(this); + default-> null; + }; } public enum ScreenType { - MAIN_MENU, MAIN_GAME, SETTINGS, STORY_SCREEN, LEVEL_SELECT, TURRET_SELECTION, LOSING_SCREEN, HELP_SCREEN + MAIN_MENU, MAIN_GAME, SETTINGS, STORY_SCREEN, LEVEL_SELECT, TURRET_SELECTION, LOSING_SCREEN, HELP_SCREEN, LOAD_SCREEN, + HELP_MOBS_SCREEN, HELP_TOWER_SCREEN, HELP_BOSS_SCREEN, Win_Screen, Next_Screen, HOW_TO_PLAY, TUTORIAL_SCREEN } /** @@ -95,4 +97,4 @@ public enum ScreenType { public void exit() { app.exit(); } -} +} \ No newline at end of file diff --git a/source/core/src/main/com/csse3200/game/ai/tasks/AITaskComponent.java b/source/core/src/main/com/csse3200/game/ai/tasks/AITaskComponent.java index 93798e8d4..7c165ef09 100644 --- a/source/core/src/main/com/csse3200/game/ai/tasks/AITaskComponent.java +++ b/source/core/src/main/com/csse3200/game/ai/tasks/AITaskComponent.java @@ -1,7 +1,6 @@ package com.csse3200.game.ai.tasks; import com.csse3200.game.components.Component; -import com.csse3200.game.components.ComponentType; import org.slf4j.Logger; import org.slf4j.LoggerFactory; @@ -46,8 +45,8 @@ public AITaskComponent addTask(PriorityTask task) { */ public T getTask(Class task) { for (PriorityTask priorityTask : priorityTasks) { - if (priorityTask.getClass() == task) { - return (T) priorityTask; + if (task.isInstance(priorityTask)) { + return task.cast(priorityTask); } } logger.info("Task {} not found", task); @@ -83,12 +82,8 @@ public void dispose() { */ public void disposeAll() { currentTask = null; - for (int i = 0; i < priorityTasks.size(); i++) { - priorityTasksToBeRestored.add(priorityTasks.get(i)); - } - for (int i = 0; i < priorityTasks.size(); i++) { - priorityTasks.remove(i); - } + priorityTasksToBeRestored.addAll(priorityTasks); + priorityTasks.clear(); } /** @@ -96,12 +91,8 @@ public void disposeAll() { * back into priorityTasks. */ public void restore() { - for (int i = 0; i < priorityTasksToBeRestored.size(); i++) { - priorityTasks.add(priorityTasksToBeRestored.get(i)); - } - for (int i = 0; i < priorityTasksToBeRestored.size(); i++) { - priorityTasksToBeRestored.remove(i); - } + priorityTasks.addAll(priorityTasksToBeRestored); + priorityTasksToBeRestored.clear(); this.update(); } diff --git a/source/core/src/main/com/csse3200/game/areas/ForestGameArea.java b/source/core/src/main/com/csse3200/game/areas/ForestGameArea.java index ca533e945..297205850 100644 --- a/source/core/src/main/com/csse3200/game/areas/ForestGameArea.java +++ b/source/core/src/main/com/csse3200/game/areas/ForestGameArea.java @@ -8,11 +8,14 @@ import com.csse3200.game.components.ProjectileEffects; +import com.csse3200.game.services.GameTime; import com.csse3200.game.utils.math.RandomUtils; import com.csse3200.game.services.ResourceService; import com.csse3200.game.services.ServiceLocator; import org.slf4j.Logger; import org.slf4j.LoggerFactory; + +import java.security.SecureRandom; import java.util.Timer; import static com.csse3200.game.entities.factories.WarningFactory.createWarning; @@ -22,9 +25,14 @@ public class ForestGameArea extends GameArea { private static final Logger logger = LoggerFactory.getLogger(ForestGameArea.class); - private Timer waveTimer; + /* Change below to change the number of ms between spawns of engineers in the case */ + private static final long ENGINEER_MIN_SPAWN_INTERVAL = 1000; + + private long lastSpawnTime = 0; + + private int wave = 0; - private static final int NUM_WEAPON_TOWERS = 3; + private Timer waveTimer; private static final GridPoint2 PLAYER_SPAWN = new GridPoint2(2, 4); // Temporary spawn point for testing private static final float WALL_WIDTH = 0.1f; @@ -162,6 +170,7 @@ public class ForestGameArea extends GameArea { "images/mobs/rocky.atlas" }; private static final String[] forestSounds = { + "sounds/ui/Open_Close/NA_SFUI_Vol1_Open_01.ogg", "sounds/Impact4.ogg", "sounds/economy/click.wav", "sounds/economy/click_1.wav", @@ -202,16 +211,18 @@ public class ForestGameArea extends GameArea { "sounds/mobBoss/patrickThunder.mp3", "sounds/mobBoss/patrickHit.mp3", "sounds/mobBoss/spawnDemonSlime.mp3", + "sounds/towers/Desert-Eagle-Far-Single-Gunshot.mp3", + "sounds/towers/5.56_single_shot.mp3", + "sounds/towers/explosion.mp3", + "sounds/towers/eco_tower_ping.mp3", + "sounds/towers/ar15_single_shot_far.mp3", "sounds/mobs/skeletonHit.mp3", "sounds/mobs/coatAttack.mp3", "sounds/mobs/archerArrow.mp3" - }; private static final String BACKGROUND_MUSIC = "sounds/background/Sci-Fi1.ogg"; private static final String[] forestMusic = {BACKGROUND_MUSIC}; - private Entity player; - private Entity waves; /** * Initialise this ForestGameArea to use the provided TerrainFactory. @@ -221,19 +232,6 @@ public ForestGameArea() { super(); } - /** - * Add this method to start the wave spawning timer when the game starts. - */ -// private void startWaveTimer() { -// waveTimer = new Timer(); -// waveTimer.scheduleAtFixedRate(new TimerTask() { -// @Override -// public void run() { -// spawnWave(); -// } -// }, 0, 10000); // 10000 milliseconds = 10 seconds -// } - /** * Add this method to stop the wave timer when the game ends or as needed. */ @@ -258,40 +256,19 @@ public void create() { spawnTerrain(); // Set up infrastructure for end game tracking -// player = spawnPlayer(); - logger.info("Creating waves"); - waves = WaveFactory.createWaves(); + Entity waves = WaveFactory.createWaves(); spawnEntity(waves); waves.getEvents().addListener("spawnWave", this::spawnMob); - // spawnCoat(); -// spawnDodgingDragonKnight(17,4); -// spawnDeflectWizard(17, 3); -// spawnSplittingXenoGrunt(17, 2); -// spawnPatrick(); -// spawnDemonBoss(); - // spawnSplittingRocky(17, 4); - // spawnFireWizard(17, 3); - // spawnNecromancer(17, 2); - - - spawnScrap(); spawnGapScanners(); -// spawnTNTTower(); -// spawnWeaponTower(); -// spawnGapScanners(); -// spawnDroidTower(); -// spawnFireWorksTower(); // Commented these out until they are needed for Demonstration -// spawnPierceTower(); -// spawnRicochetTower(); -// spawnBombship(); } - + private void displayUI() { Entity ui = new Entity(); +// ui.addComponent(new GameAreaDisplay("Box Forest")); TODO: This should be the level name? ui.addComponent(ServiceLocator.getGameEndService().getDisplay()); ui.addComponent(ServiceLocator.getCurrencyService().getDisplay()); spawnEntity(ui); @@ -332,103 +309,6 @@ private void spawnTerrain() { } - private Entity spawnPlayer() { - Entity newPlayer = PlayerFactory.createPlayer(); - spawnEntityAt(newPlayer, PLAYER_SPAWN, true, true); - return newPlayer; - } - - // Spawn player at a specific position - private Entity spawnPlayer(GridPoint2 position) { - Entity newPlayer = PlayerFactory.createPlayer(); - spawnEntityAt(newPlayer, position, true, true); - return newPlayer; - } - - // commented 383 - 386 out as there was a missing arg? - private void spawnDemonBoss() { - Entity demon = MobBossFactory.createDemonBoss(5000); - spawnEntityAt(demon, new GridPoint2(19, 5), true, false); - } - - private void spawnPatrick() { - Entity patrick = MobBossFactory.createPatrickBoss(3000); - spawnEntityAt(patrick, new GridPoint2(18, 5), true, false); - } - - private void spawnPatrickDeath() { - Entity patrickDeath = MobBossFactory.patrickDead(); - spawnEntityAt(patrickDeath, new GridPoint2(18, 5), true, false); - } - // commented 398 - 401 out as there was a missing arg? -// private void spawnIceBaby() { -// Entity iceBaby = MobBossFactory.createIceBoss(); -// spawnEntityAt(iceBaby, new GridPoint2(19, 5), true, false); -// } - -// private void spawnDemonBoss() { -// Entity demon = MobBossFactory.createDemonBoss(); -// spawnEntityAt(demon, new GridPoint2(19, 5), true, false); -// } - -// private void spawnPatrick() { -// Entity patrick = MobBossFactory.createPatrickBoss(3000); -// spawnEntityAt(patrick, new GridPoint2(18, 5), true, false); -// } -// -// private void spawnPatrickDeath() { -// Entity patrickDeath = MobBossFactory.patrickDead(); -// spawnEntityAt(patrickDeath, new GridPoint2(18, 5), true, false); -// } -// -// private void spawnIceBaby() { -// Entity iceBaby = MobBossFactory.createIceBoss(); -// spawnEntityAt(iceBaby, new GridPoint2(19, 5), true, false); -// } - - /** - * Spawns a projectile that only heads towards the enemies in its lane. - * - * @param position The position of the Entity that's shooting the projectile. - * @param targetLayer The enemy layer of the "shooter". - * @param direction The direction the projectile should head towards. - * @param speed The speed of the projectiles. - */ - private void spawnProjectile(Vector2 position, short targetLayer, int direction, Vector2 speed) { - Entity Projectile = ProjectileFactory.createFireBall(targetLayer, new Vector2(direction, position.y), speed); - Projectile.setPosition(position); - spawnEntity(Projectile); - } - - /** - * Spawns a projectile specifically for general mobs/xenohunters - * - * @param position The position of the Entity that's shooting the projectile. - * @param targetLayer The enemy layer of the "shooter". - * @param direction The direction the projectile should head towards. - * @param speed The speed of the projectiles. - */ - private void spawnProjectileTest(Vector2 position, short targetLayer, int direction, Vector2 speed) { - Entity Projectile = ProjectileFactory.createEngineerBullet(targetLayer, new Vector2(direction, position.y), speed); - Projectile.setPosition(position); - spawnEntity(Projectile); - } - - /** - * Spawns a projectile to be used for multiple projectile function. - * - * @param position The position of the Entity that's shooting the projectile. - * @param targetLayer The enemy layer of the "shooter". - * @param space The space between the projectiles' destination. - * @param direction The direction the projectile should head towards. - * @param speed The speed of the projectiles. - */ - private void spawnProjectile(Vector2 position, short targetLayer, int space, int direction, Vector2 speed) { - Entity Projectile = ProjectileFactory.createFireBall(targetLayer, new Vector2(direction, position.y + space), speed); - Projectile.setPosition(position); - spawnEntity(Projectile); - } - /** * Spawn an entity on the map. Is called during a wave. Add cases here for each mob type * @param entity mob to be spawned @@ -438,6 +318,9 @@ private void spawnProjectile(Vector2 position, short targetLayer, int space, int public void spawnMob(String entity, GridPoint2 randomPos, int health) { Entity mob; switch (entity) { + case "Xeno": + mob = NPCFactory.createXenoGrunt(health); + break; case "SplittingWaterSlime": mob = NPCFactory.createSplittingWaterSlime(health); break; @@ -450,7 +333,7 @@ public void spawnMob(String entity, GridPoint2 randomPos, int health) { case "Skeleton": mob = NPCFactory.createSkeleton(health); break; - case "DeflectWizard": + case "Wizard": mob = NPCFactory.createWizard(health); break; case "WaterQueen": @@ -474,15 +357,6 @@ public void spawnMob(String entity, GridPoint2 randomPos, int health) { case "ArcaneArcher": mob = NPCFactory.createDodgingArcaneArcher(health); break; - case "SplittingRocky": - mob = NPCFactory.createSplittingRocky(health); - break; - case "Necromancer": - mob = NPCFactory.createNecromancer(health); - break; - case "DeflectFireWizard": - mob = NPCFactory.createDeflectFireWizard(health); - break; case "PatrickBoss": mob = MobBossFactory.createPatrickBoss(health); break; @@ -501,400 +375,6 @@ public void spawnMob(String entity, GridPoint2 randomPos, int health) { flashWarning(mob); } - // * TEMPORARY FOR TESTING -// private void spawnSplittingXenoGrunt(int x, int y) { -// GridPoint2 pos = new GridPoint2(x, y); -// Entity xenoGrunt = NPCFactory.createSplittingXenoGrunt(); -// xenoGrunt.setScale(1.5f, 1.5f); -// spawnEntityAt(xenoGrunt, pos, true, true); -// } - // * TEMPORARY FOR TESTING - private void spawnSplittingRocky(int x, int y) { - GridPoint2 pos = new GridPoint2(x, y); - Entity rocky = NPCFactory.createSplittingRocky(60); - rocky.setScale(1.5f, 1.5f); - spawnEntityAt(rocky, pos, true, true); - } - - private void spawnFireWizard(int x, int y) { - GridPoint2 pos = new GridPoint2(x, y); - Entity firewiz = NPCFactory.createDeflectFireWizard(60); - firewiz.setScale(1.5f, 1.5f); - spawnEntityAt(firewiz, pos, true, true); - } - - private void spawnNecromancer(int x, int y) { - GridPoint2 pos = new GridPoint2(x, y); - Entity Necromancer = NPCFactory.createNecromancer(60); - Necromancer.setScale(1.5f, 1.5f); - spawnEntityAt(Necromancer, pos, true, true); - } - - // * TEMPORARY FOR TESTING -// private void spawnDodgingDragonKnight(int x, int y) { -// GridPoint2 pos = new GridPoint2(x, y); -// Entity fireworm = NPCFactory.createDodgingDragonKnight(); -// fireworm.setScale(1.5f, 1.5f); -// spawnEntityAt(fireworm, pos, true, true); -// } -// -// // * TEMPORARY FOR TESTING -// private void spawnDeflectXenoGrunt(int x, int y) { -// GridPoint2 pos = new GridPoint2(x, y); -// Entity xenoGrunt = NPCFactory.createDeflectXenoGrunt(); -// xenoGrunt.setScale(1.5f, 1.5f); -// spawnEntityAt(xenoGrunt, pos, true, true); -// } -// -// private void spawnFireWorm() { -// int[] pickedLanes = random.ints(1, 7) -// .distinct().limit(5).toArray(); -// for (int i = 0; i < NUM_GRUNTS; i++) { -// GridPoint2 randomPos = new GridPoint2(19, pickedLanes[i]); -// Entity fireWorm = NPCFactory.createFireWorm(); -// fireWorm.setScale(1.5f, 1.5f); -// spawnEntityAt(fireWorm, randomPos, true, false); -// } -// } -// -// private void spawnSkeleton() { -// int[] pickedLanes = new Random().ints(1, 7) -// .distinct().limit(5).toArray(); -// for (int i = 0; i < NUM_GRUNTS; i++) { -// GridPoint2 randomPos = new GridPoint2(19, pickedLanes[i]); -// Entity skeleton = NPCFactory.createSkeleton(); -// skeleton.setScale(1.5f, 1.5f); -// spawnEntityAt(skeleton, randomPos, true, false); -// } -// } - -// private void spawnDragonKnight() { -// int[] pickedLanes = random.ints(1, 7) -// .distinct().limit(5).toArray(); -// for (int i = 0; i < NUM_GRUNTS; i++) { -// GridPoint2 randomPos = new GridPoint2(19, pickedLanes[i]); -// Entity fireWorm = NPCFactory.createDragonKnight(); -// fireWorm.setScale(1.5f, 1.5f); -// spawnEntityAt(fireWorm, randomPos, true, false); -// } -// } -// -// private void spawnWizard() { -// int[] pickedLanes = new Random().ints(1, 7) -// .distinct().limit(5).toArray(); -// for (int i = 0; i < NUM_GRUNTS; i++) { -// GridPoint2 randomPos = new GridPoint2(19, pickedLanes[i]); -// Entity wizard = NPCFactory.createWizard(); -// wizard.setScale(1.5f, 1.5f); -// spawnEntityAt(wizard, randomPos, true, false); -// } -// } -// -// private void spawnWaterQueen() { -// int[] pickedLanes = new Random().ints(1, 7) -// .distinct().limit(5).toArray(); -// for (int i = 0; i < NUM_GRUNTS; i++) { -// GridPoint2 randomPos = new GridPoint2(19, pickedLanes[i]); -// Entity waterQueen = NPCFactory.createWaterQueen(); -// waterQueen.setScale(1.5f, 1.5f); -// spawnEntityAt(waterQueen, randomPos, true, false); -// } -// } -// -// private void spawnWaterSlime() { -// int[] pickedLanes = new Random().ints(1, 7) -// .distinct().limit(5).toArray(); -// for (int i = 0; i < NUM_GRUNTS; i++) { -// GridPoint2 randomPos = new GridPoint2(19, pickedLanes[i]); -// Entity waterSlime = NPCFactory.createWaterSlime(); -// waterSlime.setScale(1.5f, 1.5f); -// spawnEntityAt(waterSlime, randomPos, true, false); -// } -// } - - // * TEMPORARY FOR TESTING - private void spawnDodgingDragonKnight(int x, int y) { - GridPoint2 pos = new GridPoint2(x, y); - Entity fireworm = NPCFactory.createDodgingDragonKnight(100); - fireworm.setScale(1.5f, 1.5f); - spawnEntityAt(fireworm, pos, true, true); - } -// -// // * TEMPORARY FOR TESTING - private void spawnDeflectWizard(int x, int y) { - GridPoint2 pos = new GridPoint2(x, y); - Entity xenoGrunt = NPCFactory.createDeflectWizard(100); - xenoGrunt.setScale(1.5f, 1.5f); - spawnEntityAt(xenoGrunt, pos, true, true); - } -// -// private void spawnFireWorm() { -// -// int[] pickedLanes = rand.ints(0, ServiceLocator.getMapService().getHeight() ) -// -// .distinct().limit(5).toArray(); -// for (int i = 0; i < NUM_GRUNTS; i++) { -// GridPoint2 randomPos = new GridPoint2(19, pickedLanes[i]); -// Entity fireWorm = NPCFactory.createFireWorm(); -// fireWorm.setScale(1.5f, 1.5f); -// spawnEntityAt(fireWorm, randomPos, true, false); -// } -// } -// -// private void spawnSkeleton() { -// -// int[] pickedLanes = new Random().ints(0, ServiceLocator.getMapService().getHeight() ) -// .distinct().limit(5).toArray(); -// for (int i = 0; i < NUM_GRUNTS; i++) { -// GridPoint2 randomPos = new GridPoint2(19, pickedLanes[i]); -// Entity skeleton = NPCFactory.createSkeleton(); -// skeleton.setScale(1.5f, 1.5f); -// spawnEntityAt(skeleton, randomPos, true, false); -// } -// } -// -// private void spawnDragonKnight() { -// -// int[] pickedLanes = rand.ints(0, ServiceLocator.getMapService().getHeight() ) -// -// .distinct().limit(5).toArray(); -// for (int i = 0; i < NUM_GRUNTS; i++) { -// GridPoint2 randomPos = new GridPoint2(19, pickedLanes[i]); -// Entity fireWorm = NPCFactory.createDodgingDragonKnight(); -// fireWorm.setScale(1.5f, 1.5f); -// spawnEntityAt(fireWorm, randomPos, true, false); -// } -// } -// -// private void spawnWizard() { -// -// int[] pickedLanes = rand.ints(0, ServiceLocator.getMapService().getHeight() ) -// -// .distinct().limit(5).toArray(); -// for (int i = 0; i < NUM_GRUNTS; i++) { -// GridPoint2 randomPos = new GridPoint2(19, pickedLanes[i]); -// Entity wizard = NPCFactory.createDeflectWizard(); -// wizard.setScale(1.5f, 1.5f); -// spawnEntityAt(wizard, randomPos, true, false); -// } -// } -// -// private void spawnWaterQueen() { -// -// int[] pickedLanes = new Random().ints(0, ServiceLocator.getMapService().getHeight() ) -// -// .distinct().limit(5).toArray(); -// for (int i = 0; i < NUM_GRUNTS; i++) { -// GridPoint2 randomPos = new GridPoint2(19, pickedLanes[i]); -// Entity waterQueen = NPCFactory.createWaterQueen(); -// waterQueen.setScale(1.5f, 1.5f); -// spawnEntityAt(waterQueen, randomPos, true, false); -// } -// } -// -// private void spawnWaterSlime() { -// -// int[] pickedLanes = new Random().ints(0, ServiceLocator.getMapService().getHeight() ) -// -// .distinct().limit(5).toArray(); -// for (int i = 0; i < NUM_GRUNTS; i++) { -// GridPoint2 randomPos = new GridPoint2(19, pickedLanes[i]); -// Entity waterSlime = NPCFactory.createSplittingWaterSlime(); -// waterSlime.setScale(1.5f, 1.5f); -// spawnEntityAt(waterSlime, randomPos, true, false); -// } -// } - - private void spawnCoat() { - Entity gregMob = NPCFactory.createCoat(100); - gregMob.setScale(1.5f, 1.5f); - spawnEntityAt(gregMob, new GridPoint2(17, 4), false, false); - } - - /** - * Creates multiple projectiles that travel simultaneous. They all have same - * the starting point but different destinations. - * - * @param position The position of the Entity that's shooting the projectile. - * @param targetLayer The enemy layer of the "shooter". - * @param direction The direction the projectile should head towards. - * @param space The space between the projectiles' destination. - * @param speed The speed of the projectiles. - * @param quantity The amount of projectiles to spawn. - */ - private void spawnMultiProjectile(Vector2 position, short targetLayer, int direction, int space, Vector2 speed, int quantity) { - int half = quantity / 2; - for (int i = 0; i < quantity; i++) { - spawnProjectile(position, targetLayer, space * half, direction, speed); - --half; - } - } - - /** - * Returns projectile that can do an area of effect damage - * - * @param position The position of the Entity that's shooting the projectile. - * @param targetLayer The enemy layer of the "shooter". - * @param direction The direction the projectile should head towards. - * @param speed The speed of the projectiles. - * @param effect Type of effect. - * @param aoe Whether it is an aoe projectile. - */ - private void spawnEffectProjectile(Vector2 position, short targetLayer, int direction, Vector2 speed, - ProjectileEffects effect, boolean aoe) { - Entity Projectile = ProjectileFactory.createEffectProjectile(targetLayer, new Vector2(direction, position.y), speed, effect, aoe); - Projectile.setPosition(position); - spawnEntity(Projectile); - } - - /** - * Spawns a pierce fireball. - * Pierce fireball can go through targetlayers without disappearing but damage - * will still be applied. - * - * @param position The position of the Entity that's shooting the projectile. - * @param targetLayer The enemy layer of the "shooter". - * @param direction The direction the projectile should head towards. - * @param speed The speed of the projectiles. - */ - private void spawnPierceFireBall(Vector2 position, short targetLayer, int direction, Vector2 speed) { - Entity projectile = ProjectileFactory.createPierceFireBall(targetLayer, new Vector2(direction, position.y), speed); - projectile.setPosition(position); - spawnEntity(projectile); - } - - /** - * Spawns a ricochet fireball - * Ricochet fireballs bounce off targets with a specified maximum count of 3 - * Possible extensions: Make the bounce count flexible with a param. - * - * @param position The position of the Entity that's shooting the projectile. - * @param targetLayer The enemy layer of the "shooter". - * @param direction The direction the projectile should head towards. - * @param speed The speed of the projectiles. - */ - private void spawnRicochetFireball(Vector2 position, short targetLayer, int direction, Vector2 speed) { - // Bounce count set to 0. - Entity projectile = ProjectileFactory.createRicochetFireball(targetLayer, new Vector2(direction, position.y), speed, 0); - projectile.setPosition(position); - spawnEntity(projectile); - } - - /** - * Spawns a split firework fireball. - * Splits into mini projectiles that spreads out after collision. - * - * @param position The position of the Entity that's shooting the projectile. - * @param targetLayer The enemy layer of the "shooter". - * @param direction The direction the projectile should towards. - * @param speed The speed of the projectiles. - * @param amount The amount of projectiles appearing after collision. - */ - private void spawnSplitFireWorksFireBall(Vector2 position, short targetLayer, int direction, Vector2 speed, int amount) { - Entity projectile = ProjectileFactory.createSplitFireWorksFireball(targetLayer, new Vector2(direction, position.y), speed, amount); - projectile.setPosition(position); - spawnEntity(projectile); - } - - private void spawnWeaponTower() { - GridPoint2 minPos = new GridPoint2(0, 0); - GridPoint2 maxPos = terrain.getMapBounds(0).sub(5, 1); - - for (int i = 0; i < NUM_WEAPON_TOWERS + 7 ; i++) { - GridPoint2 randomPos1 = RandomUtils.random(minPos, maxPos); - GridPoint2 randomPos2 = RandomUtils.random(minPos, maxPos); - Entity wallTower = TowerFactory.createWallTower(); - Entity fireTower = TowerFactory.createFireTower(); - Entity stunTower = TowerFactory.createStunTower(); - spawnEntityAt(fireTower, randomPos1, true, true); - spawnEntityAt(stunTower, randomPos2, true, true); - spawnEntityAt(wallTower, randomPos2, true, true); - } - } - - // * TEMPORARY FOR TESTING - private void spawnFireTowerAt(int x, int y) { - GridPoint2 pos = new GridPoint2(x, y); - Entity fireTower = TowerFactory.createFireTower(); - - spawnEntityAt(fireTower, pos, true, true); - } - private void spawnDroidTowerAt(int x, int y) { - GridPoint2 pos = new GridPoint2(x, y); - Entity droidTower = TowerFactory.createDroidTower(); - - spawnEntityAt(droidTower, pos, true, true); - } - - private void spawnTNTTower() { - GridPoint2 minPos = new GridPoint2(0, 0); - GridPoint2 maxPos = terrain.getMapBounds(0).sub(5, 1); - - for (int i = 0; i < NUM_WEAPON_TOWERS + 5; i++) { - GridPoint2 randomPos = RandomUtils.random(minPos, maxPos); - Entity weaponTower = TowerFactory.createTNTTower(); - spawnEntityAt(weaponTower, randomPos, true, true); - } - - } - private void spawnFireWorksTower() { - GridPoint2 minPos = new GridPoint2(0, 2); - GridPoint2 maxPos = terrain.getMapBounds(0).sub(1, 1); - - for (int i = 0; i < NUM_WEAPON_TOWERS; i++) { - GridPoint2 randomPos = RandomUtils.random(minPos, maxPos); - Entity FireWorksTower = TowerFactory.createFireworksTower(); - spawnEntityAt(FireWorksTower, randomPos, true, true); - } - - } - private void spawnPierceTower() { - GridPoint2 minPos = new GridPoint2(0, 2); - GridPoint2 maxPos = terrain.getMapBounds(0).sub(3, 3); - - for (int i = 0; i < NUM_WEAPON_TOWERS; i++) { - GridPoint2 randomPos = RandomUtils.random(minPos, maxPos); - Entity PierceTower = TowerFactory.createPierceTower(); - spawnEntityAt(PierceTower, randomPos, true, true); - } - - } - private void spawnRicochetTower() { - GridPoint2 minPos = new GridPoint2(0, 2); - GridPoint2 maxPos = terrain.getMapBounds(0).sub(0, 3); - - for (int i = 0; i < NUM_WEAPON_TOWERS; i++) { - GridPoint2 randomPos = RandomUtils.random(minPos, maxPos); - Entity RicochetTower = TowerFactory.createRicochetTower(); - spawnEntityAt(RicochetTower, randomPos, true, true); - } - - } - - private void spawnBombship() { - GridPoint2 minPos = new GridPoint2(0, 0); - GridPoint2 maxPos = terrain.getMapBounds(0).sub(5, 1); - Entity bombship = BombshipFactory.createBombship(); - spawnEntityAt(bombship, minPos, true, true); - } - - private void spawnDroidTower() { - GridPoint2 minPos = new GridPoint2(0, 0); - GridPoint2 maxPos = terrain.getMapBounds(0).sub(5, 1); - - for (int i = 0; i < NUM_WEAPON_TOWERS + 5; i++) { - GridPoint2 randomPos = RandomUtils.random(minPos, maxPos); - Entity weaponTower = TowerFactory.createDroidTower(); - spawnEntityAt(weaponTower, randomPos, true, false); - } - } - - private void playMusic() { - Music music = ServiceLocator.getResourceService().getAsset(BACKGROUND_MUSIC, Music.class); - music.setLooping(true); - music.setVolume(0.3f); - music.play(); - } private void loadAssets() { logger.debug("Loading assets"); @@ -943,26 +423,24 @@ private void spawnScrap() { } } - private void spawnIncome() { - GridPoint2 minPos = new GridPoint2(0, 0); - GridPoint2 maxPos = terrain.getMapBounds(0).sub(2, 2); - - for (int i = 0; i < 50; i++) { - GridPoint2 randomPos = RandomUtils.random(minPos, maxPos); - Entity towerfactory = TowerFactory.createIncomeTower(); - spawnEntityAt(towerfactory, randomPos, true, true); - } - } - /** * Creates the scanners (one per lane) that detect absence of towers and presence of mobs, * and trigger engineer spawning */ private void spawnGapScanners() { + GameTime gameTime = ServiceLocator.getTimeSource(); + long currSpawnTime = gameTime.getTime(); + + long diff = currSpawnTime - this.lastSpawnTime; + if (diff < ENGINEER_MIN_SPAWN_INTERVAL) { + return; + } + for (int i = 0; i < terrain.getMapBounds(0).y; i++) { Entity scanner = GapScannerFactory.createScanner(); spawnEntityAt(scanner, new GridPoint2(0, i), true, true); } + this.lastSpawnTime = currSpawnTime; } private void flashWarning(Entity mob) { diff --git a/source/core/src/main/com/csse3200/game/areas/terrain/TerrainComponent.java b/source/core/src/main/com/csse3200/game/areas/terrain/TerrainComponent.java index c6c67b642..e4706c022 100644 --- a/source/core/src/main/com/csse3200/game/areas/terrain/TerrainComponent.java +++ b/source/core/src/main/com/csse3200/game/areas/terrain/TerrainComponent.java @@ -16,8 +16,6 @@ import com.csse3200.game.services.CurrencyService; import com.csse3200.game.services.ResourceService; import com.csse3200.game.services.ServiceLocator; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; /** * Render a tiled terrain for a given tiled map and orientation. A terrain is a map of tiles that diff --git a/source/core/src/main/com/csse3200/game/components/EffectComponent.java b/source/core/src/main/com/csse3200/game/components/EffectComponent.java index 32b32e3d3..db110417e 100644 --- a/source/core/src/main/com/csse3200/game/components/EffectComponent.java +++ b/source/core/src/main/com/csse3200/game/components/EffectComponent.java @@ -54,9 +54,6 @@ public void update() { // apply slow effect if (mob) { - if (target == null) { - return; - } if (slowFlag) { isSlowed = true; Vector2 half_speed = new Vector2(defaultTargetSpeed.x / 2, defaultTargetSpeed.y / 2); @@ -85,9 +82,6 @@ public void update() { } targetPhysics.setSpeed(STUN_SPEED); } else if (isStunned) { - if (target == null) { - return; - } isStunned = false; targetPhysics.setSpeed(defaultTargetSpeed); } diff --git a/source/core/src/main/com/csse3200/game/components/EffectsComponent.java b/source/core/src/main/com/csse3200/game/components/EffectsComponent.java index 596bb71f2..d5bcf90af 100644 --- a/source/core/src/main/com/csse3200/game/components/EffectsComponent.java +++ b/source/core/src/main/com/csse3200/game/components/EffectsComponent.java @@ -2,20 +2,13 @@ import java.util.ArrayList; -import com.badlogic.gdx.math.Vector2; import com.badlogic.gdx.physics.box2d.Fixture; -import com.csse3200.game.ai.tasks.AITaskComponent; -import com.csse3200.game.components.tower.TowerUpgraderComponent; import com.csse3200.game.entities.Entity; import com.csse3200.game.physics.BodyUserData; import com.csse3200.game.physics.PhysicsLayer; import com.csse3200.game.physics.components.HitboxComponent; -import com.csse3200.game.physics.components.PhysicsMovementComponent; import com.csse3200.game.services.ServiceLocator; -import com.badlogic.gdx.utils.Timer; -import com.badlogic.gdx.utils.Timer.Task; - import com.badlogic.gdx.utils.Array; /** @@ -29,7 +22,6 @@ public class EffectsComponent extends Component { private final boolean aoe; private HitboxComponent hitboxComponent; private final short targetLayer; - private Array burnEntities = new Array<>(); private ArrayList stunnedEntities = new ArrayList<>(); /** @@ -102,6 +94,7 @@ public void applySingleEffect(ProjectileEffects effect, Entity targetEntity) { } effectComponent.applyEffect(effect, hostEntity, targetEntity); } + /** * Used for aoe projectiles to apply effects to all entities within the area of effect (radius). * @param effect effect to be applied to entities within radius @@ -136,132 +129,4 @@ public void applyAoeEffect(ProjectileEffects effect) { effectComponent.applyEffect(effect, hostEntity, targetEntity); } } - - /** - * Deals damage to target based on hosts' CombatStatsComponent - * @param target CombatStatsComponent of entity hit by projectile - * @param host CombatStatsComponent of projectile - */ - private void fireballEffect(CombatStatsComponent target, CombatStatsComponent host) { - target.hit(host); - } - -// /** -// * Applies 5 ticks of damage from hosts' CombatStatsComponent over 5 seconds -// * @param target CombatStatsComponent of entity hit by projectile -// * @param host CombatStatsComponent of projectile -// */ -// private void burnEffect(CombatStatsComponent target, CombatStatsComponent host) { -// // Ensure burn effects aren't applied multiple times by same projectile -// if (burnEntities.contains(target, false)) { -// return; -// } -// burnEntities.add(target); -// // Create a timer task to apply the effect repeatedly -// int numberOfTicks = 5; -// long delay = 1; -// Timer.schedule(new Timer.Task() { -// private int count = 0; -// -// @Override -// public void run() { -// if (count < numberOfTicks) { -// target.hit(host); -// count++; -// } else { -// // Ensure to cancel the task when it's done -// this.cancel(); -// } -// } -// }, delay, delay); -// } - - /** - * Applies slow effect to targetEntity. If entity is a mob, speed - * and firing rate will be slowed. If entity is a tower, firing rate - * will be slowed - * @param targetEntity Entity for slow effect to be applied to - */ - private void slowEffect(Entity targetEntity) { - boolean towerFlag = false; - boolean mobFlag = false; - - PhysicsMovementComponent targetPhysics = null; - float xSpeed = 0; - float ySpeed = 0; - - // Create a timer task to apply the effect repeatedly - if (PhysicsLayer.contains(PhysicsLayer.HUMANS, targetEntity.getComponent(HitboxComponent.class).getLayer())) { - // towers - towerFlag = true; - targetEntity.getEvents().trigger("upgradeTower", TowerUpgraderComponent.UPGRADE.FIRERATE, -30); - } else if (PhysicsLayer.contains(PhysicsLayer.NPC, targetEntity.getComponent(HitboxComponent.class).getLayer())) { - // mobs - mobFlag = true; - targetPhysics = targetEntity.getComponent(PhysicsMovementComponent.class); - if (targetPhysics == null) { - return; - } - - // Halve the mob speed - xSpeed = targetPhysics.getSpeed().x; - ySpeed = targetPhysics.getSpeed().y; - targetPhysics.setSpeed(new Vector2(xSpeed/2, ySpeed/2)); - } else { - return; - } - - // Reset speed - boolean finalTowerFlag = towerFlag; - boolean finalMobFlag = mobFlag; - PhysicsMovementComponent finalTargetPhysics = targetPhysics; - float finalXSpeed = xSpeed; - float finalYSpeed = ySpeed; - Timer.schedule(new Task() { - @Override - public void run() { - if (finalTowerFlag) { - targetEntity.getEvents().trigger("upgradeTower", TowerUpgraderComponent.UPGRADE.FIRERATE, 30); - } else if (finalMobFlag) { - finalTargetPhysics.setSpeed(new Vector2(finalXSpeed, finalYSpeed)); - } - } - }, 5); // 5 seconds delay - } - - /** - * Applies stun effect to a taget entity. - * @param targetEntity Entity for stun effect to be applied to. - */ - private void stunEffect(Entity targetEntity) { - CombatStatsComponent hostCombatStats = targetEntity.getComponent(CombatStatsComponent.class); - AITaskComponent taskComponent = targetEntity.getComponent(AITaskComponent.class); - - if (hostCombatStats == null || taskComponent == null) { - return; - } - - hostCombatStats.setBaseAttack(0); - - if (stunnedEntities.contains(targetEntity)) { - return; - } - - taskComponent.disposeAll(); - stunnedEntities.add(targetEntity); - - new java.util.Timer().schedule( - new java.util.TimerTask() { - @Override - public void run() { - taskComponent.restore(); - for (int i = 0; i < stunnedEntities.size(); i++) { - if (stunnedEntities.get(i).equals(targetEntity)) { - stunnedEntities.remove(stunnedEntities.get(i)); - } - } - this.cancel(); - } - }, 5000); - } } \ No newline at end of file diff --git a/source/core/src/main/com/csse3200/game/components/gamearea/CurrencyDisplay.java b/source/core/src/main/com/csse3200/game/components/gamearea/CurrencyDisplay.java index 7444dfc4f..b110de999 100644 --- a/source/core/src/main/com/csse3200/game/components/gamearea/CurrencyDisplay.java +++ b/source/core/src/main/com/csse3200/game/components/gamearea/CurrencyDisplay.java @@ -61,12 +61,12 @@ private void addActors() { table.add(crystalsTb).width(crystalsTb.getWidth() * 0.5f).height(crystalsTb.getHeight() * 0.5f); stage.addActor(table); - scrapsTb.setPosition(table.getX() - 200, Gdx.graphics.getHeight() - 205); - scrapsTb.addAction(new SequenceAction(Actions.moveTo(table.getX() + 20, Gdx.graphics.getHeight() - 205, + scrapsTb.setPosition(table.getX() - 200f, Gdx.graphics.getHeight() - 205f); + scrapsTb.addAction(new SequenceAction(Actions.moveTo(table.getX() + 20f, Gdx.graphics.getHeight() - 205f, 1f, Interpolation.fastSlow))); - crystalsTb.setPosition(table.getX() - 200, Gdx.graphics.getHeight() - 268); - crystalsTb.addAction(new SequenceAction(Actions.moveTo(table.getX() + 20, Gdx.graphics.getHeight() - 268, + crystalsTb.setPosition(table.getX() - 200f, Gdx.graphics.getHeight() - 268f); + crystalsTb.addAction(new SequenceAction(Actions.moveTo(table.getX() + 20f, Gdx.graphics.getHeight() - 268f, 1f, Interpolation.fastSlow))); } diff --git a/source/core/src/main/com/csse3200/game/components/gamearea/EngineerCountDisplay.java b/source/core/src/main/com/csse3200/game/components/gamearea/EngineerCountDisplay.java index 8111c92ae..42510a607 100644 --- a/source/core/src/main/com/csse3200/game/components/gamearea/EngineerCountDisplay.java +++ b/source/core/src/main/com/csse3200/game/components/gamearea/EngineerCountDisplay.java @@ -60,8 +60,8 @@ private void addActors() { stage.addActor(table); // Animate the engineer count label - engineerTb.setPosition(table.getX() - 200, Gdx.graphics.getHeight() - 145); - engineerTb.addAction(new SequenceAction(Actions.moveTo(table.getX() + 20, Gdx.graphics.getHeight() - 145, + engineerTb.setPosition(table.getX() - 200f, Gdx.graphics.getHeight() - 145f); + engineerTb.addAction(new SequenceAction(Actions.moveTo(table.getX() + 20f, Gdx.graphics.getHeight() - 145f, 1f, Interpolation.fastSlow))); } @@ -69,13 +69,15 @@ private void addActors() { * Updates the engineer count on the UI component */ public void updateCount() { - int currentCount = ServiceLocator.getGameEndService().getEngineerCount(); - String text = String.format("%d", currentCount); - engineerTb.getLabel().setText(text); - if (currentCount < ServiceLocator.getGameEndService().getThreshold()) { + if (engineerTb != null) { // fix for null pointer exception + int currentCount = ServiceLocator.getGameEndService().getEngineerCount(); + String text = String.format("%d", currentCount); + engineerTb.getLabel().setText(text); + if (currentCount < ServiceLocator.getGameEndService().getThreshold()) { // engineerTb.addAction(Actions.color(Color.RED, 0.5f, Interpolation.swingIn)); - engineerTb.addAction(Actions.forever(new SequenceAction(Actions.fadeOut(0.5f), - Actions.fadeIn(0.5f)))); + engineerTb.addAction(Actions.forever(new SequenceAction(Actions.fadeOut(0.5f), + Actions.fadeIn(0.5f)))); + } } } diff --git a/source/core/src/main/com/csse3200/game/components/gamearea/GameAreaDisplay.java b/source/core/src/main/com/csse3200/game/components/gamearea/GameAreaDisplay.java index 9a8149869..deac53d83 100644 --- a/source/core/src/main/com/csse3200/game/components/gamearea/GameAreaDisplay.java +++ b/source/core/src/main/com/csse3200/game/components/gamearea/GameAreaDisplay.java @@ -14,7 +14,6 @@ * Displays the name of the current game area. */ public class GameAreaDisplay extends UIComponent { - private static final Logger logger = LoggerFactory.getLogger(GameAreaDisplay.class); private static final String DEFAULT_STYLE = "default"; private String gameAreaName = ""; private Label title; @@ -27,7 +26,6 @@ public GameAreaDisplay(String gameAreaName) { public void create() { super.create(); addActors(); - final Skin skin = new Skin(); } public void render(float delta) { diff --git a/source/core/src/main/com/csse3200/game/components/maingame/LevelProgressBar.java b/source/core/src/main/com/csse3200/game/components/maingame/LevelProgressBar.java index d885beb87..44171b44a 100644 --- a/source/core/src/main/com/csse3200/game/components/maingame/LevelProgressBar.java +++ b/source/core/src/main/com/csse3200/game/components/maingame/LevelProgressBar.java @@ -1,6 +1,5 @@ package com.csse3200.game.components.maingame; -import com.badlogic.gdx.audio.Music; import com.badlogic.gdx.graphics.Color; import com.badlogic.gdx.graphics.Pixmap; import com.badlogic.gdx.graphics.Texture; diff --git a/source/core/src/main/com/csse3200/game/components/maingame/MainGameActions.java b/source/core/src/main/com/csse3200/game/components/maingame/MainGameActions.java index 7253f6d3b..bbdd88acc 100644 --- a/source/core/src/main/com/csse3200/game/components/maingame/MainGameActions.java +++ b/source/core/src/main/com/csse3200/game/components/maingame/MainGameActions.java @@ -21,7 +21,19 @@ public MainGameActions(GdxGame game) { public void create() { entity.getEvents().addListener("exit", this::onExit); entity.getEvents().addListener("lose", this::onLose); - //entity.getEvents().addListener("win", this::onWin); + entity.getEvents().addListener("win", this::WinningScreen); + entity.getEvents().addListener("Next Level", this::NextLevel); + } + + private void NextLevel() { + logger.info("Next level"); + game.setScreen(GdxGame.ScreenType.Next_Screen); + + } + + private void WinningScreen() { + logger.info("Uer Won the game"); + game.setScreen(GdxGame.ScreenType.Win_Screen); } /** @@ -35,6 +47,4 @@ private void onExit() { private void onLose() { game.setScreen(GdxGame.ScreenType.LOSING_SCREEN); } - -// private void onWin() { game.setScreen(GdxGame.ScreenType.WIN_SCREEN);} // TODO : Uncomment this once win screen implemented } diff --git a/source/core/src/main/com/csse3200/game/components/maingame/MainGameDisplay.java b/source/core/src/main/com/csse3200/game/components/maingame/MainGameDisplay.java index ec5c56489..bb0e32a50 100644 --- a/source/core/src/main/com/csse3200/game/components/maingame/MainGameDisplay.java +++ b/source/core/src/main/com/csse3200/game/components/maingame/MainGameDisplay.java @@ -239,9 +239,9 @@ private TextButton createPauseButton() { TextButton pauseBtn = ButtonFactory.createButton("Pause"); // Starting animation for pause button - pauseBtn.setPosition(Gdx.graphics.getWidth(), (float)(Gdx.graphics.getHeight() - 150)); - pauseBtn.addAction(new SequenceAction(Actions.moveTo(Gdx.graphics.getWidth() - 200, - Gdx.graphics.getHeight() - 150, 1f, Interpolation.fastSlow))); + pauseBtn.setPosition((float)Gdx.graphics.getWidth(), (Gdx.graphics.getHeight() - 150f)); + pauseBtn.addAction(new SequenceAction(Actions.moveTo(Gdx.graphics.getWidth() - 200f, + Gdx.graphics.getHeight() - 150f, 1f, Interpolation.fastSlow))); // Spawns a pause menu when the button is pressed. pauseBtn.addListener( diff --git a/source/core/src/main/com/csse3200/game/components/maingame/UIElementsDisplay.java b/source/core/src/main/com/csse3200/game/components/maingame/UIElementsDisplay.java index ecf3c9758..cd005fe8b 100644 --- a/source/core/src/main/com/csse3200/game/components/maingame/UIElementsDisplay.java +++ b/source/core/src/main/com/csse3200/game/components/maingame/UIElementsDisplay.java @@ -21,6 +21,7 @@ public class UIElementsDisplay extends UIComponent { private final Table buttonTable = new Table(); private TextButton remainingMobsButton; private TextButton timerButton; + private long time = 0; @Override public void create() { @@ -36,9 +37,9 @@ private void addActors() { remainingMobsButton = ButtonFactory.createButton("Mobs:" + ServiceLocator.getWaveService().getEnemyCount()); - remainingMobsButton.setPosition(Gdx.graphics.getWidth(), Gdx.graphics.getHeight() - 230); - remainingMobsButton.addAction(new SequenceAction(Actions.moveTo(Gdx.graphics.getWidth() - 217, - Gdx.graphics.getHeight() - 230, 1f, Interpolation.fastSlow))); + remainingMobsButton.setPosition(Gdx.graphics.getWidth(), Gdx.graphics.getHeight() - 230f); + remainingMobsButton.addAction(new SequenceAction(Actions.moveTo(Gdx.graphics.getWidth() - 217f, + Gdx.graphics.getHeight() - 230f, 1f, Interpolation.fastSlow))); buttonTable.top().right().padTop(160f).padRight(80f); buttonTable.setFillParent(true); @@ -73,9 +74,9 @@ public void createTimerButton() { timerButton = ButtonFactory.createButton("Next wave in:" + (ServiceLocator.getWaveService().getNextWaveTime() / 1000)); - timerButton.setPosition(Gdx.graphics.getWidth(), Gdx.graphics.getHeight() - 300); - timerButton.addAction(new SequenceAction(Actions.moveTo(Gdx.graphics.getWidth() - 435, - Gdx.graphics.getHeight() - 300, 1f, Interpolation.fastSlow))); + timerButton.setPosition(Gdx.graphics.getWidth(), Gdx.graphics.getHeight() - 300f); + timerButton.addAction(new SequenceAction(Actions.moveTo(Gdx.graphics.getWidth() - 435f, + Gdx.graphics.getHeight() - 300f, 1f, Interpolation.fastSlow))); timerButton.setDisabled(true); buttonTable.row(); buttonTable.add(timerButton); @@ -98,7 +99,12 @@ public void updateTimerButton() { createTimerButton(); } timerButton.setText("Next wave in: " + finalTime); + time = ServiceLocator.getTimeSource().getTime(); } else { + if (ServiceLocator.getTimeSource().getTime() < time + 2000) { + ServiceLocator.getMapService().shakeCameraMap(); + ServiceLocator.getMapService().shakeCameraGrid(); + } remainingMobsButton.setDisabled(true); timerButton.addAction(new SequenceAction(Actions.fadeOut(1f), Actions.removeActor())); stage.act(); diff --git a/source/core/src/main/com/csse3200/game/components/mainmenu/MainMenuActions.java b/source/core/src/main/com/csse3200/game/components/mainmenu/MainMenuActions.java index 2f57c0ecc..7562fd92c 100644 --- a/source/core/src/main/com/csse3200/game/components/mainmenu/MainMenuActions.java +++ b/source/core/src/main/com/csse3200/game/components/mainmenu/MainMenuActions.java @@ -2,7 +2,6 @@ import com.csse3200.game.GdxGame; import com.csse3200.game.components.Component; -import com.csse3200.game.screens.HelpScreen; import org.slf4j.Logger; import org.slf4j.LoggerFactory; diff --git a/source/core/src/main/com/csse3200/game/components/mainmenu/MainMenuDisplay.java b/source/core/src/main/com/csse3200/game/components/mainmenu/MainMenuDisplay.java index f0a67c1ff..2fdc6ddf1 100644 --- a/source/core/src/main/com/csse3200/game/components/mainmenu/MainMenuDisplay.java +++ b/source/core/src/main/com/csse3200/game/components/mainmenu/MainMenuDisplay.java @@ -53,14 +53,13 @@ private void addActors() { cursorPixmap.dispose(); // Dispose of the Pixmap to release resources table = new Table(); - table1 = new Table(); table.setFillParent(true); - table1.setFillParent(true); + stage.addActor(table); - Image title = - new Image( - ServiceLocator.getResourceService() - .getAsset("images/background/main_menu/main_menu_bg.png", Texture.class)); + Texture backgroundTexture = ServiceLocator.getResourceService().getAsset("images/background/main_menu/main_menu_bg.png", Texture.class); + Image title = new Image(backgroundTexture); + title.setFillParent(true); + table.addActorAt(0, title); title.setWidth(Gdx.graphics.getWidth()); title.setHeight(Gdx.graphics.getHeight()); title.setPosition(0, 0); @@ -126,18 +125,13 @@ public void changed(ChangeEvent changeEvent, Actor actor) { float padTopOtherBtns = 15f / originalScreenHeight * Gdx.graphics.getHeight(); - table.add(title); - table1.row(); - table1.add(startBtn).padTop(padTopStartBtn); - table1.row(); - table1.add(helpBtn).padTop(padTopOtherBtns); - table1.row(); - table1.add(settingsBtn).padTop(padTopOtherBtns); - table1.row(); - table1.add(exitBtn).padTop(padTopOtherBtns); + table.center(); + table.add(startBtn).padTop(250f).center().row(); + table.add(helpBtn).padTop(15f).center().row(); + table.add(settingsBtn).padTop(15f).center().row(); + table.add(exitBtn).padTop(15f).center().row(); stage.addActor(table); - stage.addActor(table1); } @Override diff --git a/source/core/src/main/com/csse3200/game/components/npc/ArcaneArcherAnimationController.java b/source/core/src/main/com/csse3200/game/components/npc/ArcaneArcherAnimationController.java index 71fa5359d..82c587f18 100644 --- a/source/core/src/main/com/csse3200/game/components/npc/ArcaneArcherAnimationController.java +++ b/source/core/src/main/com/csse3200/game/components/npc/ArcaneArcherAnimationController.java @@ -5,19 +5,14 @@ import com.csse3200.game.rendering.AnimationRenderComponent; import com.csse3200.game.services.ServiceLocator; -import java.security.SecureRandom; /** * This class listens to events relevant to a ghost entity's state and plays the animation when one * of the events is triggered. */ public class ArcaneArcherAnimationController extends Component { - // // For on collision sounds later - // private static final String COLLISION_SFX = "sounds/projectiles/on_collision.mp3"; - // Sound onCollisionSound = ServiceLocator.getResourceService().getAsset( - // COLLISION_SFX, Sound.class); + AnimationRenderComponent animator; - private SecureRandom rand = new SecureRandom(); private static final String ATTACK_SOUND = "sounds/mobs/archerArrow.mp3"; Sound attackSound = ServiceLocator.getResourceService().getAsset( diff --git a/source/core/src/main/com/csse3200/game/components/npc/CoatAnimationController.java b/source/core/src/main/com/csse3200/game/components/npc/CoatAnimationController.java index e9bf8170e..c023e0aa5 100644 --- a/source/core/src/main/com/csse3200/game/components/npc/CoatAnimationController.java +++ b/source/core/src/main/com/csse3200/game/components/npc/CoatAnimationController.java @@ -5,19 +5,13 @@ import com.csse3200.game.rendering.AnimationRenderComponent; import com.csse3200.game.services.ServiceLocator; -import java.security.SecureRandom; /** * This class listens to events relevant to a ghost entity's state and plays the animation when one * of the events is triggered. */ public class CoatAnimationController extends Component { - // // For on collision sounds later - // private static final String COLLISION_SFX = "sounds/projectiles/on_collision.mp3"; - // Sound onCollisionSound = ServiceLocator.getResourceService().getAsset( - // COLLISION_SFX, Sound.class); AnimationRenderComponent animator; - private SecureRandom rand = new SecureRandom(); private static final String ATTACK_SOUND = "sounds/mobs/coatAttack.mp3"; Sound attackSound = ServiceLocator.getResourceService().getAsset( diff --git a/source/core/src/main/com/csse3200/game/components/npc/DeflectingComponent.java b/source/core/src/main/com/csse3200/game/components/npc/DeflectingComponent.java index ab5aa645e..15fe60fcd 100644 --- a/source/core/src/main/com/csse3200/game/components/npc/DeflectingComponent.java +++ b/source/core/src/main/com/csse3200/game/components/npc/DeflectingComponent.java @@ -60,7 +60,7 @@ public void create() { @Override public void update() { - resetHealth(); + super.update(); } /** @@ -77,8 +77,6 @@ private void deflectProj(Fixture me, Fixture other) { if (deflectLimitAmount-- <= 0) { // Reached deflect limit amt, return. entity.getComponent(this.getClass()).setEnabled(false); - // reset health - resetHealth(); return; } diff --git a/source/core/src/main/com/csse3200/game/components/npc/DodgingComponent.java b/source/core/src/main/com/csse3200/game/components/npc/DodgingComponent.java index 14a3b07d9..26aadd4d4 100644 --- a/source/core/src/main/com/csse3200/game/components/npc/DodgingComponent.java +++ b/source/core/src/main/com/csse3200/game/components/npc/DodgingComponent.java @@ -1,5 +1,6 @@ package com.csse3200.game.components.npc; +import com.badlogic.gdx.math.MathUtils; import com.badlogic.gdx.math.Vector2; import com.csse3200.game.components.Component; import com.csse3200.game.physics.PhysicsEngine; @@ -7,6 +8,8 @@ import com.csse3200.game.physics.raycast.RaycastHit; import com.csse3200.game.services.ServiceLocator; +import java.util.Random; + /** * A component that adds a dodging event listener to the current attached * entity. The entity will dodge another entity that is presumably coming at its @@ -27,7 +30,7 @@ public class DodgingComponent extends Component { private final RaycastHit hit = new RaycastHit(); private short targetLayer; private float rangeDetection; - private float dodgeSpeed = 1.75f; + private float dodgeSpeed = 1.75f; private float originalSpeed; // Original entity vertical speed private PhysicsEngine physics; @@ -92,11 +95,15 @@ public void create() { * @param mobPos The current Vector2 mob position in the map. */ public void changeTraverseDirection(Vector2 mobPos) { + int randDirection = MathUtils.random(0,2) == 1 ? -1 : 1; + if (isTargetVisible(mobPos)) { // If mob is in the top half quadrant of the map grid, make the entity dodge // downwards. - setVerticalAngleDirection(mobPos.y > 3.5 ? mobPos.y - 15 : mobPos.y + 15); - setVerticalSpeed(dodgeSpeed); + // setVerticalAngleDirection(mobPos.y > 3.5 ? mobPos.y - 15 : mobPos.y + 15); + // Random direction + setVerticalAngleDirection(mobPos.y + (15 * randDirection)); + setVerticalSpeed(dodgeSpeed); } else { setVerticalAngleDirection(mobPos.y); setVerticalSpeed(originalSpeed); @@ -110,7 +117,7 @@ public void changeTraverseDirection(Vector2 mobPos) { * @param mobPos The current Vector2 position of the mob * @return True if a target is visible, false otherwise. */ - private boolean isTargetVisible(Vector2 mobPos) { + public boolean isTargetVisible(Vector2 mobPos) { Vector2 maxRange = new Vector2(mobPos.x - rangeDetection, mobPos.y); // check also the upper and lower boundaries of the mob with the offset y mob // detection. diff --git a/source/core/src/main/com/csse3200/game/components/npc/DragonKnightAnimationController.java b/source/core/src/main/com/csse3200/game/components/npc/DragonKnightAnimationController.java index 6d9d54215..7302a1cdc 100644 --- a/source/core/src/main/com/csse3200/game/components/npc/DragonKnightAnimationController.java +++ b/source/core/src/main/com/csse3200/game/components/npc/DragonKnightAnimationController.java @@ -2,19 +2,13 @@ import com.csse3200.game.components.Component; import com.csse3200.game.rendering.AnimationRenderComponent; -import java.security.SecureRandom; /** * This class listens to events relevant to a ghost entity's state and plays the animation when one * of the events is triggered. */ public class DragonKnightAnimationController extends Component { - // // For on collision sounds later - // private static final String COLLISION_SFX = "sounds/projectiles/on_collision.mp3"; - // Sound onCollisionSound = ServiceLocator.getResourceService().getAsset( - // COLLISION_SFX, Sound.class); AnimationRenderComponent animator; - private SecureRandom rand = new SecureRandom(); @Override public void create() { diff --git a/source/core/src/main/com/csse3200/game/components/npc/EngineerMenuComponent.java b/source/core/src/main/com/csse3200/game/components/npc/EngineerMenuComponent.java deleted file mode 100644 index 096fb5962..000000000 --- a/source/core/src/main/com/csse3200/game/components/npc/EngineerMenuComponent.java +++ /dev/null @@ -1,150 +0,0 @@ -package com.csse3200.game.components.npc; - -import com.badlogic.gdx.graphics.Camera; -import com.badlogic.gdx.graphics.Texture; -import com.badlogic.gdx.graphics.g2d.BitmapFont; -import com.badlogic.gdx.graphics.g2d.SpriteBatch; -import com.badlogic.gdx.graphics.g2d.TextureRegion; -import com.badlogic.gdx.math.Vector2; -import com.badlogic.gdx.math.Vector3; -import com.badlogic.gdx.scenes.scene2d.InputEvent; -import com.badlogic.gdx.scenes.scene2d.ui.Table; -import com.badlogic.gdx.scenes.scene2d.ui.TextButton; -import com.badlogic.gdx.scenes.scene2d.utils.ClickListener; -import com.badlogic.gdx.scenes.scene2d.utils.Drawable; -import com.badlogic.gdx.scenes.scene2d.utils.TextureRegionDrawable; -import com.badlogic.gdx.utils.Array; -import com.csse3200.game.components.player.HumanAnimationController; -import com.csse3200.game.components.tower.TowerUpgraderComponent; -import com.csse3200.game.entities.Entity; -import com.csse3200.game.entities.EntityService; -import com.csse3200.game.physics.PhysicsLayer; -import com.csse3200.game.input.EngineerInputComponent; -import com.csse3200.game.rendering.AnimationRenderComponent; -import com.csse3200.game.services.ServiceLocator; -import com.csse3200.game.ui.UIComponent; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -public class EngineerMenuComponent extends UIComponent { - private Logger logger = LoggerFactory.getLogger(EngineerMenuComponent.class); - Table table; - - @Override - public void create() { - super.create(); - } - - @Override - public void draw(SpriteBatch batch) { - // draw is handled by the stage - } - - /** - * Creates a menu for the engineer - * @param x cursor x coordinate - * @param y cursor y coordinate - * @param camera camera of the game - */ - public void createMenu(float x, float y, Camera camera) { - this.table = createTable(x, y, camera); - - // add buttons - TextButton moveButton = createButton("Move"); - TextButton repairButton = createButton("Repair"); - - // add listeners to buttons - AnimationRenderComponent animator = getEntity().getComponent(AnimationRenderComponent.class); - HumanAnimationController controller = getEntity().getComponent(HumanAnimationController.class); - EngineerInputComponent input = ServiceLocator.getInputService().getEngineerInput(); - - moveButton.addListener(new ClickListener() { - @Override - public void clicked(InputEvent event, float x, float y) { - input.setMoveClicked(true); - controller.deselectEngineer(animator.getCurrentAnimation()); - - //logger.info("Move button clicked"); - } - }); - - repairButton.addListener(new ClickListener() { - @Override - public void clicked(InputEvent event, float x, float y) { - controller.deselectEngineer(animator.getCurrentAnimation()); - EntityService entityService = ServiceLocator.getEntityService(); - Array tower = entityService.getEntitiesInLayer(getEntity(), 0.1f, PhysicsLayer.TOWER); - if (tower.size == 0) { - logger.info("No tower to repair"); - return; - } - logger.info("repairing"); - tower.get(0).getComponent(TowerUpgraderComponent.class).repairTower(); - //logger.info("Repair button clicked"); - } - }); - - table.add(moveButton).grow(); - table.row(); - table.add(repairButton).grow(); - table.row(); - stage.addActor(table); - - } - - /** - * Creates a table for the menu - * @param x cursor x coordinate - * @param y cursor y coordinate - * @param camera camera of the game - * @return table for the menu - */ - private Table createTable(float x, float y, Camera camera) { - Table table = new Table(); - table.top(); - table.defaults().pad(0).space(0); - table.setSize(90, 60); // fixed table size - - // convert cursor position to stage coordinates - Vector3 entityCoordinates = new Vector3(x, y, 0); - Vector3 entityScreenCoordinate = camera.project(entityCoordinates); - Vector2 stageCoordinates = stage.screenToStageCoordinates( - new Vector2(entityScreenCoordinate.x, entityScreenCoordinate.y)); - stage.getViewport().unproject(stageCoordinates); - table.setPosition(stageCoordinates.x, stageCoordinates.y); - - // set table background - String imageFilePath = "images/ui/Sprites/UI_Glass_Frame_Standard_01a.png"; - Drawable drawable = new TextureRegionDrawable(new TextureRegion(new Texture(imageFilePath))); - table.setBackground(drawable); - - return table; - } - - /** - * Creates a button for the menu - * @param text text to be displayed on the button - * @return the button - */ - private TextButton createButton(String text) { - String upImageFilePath = "images/ui/Sprites/UI_Glass_Button_Medium_Lock_01a2.png"; - String downImageFilePath = "images/ui/Sprites/UI_Glass_Button_Medium_Press_01a2.png"; - - Drawable upDrawable = new TextureRegionDrawable(new TextureRegion(new Texture(upImageFilePath))); - Drawable downDrawable = new TextureRegionDrawable(new TextureRegion(new Texture(downImageFilePath))); - TextButton button = new TextButton(text, - new TextButton.TextButtonStyle(upDrawable, downDrawable, null, new BitmapFont())); - - button.setTransform(true); - return button; - } - - /** - * Removes the menu from the stage - */ - public void removeMenu() { - table.clear(); - table.remove(); - } - -} diff --git a/source/core/src/main/com/csse3200/game/components/npc/FireWormAnimationController.java b/source/core/src/main/com/csse3200/game/components/npc/FireWormAnimationController.java index 141d43d69..28d118210 100644 --- a/source/core/src/main/com/csse3200/game/components/npc/FireWormAnimationController.java +++ b/source/core/src/main/com/csse3200/game/components/npc/FireWormAnimationController.java @@ -4,19 +4,13 @@ import com.csse3200.game.components.Component; import com.csse3200.game.rendering.AnimationRenderComponent; import com.csse3200.game.services.ServiceLocator; -import java.security.SecureRandom; /** * This class listens to events relevant to a ghost entity's state and plays the animation when one * of the events is triggered. */ public class FireWormAnimationController extends Component { - // // For on collision sounds later - // private static final String COLLISION_SFX = "sounds/projectiles/on_collision.mp3"; - // Sound onCollisionSound = ServiceLocator.getResourceService().getAsset( - // COLLISION_SFX, Sound.class); AnimationRenderComponent animator; - private SecureRandom rand = new SecureRandom(); private static final String ATTACK_SOUND = "sounds/mobs/fireWormRoar.mp3"; Sound attackSound = ServiceLocator.getResourceService().getAsset( diff --git a/source/core/src/main/com/csse3200/game/components/npc/FirewizardAnimationController.java b/source/core/src/main/com/csse3200/game/components/npc/FirewizardAnimationController.java index 267c23404..b10236761 100644 --- a/source/core/src/main/com/csse3200/game/components/npc/FirewizardAnimationController.java +++ b/source/core/src/main/com/csse3200/game/components/npc/FirewizardAnimationController.java @@ -3,12 +3,9 @@ import com.csse3200.game.components.Component; import com.csse3200.game.rendering.AnimationRenderComponent; -import java.security.SecureRandom; - public class FirewizardAnimationController extends Component { AnimationRenderComponent animator; - private SecureRandom rand = new SecureRandom(); @Override diff --git a/source/core/src/main/com/csse3200/game/components/npc/NecromancerAnimationController.java b/source/core/src/main/com/csse3200/game/components/npc/NecromancerAnimationController.java index 3bdad9407..dfbd4bc9a 100644 --- a/source/core/src/main/com/csse3200/game/components/npc/NecromancerAnimationController.java +++ b/source/core/src/main/com/csse3200/game/components/npc/NecromancerAnimationController.java @@ -3,13 +3,9 @@ import com.csse3200.game.components.Component; import com.csse3200.game.rendering.AnimationRenderComponent; -import java.security.SecureRandom; - public class NecromancerAnimationController extends Component { AnimationRenderComponent animator; - private SecureRandom rand = new SecureRandom(); - @Override public void create() { diff --git a/source/core/src/main/com/csse3200/game/components/npc/NightBorneAnimationController.java b/source/core/src/main/com/csse3200/game/components/npc/NightBorneAnimationController.java index d821f6cca..fccdfe580 100644 --- a/source/core/src/main/com/csse3200/game/components/npc/NightBorneAnimationController.java +++ b/source/core/src/main/com/csse3200/game/components/npc/NightBorneAnimationController.java @@ -2,7 +2,6 @@ import com.csse3200.game.components.Component; import com.csse3200.game.rendering.AnimationRenderComponent; -import java.security.SecureRandom; /** * This class listens to events relevant to a ghost entity's state and plays the animation when one @@ -13,9 +12,7 @@ public class NightBorneAnimationController extends Component { // private static final String COLLISION_SFX = "sounds/projectiles/on_collision.mp3"; // Sound onCollisionSound = ServiceLocator.getResourceService().getAsset( // COLLISION_SFX, Sound.class); - AnimationRenderComponent animator; - private SecureRandom rand = new SecureRandom(); - + AnimationRenderComponent animator; @Override public void create() { diff --git a/source/core/src/main/com/csse3200/game/components/npc/RockyAnimationController.java b/source/core/src/main/com/csse3200/game/components/npc/RockyAnimationController.java index f6743dd9d..63f850085 100644 --- a/source/core/src/main/com/csse3200/game/components/npc/RockyAnimationController.java +++ b/source/core/src/main/com/csse3200/game/components/npc/RockyAnimationController.java @@ -3,11 +3,8 @@ import com.csse3200.game.components.Component; import com.csse3200.game.rendering.AnimationRenderComponent; -import java.security.SecureRandom; - public class RockyAnimationController extends Component { AnimationRenderComponent animator; - private SecureRandom rand = new SecureRandom(); @Override diff --git a/source/core/src/main/com/csse3200/game/components/npc/SkeletonAnimationController.java b/source/core/src/main/com/csse3200/game/components/npc/SkeletonAnimationController.java index d5ea57bfa..7a821ef75 100644 --- a/source/core/src/main/com/csse3200/game/components/npc/SkeletonAnimationController.java +++ b/source/core/src/main/com/csse3200/game/components/npc/SkeletonAnimationController.java @@ -4,19 +4,13 @@ import com.csse3200.game.components.Component; import com.csse3200.game.rendering.AnimationRenderComponent; import com.csse3200.game.services.ServiceLocator; -import java.security.SecureRandom; /** * This class listens to events relevant to a ghost entity's state and plays the animation when one * of the events is triggered. */ public class SkeletonAnimationController extends Component { - // // For on collision sounds later - // private static final String COLLISION_SFX = "sounds/projectiles/on_collision.mp3"; - // Sound onCollisionSound = ServiceLocator.getResourceService().getAsset( - // COLLISION_SFX, Sound.class); AnimationRenderComponent animator; - private SecureRandom rand = new SecureRandom(); /** Sound variables */ private static final String ATTACK_SOUND = "sounds/mobs/boneBreak.mp3"; diff --git a/source/core/src/main/com/csse3200/game/components/npc/SplitMoblings.java b/source/core/src/main/com/csse3200/game/components/npc/SplitMoblings.java index 3fd20bb9d..6b92dd1cf 100644 --- a/source/core/src/main/com/csse3200/game/components/npc/SplitMoblings.java +++ b/source/core/src/main/com/csse3200/game/components/npc/SplitMoblings.java @@ -97,7 +97,7 @@ private void onDeath() { // left. if (amount == 1) { float newXPosition = (float) (entity.getPosition().x - OFFSET_DISTANCE); - float newYPosition = (float) (entity.getPosition().y); + float newYPosition = (entity.getPosition().y); if (withinBounds(newXPosition, newYPosition)) { spawnAdditionalMob(newXPosition, newYPosition, initialScaleX, initialScaleY); @@ -133,37 +133,31 @@ private void onDeath() { */ public void spawnAdditionalMob(float positionX, float positionY, float initialScaleX, float initialScaleY) { - // Entity waterSlime = NPCFactory.createBaseWaterSlime(60); Entity entityType; switch (mobType) { - case WATER_SLIME -> { + case WATER_SLIME -> entityType = NPCFactory.createBaseWaterSlime(baseMoblingHealth); - } - case NIGHT_BORNE -> { + case NIGHT_BORNE -> entityType = NPCFactory.createNightBorne(baseMoblingHealth); - } - case ROCKY -> { + case ROCKY -> entityType = NPCFactory.createRocky(baseMoblingHealth); - } - default -> { + default -> entityType = NPCFactory.createBaseWaterSlime(baseMoblingHealth); - } } - + entityType.setPosition(positionX, positionY); switch (mobType) { - case NIGHT_BORNE -> { + case NIGHT_BORNE -> entityType.setScale(initialScaleX, initialScaleY); - } - default -> { + + default -> entityType.setScale(initialScaleX * scaleX, initialScaleY * scaleY); - } - } + } ServiceLocator.getEntityService().register(entityType); @@ -180,12 +174,9 @@ public void spawnAdditionalMob(float positionX, float positionY, * False otherwise. */ private boolean withinBounds(float currX, float currY) { - if (currX >= MIN_X_BOUNDS + return currX >= MIN_X_BOUNDS && currX <= MAX_X_BOUNDS && currY >= MIN_Y_BOUNDS - && currY <= MAX_Y_BOUNDS) { - return true; - } - return false; + && currY <= MAX_Y_BOUNDS; } } diff --git a/source/core/src/main/com/csse3200/game/components/npc/WaterQueenAnimationController.java b/source/core/src/main/com/csse3200/game/components/npc/WaterQueenAnimationController.java index 0b8a00b75..294f312b1 100644 --- a/source/core/src/main/com/csse3200/game/components/npc/WaterQueenAnimationController.java +++ b/source/core/src/main/com/csse3200/game/components/npc/WaterQueenAnimationController.java @@ -4,19 +4,13 @@ import com.csse3200.game.components.Component; import com.csse3200.game.rendering.AnimationRenderComponent; import com.csse3200.game.services.ServiceLocator; -import java.security.SecureRandom; /** * This class listens to events relevant to a ghost entity's state and plays the animation when one * of the events is triggered. */ public class WaterQueenAnimationController extends Component { - // // For on collision sounds later - // private static final String COLLISION_SFX = "sounds/projectiles/on_collision.mp3"; - // Sound onCollisionSound = ServiceLocator.getResourceService().getAsset( - // COLLISION_SFX, Sound.class); AnimationRenderComponent animator; - private SecureRandom rand = new SecureRandom(); private static final String ATTACK_SOUND = "sounds/mobs/waterQueenSpell.mp3"; Sound attackSound = ServiceLocator.getResourceService().getAsset( diff --git a/source/core/src/main/com/csse3200/game/components/npc/WaterSlimeAnimationController.java b/source/core/src/main/com/csse3200/game/components/npc/WaterSlimeAnimationController.java index 6909dad10..52163ddcb 100644 --- a/source/core/src/main/com/csse3200/game/components/npc/WaterSlimeAnimationController.java +++ b/source/core/src/main/com/csse3200/game/components/npc/WaterSlimeAnimationController.java @@ -2,19 +2,13 @@ import com.csse3200.game.components.Component; import com.csse3200.game.rendering.AnimationRenderComponent; -import java.security.SecureRandom; /** * This class listens to events relevant to a ghost entity's state and plays the animation when one * of the events is triggered. */ public class WaterSlimeAnimationController extends Component { - // // For on collision sounds later - // private static final String COLLISION_SFX = "sounds/projectiles/on_collision.mp3"; - // Sound onCollisionSound = ServiceLocator.getResourceService().getAsset( - // COLLISION_SFX, Sound.class); AnimationRenderComponent animator; - private SecureRandom rand = new SecureRandom(); @Override public void create() { diff --git a/source/core/src/main/com/csse3200/game/components/npc/WizardAnimationController.java b/source/core/src/main/com/csse3200/game/components/npc/WizardAnimationController.java index c8c8e7073..9b4660afb 100644 --- a/source/core/src/main/com/csse3200/game/components/npc/WizardAnimationController.java +++ b/source/core/src/main/com/csse3200/game/components/npc/WizardAnimationController.java @@ -4,19 +4,13 @@ import com.csse3200.game.components.Component; import com.csse3200.game.rendering.AnimationRenderComponent; import com.csse3200.game.services.ServiceLocator; -import java.security.SecureRandom; /** * This class listens to events relevant to a ghost entity's state and plays the animation when one * of the events is triggered. */ public class WizardAnimationController extends Component { - // // For on collision sounds later - // private static final String COLLISION_SFX = "sounds/projectiles/on_collision.mp3"; - // Sound onCollisionSound = ServiceLocator.getResourceService().getAsset( - // COLLISION_SFX, Sound.class); AnimationRenderComponent animator; - private SecureRandom rand = new SecureRandom(); /** Sound variables */ private static final String ATTACK_SOUND = "sounds/mobs/wizardSpell.mp3"; @@ -31,8 +25,6 @@ public void create() { entity.getEvents().addListener("mob_walk", this::animateWalk); entity.getEvents().addListener("mob_attack", this::animateAttack); entity.getEvents().addListener("mob_death", this::animateDeath); - - } void animateWalk() { diff --git a/source/core/src/main/com/csse3200/game/components/player/HumanAnimationController.java b/source/core/src/main/com/csse3200/game/components/player/HumanAnimationController.java index bcaf10f1f..5317c6b4b 100644 --- a/source/core/src/main/com/csse3200/game/components/player/HumanAnimationController.java +++ b/source/core/src/main/com/csse3200/game/components/player/HumanAnimationController.java @@ -2,7 +2,6 @@ import com.badlogic.gdx.audio.Sound; import com.csse3200.game.components.Component; -import com.csse3200.game.components.npc.EngineerMenuComponent; import com.csse3200.game.rendering.AnimationRenderComponent; import com.csse3200.game.services.ServiceLocator; @@ -108,7 +107,6 @@ void animateLeftWalk() { } else { animator.startAnimation(WALKL_ANIM); } -// runSound.play(); } /** @@ -206,18 +204,4 @@ public boolean isClicked() { public void setClicked(boolean clicked) { this.clicked = clicked; } - - /** - * Deseelects the engineer entity by starting the appropriate animation without an outline - * and removes the engineer menu from the stage - * @param currentAnimation the current animation of the entity - */ - public void deselectEngineer(String currentAnimation) { - AnimationRenderComponent animator = this.entity.getComponent(AnimationRenderComponent.class); - EngineerMenuComponent menu = this.entity.getComponent(EngineerMenuComponent.class); - - animator.startAnimation(currentAnimation.substring(0, currentAnimation.lastIndexOf('_'))); - menu.removeMenu(); - setClicked(false); - } } \ No newline at end of file diff --git a/source/core/src/main/com/csse3200/game/components/tasks/CurrencyTask.java b/source/core/src/main/com/csse3200/game/components/tasks/CurrencyTask.java index a20c7464f..19f7fa624 100644 --- a/source/core/src/main/com/csse3200/game/components/tasks/CurrencyTask.java +++ b/source/core/src/main/com/csse3200/game/components/tasks/CurrencyTask.java @@ -21,7 +21,7 @@ public class CurrencyTask extends DefaultTask implements PriorityTask { private long endTime; private int interval; private final Scrap scrap = new Scrap(); // currency to update - private final int currencyAmount = 30; // amount of currency to update + private final int currencyAmount = 10; // amount of currency to update private static final String IDLE = "idleStartEco"; private static final String MOVE = "moveStartEco"; private static final String DEATH = "deathStartEco"; diff --git a/source/core/src/main/com/csse3200/game/components/tasks/DroidCombatTask.java b/source/core/src/main/com/csse3200/game/components/tasks/DroidCombatTask.java index b3c15e9c2..3d33e15a2 100644 --- a/source/core/src/main/com/csse3200/game/components/tasks/DroidCombatTask.java +++ b/source/core/src/main/com/csse3200/game/components/tasks/DroidCombatTask.java @@ -105,27 +105,13 @@ public void updateTowerState() { } switch (towerState) { - case WALK -> { - handleWalkState(); - } - case IDLE -> { - handleIdleState(); - } - case SHOOT_DOWN -> { - handleShootDownState(); - } - case SHOOT_UP -> { - handleShootUpState(); - } - case DOWN -> { - handleDownState(); - } - case UP -> { - handleUpState(); - } - default -> { // DIE - handleDieState(); - } + case WALK -> handleWalkState(); + case IDLE -> handleIdleState(); + case SHOOT_DOWN -> handleShootDownState(); + case SHOOT_UP -> handleShootUpState(); + case DOWN -> handleDownState(); + case UP -> handleUpState(); + default -> handleDieState(); // DIE } } @@ -138,6 +124,10 @@ public STATE getState() { return this.towerState; } + /** + * Function for setting the tower's state. + * @param state The new state of this tower. + */ public void setState(STATE state) { this.towerState = state; } @@ -157,7 +147,9 @@ public int getPriority() { */ public boolean isTargetVisible() { // If there is an obstacle in the path to the max range point, mobs visible. - return physics.raycast(towerPosition, maxRangePosition, TARGET, hit); + boolean top = physics.raycast(towerPosition.add(0f,0.4f), maxRangePosition.add(0f,0.4f), TARGET, hit); + boolean bottom = physics.raycast(towerPosition.sub(0f,0.4f), maxRangePosition.sub(0f,0.4f), TARGET, hit); + return top || bottom; } private void changeFireRateInterval(int newInterval) { diff --git a/source/core/src/main/com/csse3200/game/components/tasks/FireTowerCombatTask.java b/source/core/src/main/com/csse3200/game/components/tasks/FireTowerCombatTask.java index 60714c560..5654f3b15 100644 --- a/source/core/src/main/com/csse3200/game/components/tasks/FireTowerCombatTask.java +++ b/source/core/src/main/com/csse3200/game/components/tasks/FireTowerCombatTask.java @@ -101,18 +101,10 @@ public void updateTowerState() { } switch (towerState) { - case IDLE -> { - handleIdleState(); - } - case PREP_ATTACK -> { - handlePrepAttackState(); - } - case ATTACK -> { - handleAttackState(); - } - default -> { // DEATH - handleDeathState(); - } + case IDLE -> handleIdleState(); + case PREP_ATTACK -> handlePrepAttackState(); + case ATTACK -> handleAttackState(); + default -> handleDeathState(); // DEATH } } @@ -140,43 +132,20 @@ public int getPriority() { return !isTargetVisible() ? 0 : priority; } - /** - * not currently used. - * @return the priority for this task - */ - public int getActivePriority() { - return !isTargetVisible() ? 0 : priority; - } - - /** - * not currently used. - * @return - */ - public int getInactivePriority() { - return isTargetVisible() ? priority : 0; - } - /** * detects targets from the centre of the tower to maxRange in a straight line. * @return true if mobs are present and false otherwise. */ public boolean isTargetVisible() { - return physics.raycast(towerPosition, maxRangePosition, TARGET, hit); + boolean top = physics.raycast(towerPosition.add(0f,0.4f), maxRangePosition.add(0f,0.4f), TARGET, hit); + boolean bottom = physics.raycast(towerPosition.sub(0f,0.4f), maxRangePosition.sub(0f,0.4f), TARGET, hit); + return top || bottom; } private void changeFireRateInterval(int newInterval) { fireRateInterval = 1 / ((float) newInterval / 5); } - /** - * Function for getting the turret's fire rate. - * - * @return The fireRateInterval variable - */ - public float getFireRateInterval() { - return fireRateInterval; - } - /** * Function for getting the tower's state. * diff --git a/source/core/src/main/com/csse3200/game/components/tasks/FireworksTowerCombatTask.java b/source/core/src/main/com/csse3200/game/components/tasks/FireworksTowerCombatTask.java index b40cfa0db..e8bcbe5de 100644 --- a/source/core/src/main/com/csse3200/game/components/tasks/FireworksTowerCombatTask.java +++ b/source/core/src/main/com/csse3200/game/components/tasks/FireworksTowerCombatTask.java @@ -13,6 +13,8 @@ import com.csse3200.game.services.GameTime; import com.csse3200.game.services.ServiceLocator; +import static java.lang.Math.round; + /** * The FireworksTowerCombatTask runs the AI for the FireworksTower class. The tower scans for mobs and targets in a @@ -38,6 +40,7 @@ public class FireworksTowerCombatTask extends DefaultTask implements PriorityTas private final Vector2 maxRangePosition = new Vector2(); private PhysicsEngine physics; private GameTime timeSource; + private float fireRateInterval; private long endTime; private final RaycastHit hit = new RaycastHit(); private boolean shoot = true; @@ -54,6 +57,7 @@ public enum STATE { public FireworksTowerCombatTask(int priority, float maxRange) { this.priority = priority; this.maxRange = maxRange; + this.fireRateInterval = 1; physics = ServiceLocator.getPhysicsService().getPhysics(); timeSource = ServiceLocator.getTimeSource(); } @@ -81,7 +85,12 @@ public void start() { public void update() { if (timeSource.getTime() >= endTime) { updateTowerState(); - endTime = timeSource.getTime() + (INTERVAL * 1000); + if (towerState == STATE.ATTACK) { + endTime = timeSource.getTime() + round(fireRateInterval * 1000); + } else { + endTime = timeSource.getTime() + (INTERVAL * 1000); + } + } } @@ -159,7 +168,9 @@ public int getPriority() { * @return true if targets are detected, false otherwise */ public boolean isTargetVisible() { - return physics.raycast(towerPosition, maxRangePosition, TARGET, hit); + boolean top = physics.raycast(towerPosition.add(0f,0.4f), maxRangePosition.add(0f,0.4f), TARGET, hit); + boolean bottom = physics.raycast(towerPosition.sub(0f,0.4f), maxRangePosition.sub(0f,0.4f), TARGET, hit); + return top || bottom; } /** diff --git a/source/core/src/main/com/csse3200/game/components/tasks/MobDeathTask.java b/source/core/src/main/com/csse3200/game/components/tasks/MobDeathTask.java index 5782ca8f9..85d29c15b 100644 --- a/source/core/src/main/com/csse3200/game/components/tasks/MobDeathTask.java +++ b/source/core/src/main/com/csse3200/game/components/tasks/MobDeathTask.java @@ -1,5 +1,6 @@ package com.csse3200.game.components.tasks; +import com.badlogic.gdx.math.MathUtils; import com.csse3200.game.ai.tasks.DefaultTask; import com.csse3200.game.ai.tasks.PriorityTask; import com.csse3200.game.components.CombatStatsComponent; @@ -11,6 +12,8 @@ import com.csse3200.game.services.GameTime; +/// THIS CODE IS REDUNDANT /// + /** * Task that prints a message to the terminal whenever it is called. */ @@ -97,10 +100,18 @@ private void killMob() { } private void dropCurrency() { + float randomValue = MathUtils.random(0,1); + Entity currency; + if (randomValue <= 0.1f) { + currency = DropFactory.createCrystalDrop(); + + } + else { + currency = DropFactory.createScrapDrop(); - Entity scrap = DropFactory.createScrapDrop(); - scrap.setPosition(mobPosition.x,mobPosition.y); - ServiceLocator.getEntityService().register(scrap); + } + currency.setPosition(mobPosition.x,mobPosition.y); + ServiceLocator.getEntityService().register(currency); } diff --git a/source/core/src/main/com/csse3200/game/components/tasks/MobTask/MobTask.java b/source/core/src/main/com/csse3200/game/components/tasks/MobTask/MobTask.java index 08aa0b627..92ee1f3a9 100644 --- a/source/core/src/main/com/csse3200/game/components/tasks/MobTask/MobTask.java +++ b/source/core/src/main/com/csse3200/game/components/tasks/MobTask/MobTask.java @@ -1,13 +1,16 @@ package com.csse3200.game.components.tasks.MobTask; +import com.badlogic.gdx.math.MathUtils; import com.badlogic.gdx.math.Vector2; import com.badlogic.gdx.utils.Timer; import com.csse3200.game.ai.tasks.DefaultTask; import com.csse3200.game.ai.tasks.PriorityTask; import com.csse3200.game.components.CombatStatsComponent; import com.csse3200.game.components.ProjectileEffects; +import com.csse3200.game.components.npc.DodgingComponent; import com.csse3200.game.components.tasks.MovementTask; import com.csse3200.game.entities.Entity; +import com.csse3200.game.entities.factories.DropFactory; import com.csse3200.game.entities.factories.ProjectileFactory; import com.csse3200.game.physics.PhysicsLayer; import com.csse3200.game.physics.components.HitboxComponent; @@ -15,6 +18,10 @@ import com.csse3200.game.rendering.AnimationRenderComponent; import com.csse3200.game.services.GameTime; import com.csse3200.game.services.ServiceLocator; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + import com.csse3200.game.components.tasks.MobTask.MobType; /** @@ -31,10 +38,17 @@ public class MobTask extends DefaultTask implements PriorityTask { private static final int MELEE_DAMAGE = 10; private static final long MELEE_ATTACK_SPEED = 2000; private static final long RANGE_ATTACK_SPEED = 5000; - private static final float MELEE_ATTACK_RANGE = 0.2f; + private static final float MELEE_ATTACK_RANGE = 0f; + + private static final float CRYSTAL_DROP_RATE = 0.1f; + private static final float SCRAP_DROP_RATE = 0.6f; + + private static final Logger logger = LoggerFactory.getLogger(MobTask.class); + // Private variables private final MobType mobType; + private State state = State.DEFAULT; private Entity mob; private AnimationRenderComponent animation; @@ -56,7 +70,7 @@ public class MobTask extends DefaultTask implements PriorityTask { // Enums private enum State { - RUN, ATTACK, DEATH, DEFAULT + RUN, ATTACK, DEATH, DEFAULT, DODGE } /** @@ -88,7 +102,6 @@ public void start() { super.start(); mob = owner.getEntity(); animation = mob.getComponent(AnimationRenderComponent.class); - mob.getComponent(PhysicsMovementComponent.class).setSpeed(MELEE_MOB_SPEED); melee = mobType.isMelee(); movementTask = new MovementTask(new Vector2(0f, mob.getPosition().y)); @@ -101,8 +114,10 @@ public void start() { if (melee) { mob.getComponent(PhysicsMovementComponent.class).setSpeed(MELEE_MOB_SPEED); + mob.getComponent(PhysicsMovementComponent.class).setNormalSpeed(MELEE_MOB_SPEED); } else { mob.getComponent(PhysicsMovementComponent.class).setSpeed(MELEE_RANGE_SPEED); + mob.getComponent(PhysicsMovementComponent.class).setNormalSpeed(MELEE_RANGE_SPEED); } } @@ -112,22 +127,39 @@ public void start() { @Override public void update() { + if(mob.getCenterPosition().x <= 1) { + mob.getComponent(CombatStatsComponent.class).setHealth(0); + + } + // death check - if (mob.getComponent(CombatStatsComponent.class).getHealth() <= 0 && !deathFlag) { + if ((mob.getComponent(CombatStatsComponent.class).getHealth() <= 0 && !deathFlag)) { + // decrement engineer count + // ! tests failing because of textbox + ServiceLocator.getGameEndService().updateEngineerCount(); changeState(State.DEATH); animate(); movementTask.stop(); deathFlag = true; + + } else if (deathFlag && animation.isFinished()) { ServiceLocator.getWaveService().updateEnemyCount(); mob.setFlagForDelete(true); + dropCurrency(); } + // Uhhh if(gameTime.getTime() >= dodgeEndTime) { if (canDodge) { mob.getEvents().trigger("dodgeIncomingEntity", mob.getCenterPosition()); - } + if(mob.getComponent(DodgingComponent.class).isTargetVisible(mob.getCenterPosition())) { + changeState(State.DODGE); + animate(); + // movementTask.stop(); + } + } dodgeEndTime = gameTime.getTime() + 500; // 500ms } @@ -158,9 +190,16 @@ public void update() { rangeAttack(); rangeAttackFlag = false; } + if (animation.isFinished()) { + changeState(State.RUN); + runFlag = true; + } + } + case DODGE -> { if (animation.isFinished()) { movementTask.start(); changeState(State.RUN); + animate(); runFlag = true; } } @@ -178,6 +217,7 @@ private void animate() { owner.getEntity().getEvents().trigger("mob_death"); owner.getEntity().getEvents().trigger("splitDeath"); } + case DODGE -> owner.getEntity().getEvents().trigger("mob_dodge"); case DEFAULT -> owner.getEntity().getEvents().trigger("mob_default"); } } @@ -268,4 +308,22 @@ public int getPriority() { public void setDodge(boolean dodgeFlag) { this.canDodge = dodgeFlag; } + + private void dropCurrency() { + float randomValue = MathUtils.random(0f,1f); + logger.info("Random value: " + randomValue); + Entity currency; + if (randomValue <= SCRAP_DROP_RATE && randomValue > CRYSTAL_DROP_RATE) { + currency = DropFactory.createScrapDrop(); + currency.setPosition(mob.getPosition().x,mob.getPosition().y); + ServiceLocator.getEntityService().register(currency); + } else if (randomValue <= CRYSTAL_DROP_RATE) { + currency = DropFactory.createCrystalDrop(); + currency.setPosition(mob.getPosition().x,mob.getPosition().y); + ServiceLocator.getEntityService().register(currency); + } + + + + } } diff --git a/source/core/src/main/com/csse3200/game/components/tasks/PierceTowerCombatTask.java b/source/core/src/main/com/csse3200/game/components/tasks/PierceTowerCombatTask.java index 75268d9b6..02d36a77c 100644 --- a/source/core/src/main/com/csse3200/game/components/tasks/PierceTowerCombatTask.java +++ b/source/core/src/main/com/csse3200/game/components/tasks/PierceTowerCombatTask.java @@ -13,6 +13,8 @@ import com.csse3200.game.services.GameTime; import com.csse3200.game.services.ServiceLocator; +import static java.lang.Math.round; + /** * The PierceTowerCombatTask runs the AI for the PierceTower class. The tower scans for mobs and targets in a straight * line from its centre coordinate and executes the trigger phrases for animations depeending on the current state of @@ -31,6 +33,7 @@ public class PierceTowerCombatTask extends DefaultTask implements PriorityTask { // Class attributes private final int priority; + private float fireRateInterval; private final float maxRange; private Vector2 towerPosition = new Vector2(10, 10); private final Vector2 maxRangePosition = new Vector2(); @@ -52,6 +55,7 @@ public enum STATE { public PierceTowerCombatTask(int priority, float maxRange) { this.priority = priority; this.maxRange = maxRange; + this.fireRateInterval = 1; physics = ServiceLocator.getPhysicsService().getPhysics(); timeSource = ServiceLocator.getTimeSource(); } @@ -79,6 +83,9 @@ public void start() { public void update() { if (timeSource.getTime() >= endTime) { updateTowerState(); + if (towerState == STATE.ATTACK) { + endTime = timeSource.getTime() + round(fireRateInterval * 1000); + } endTime = timeSource.getTime() + (INTERVAL * 1000); } } @@ -159,7 +166,9 @@ public int getPriority() { * @return true if targets are detected, false otherwise */ public boolean isTargetVisible() { - return physics.raycast(towerPosition, maxRangePosition, TARGET, hit); + boolean top = physics.raycast(towerPosition.add(0f,0.4f), maxRangePosition.add(0f,0.4f), TARGET, hit); + boolean bottom = physics.raycast(towerPosition.sub(0f,0.4f), maxRangePosition.sub(0f,0.4f), TARGET, hit); + return top || bottom; } /** diff --git a/source/core/src/main/com/csse3200/game/components/tasks/RicochetTowerCombatTask.java b/source/core/src/main/com/csse3200/game/components/tasks/RicochetTowerCombatTask.java index 6e60bf01d..004ab84d8 100644 --- a/source/core/src/main/com/csse3200/game/components/tasks/RicochetTowerCombatTask.java +++ b/source/core/src/main/com/csse3200/game/components/tasks/RicochetTowerCombatTask.java @@ -13,6 +13,8 @@ import com.csse3200.game.services.GameTime; import com.csse3200.game.services.ServiceLocator; +import static java.lang.Math.round; + /** * The RicochetTowerCombatTask runs the AI for the RicochetTower class. The tower scans for mobs and targets in a @@ -37,6 +39,7 @@ public class RicochetTowerCombatTask extends DefaultTask implements PriorityTask private final Vector2 maxRangePosition = new Vector2(); private PhysicsEngine physics; private GameTime timeSource; + private float fireRateInterval; private long endTime; private final RaycastHit hit = new RaycastHit(); private boolean shoot = true; @@ -54,6 +57,7 @@ public enum STATE { public RicochetTowerCombatTask(int priority, float maxRange) { this.priority = priority; this.maxRange = maxRange; + this.fireRateInterval = 1; physics = ServiceLocator.getPhysicsService().getPhysics(); timeSource = ServiceLocator.getTimeSource(); } @@ -81,7 +85,11 @@ public void start() { public void update() { if (timeSource.getTime() >= endTime) { updateTowerState(); - endTime = timeSource.getTime() + (INTERVAL * 1000); + if (towerState == STATE.ATTACK) { + endTime = timeSource.getTime() + round(fireRateInterval * 1000); + } else { + endTime = timeSource.getTime() + (INTERVAL * 1000); + } } } @@ -167,6 +175,8 @@ public int getPriority() { * @return true if targets are detected, false otherwise */ public boolean isTargetVisible() { - return physics.raycast(towerPosition, maxRangePosition, TARGET, hit); + boolean top = physics.raycast(towerPosition.add(0f,0.4f), maxRangePosition.add(0f,0.4f), TARGET, hit); + boolean bottom = physics.raycast(towerPosition.sub(0f,0.4f), maxRangePosition.sub(0f,0.4f), TARGET, hit); + return top || bottom; } } diff --git a/source/core/src/main/com/csse3200/game/components/tasks/StunTowerCombatTask.java b/source/core/src/main/com/csse3200/game/components/tasks/StunTowerCombatTask.java index 03abc3552..5a1de08a2 100644 --- a/source/core/src/main/com/csse3200/game/components/tasks/StunTowerCombatTask.java +++ b/source/core/src/main/com/csse3200/game/components/tasks/StunTowerCombatTask.java @@ -175,32 +175,18 @@ public int getPriority() { return !isTargetVisible() ? 0 : priority; } - public int getActivePriority() { - return !isTargetVisible() ? 0 : priority; - } - - public int getInactivePriority() { - return isTargetVisible() ? priority : 0; - } - /** * Searches for enemies/mobs in a straight line from the centre of the tower to maxRange in a straight line. * @return true if targets are detected, false otherwise */ public boolean isTargetVisible() { - return physics.raycast(towerPosition, maxRangePosition, TARGET, hit); + boolean top = physics.raycast(towerPosition.add(0f,0.4f), maxRangePosition.add(0f,0.4f), TARGET, hit); + boolean bottom = physics.raycast(towerPosition.sub(0f,0.4f), maxRangePosition.sub(0f,0.4f), TARGET, hit); + return top || bottom; } private void changeFireRateInterval(int newInterval) { fireRateInterval = 1 / ((float) newInterval / 5); } - /** - * Function for getting the turret's fire rate. - * - * @return The fireRateInterval variable - */ - public float getFireRateInterval() { - return fireRateInterval; - } } diff --git a/source/core/src/main/com/csse3200/game/components/tasks/TNTTowerCombatTask.java b/source/core/src/main/com/csse3200/game/components/tasks/TNTTowerCombatTask.java index 4c8016d24..db7f6a06a 100644 --- a/source/core/src/main/com/csse3200/game/components/tasks/TNTTowerCombatTask.java +++ b/source/core/src/main/com/csse3200/game/components/tasks/TNTTowerCombatTask.java @@ -133,7 +133,9 @@ public STATE getState() { */ public boolean isTargetVisible() { // If there is an obstacle in the path to the max range point, mobs visible. - return physics.raycast(towerPosition, maxRangePosition, TARGET, hit); + boolean top = physics.raycast(towerPosition.add(0f,0.4f), maxRangePosition.add(0f,0.4f), TARGET, hit); + boolean bottom = physics.raycast(towerPosition.sub(0f,0.4f), maxRangePosition.sub(0f,0.4f), TARGET, hit); + return top || bottom; } public boolean isReadyToDelete() { diff --git a/source/core/src/main/com/csse3200/game/components/tasks/TowerCombatTask.java b/source/core/src/main/com/csse3200/game/components/tasks/TowerCombatTask.java index b5dd8f45c..a882cd1bb 100644 --- a/source/core/src/main/com/csse3200/game/components/tasks/TowerCombatTask.java +++ b/source/core/src/main/com/csse3200/game/components/tasks/TowerCombatTask.java @@ -3,7 +3,6 @@ import com.badlogic.gdx.math.Vector2; import com.csse3200.game.ai.tasks.DefaultTask; import com.csse3200.game.ai.tasks.PriorityTask; -import com.csse3200.game.areas.ForestGameArea; import com.csse3200.game.components.CombatStatsComponent; import com.csse3200.game.entities.Entity; import com.csse3200.game.entities.factories.ProjectileFactory; @@ -57,7 +56,6 @@ private enum STATE { * @param maxRange Maximum effective range of the weapon tower. This determines the detection distance of targets */ public TowerCombatTask(int priority, float maxRange) { - this.priority = priority; this.maxRange = maxRange; this.fireRateInterval = 1; @@ -173,7 +171,9 @@ private int getInactivePriority() { */ private boolean isTargetVisible() { // If there is an obstacle in the path to the max range point, mobs visible. - return physics.raycast(towerPosition, maxRangePosition, TARGET, hit); + boolean top = physics.raycast(towerPosition.add(0f,0.4f), maxRangePosition.add(0f,0.4f), TARGET, hit); + boolean bottom = physics.raycast(towerPosition.sub(0f,0.4f), maxRangePosition.sub(0f,0.4f), TARGET, hit); + return top || bottom; } /** @@ -182,7 +182,7 @@ private boolean isTargetVisible() { * @param newInterval The rate at which the tower should fire projectiles in shots per second. */ private void changeFireRateInterval(int newInterval) { - logger.info("Changing fire rate to: %d", newInterval); + logger.info(String.format("Changing fire rate to: %d", newInterval)); fireRateInterval = 1 / ((float) newInterval / 5); } diff --git a/source/core/src/main/com/csse3200/game/components/tasks/WallTowerDestructionTask.java b/source/core/src/main/com/csse3200/game/components/tasks/WallTowerDestructionTask.java index 749647851..1e2f3ff90 100644 --- a/source/core/src/main/com/csse3200/game/components/tasks/WallTowerDestructionTask.java +++ b/source/core/src/main/com/csse3200/game/components/tasks/WallTowerDestructionTask.java @@ -4,8 +4,6 @@ import com.csse3200.game.ai.tasks.DefaultTask; import com.csse3200.game.ai.tasks.PriorityTask; import com.csse3200.game.components.CombatStatsComponent; -import com.csse3200.game.entities.Entity; -import com.csse3200.game.entities.factories.ProjectileFactory; import com.csse3200.game.physics.PhysicsEngine; import com.csse3200.game.physics.PhysicsLayer; import com.csse3200.game.physics.raycast.RaycastHit; @@ -43,7 +41,7 @@ public class WallTowerDestructionTask extends DefaultTask implements PriorityTas public enum STATE { IDLE, DEATH } - public STATE towerState = STATE.IDLE; + private STATE towerState = STATE.IDLE; /** * @param priority Task priority when targets are detected (0 when nothing is present) @@ -75,6 +73,7 @@ public void start() { * updates the current state of the tower based on the current state of the game. If enemies are detected, attack * state is activated and otherwise idle state remains. */ + @Override public void update() { if (timeSource.getTime() >= endTime) { updateTowerState(); @@ -87,22 +86,19 @@ public void update() { * of the game. If enemies are detected, state of the tower is changed to attack state. */ public void updateTowerState() { - if (owner.getEntity().getComponent(CombatStatsComponent.class).getHealth() <= 0 && towerState != STATE.DEATH) { owner.getEntity().getEvents().trigger(DEATH); towerState = STATE.DEATH; return; } - switch (towerState) { - case IDLE -> { - owner.getEntity().getEvents().trigger(IDLE); - towerState = STATE.IDLE; - } - case DEATH -> { - if (owner.getEntity().getComponent(AnimationRenderComponent.class).isFinished()) { - owner.getEntity().setFlagForDelete(true); - } + // Replace "switch" statement by "if" statements to increase readability. + if (towerState == STATE.IDLE) { + owner.getEntity().getEvents().trigger(IDLE); + towerState = STATE.IDLE; + } else { // DEATH + if (owner.getEntity().getComponent(AnimationRenderComponent.class).isFinished()) { + owner.getEntity().setFlagForDelete(true); } } } @@ -115,9 +111,18 @@ public STATE getState() { return this.towerState; } + /** + * Function for setting the tower's state. + * @param newState The new state of this tower. + */ + public void setState(STATE newState) { + this.towerState = newState; + } + /** * stops the current animation and switches back the state of the tower to IDLE. */ + @Override public void stop() { super.stop(); owner.getEntity().getEvents().trigger(IDLE); diff --git a/source/core/src/main/com/csse3200/game/components/tasks/bombship/BombshipCombatTask.java b/source/core/src/main/com/csse3200/game/components/tasks/bombship/BombshipCombatTask.java index 0c044d89f..3dd7a3124 100644 --- a/source/core/src/main/com/csse3200/game/components/tasks/bombship/BombshipCombatTask.java +++ b/source/core/src/main/com/csse3200/game/components/tasks/bombship/BombshipCombatTask.java @@ -4,28 +4,18 @@ import com.badlogic.gdx.math.Vector2; import com.csse3200.game.ai.tasks.DefaultTask; import com.csse3200.game.ai.tasks.PriorityTask; -import com.csse3200.game.entities.Entity; -import com.csse3200.game.entities.factories.ProjectileFactory; import com.csse3200.game.physics.PhysicsEngine; -import com.csse3200.game.physics.PhysicsLayer; -import com.csse3200.game.physics.raycast.RaycastHit; import com.csse3200.game.services.GameTime; import com.csse3200.game.services.ServiceLocator; - -import java.util.ArrayList; - /** * The AI Task for the Engineer entity. The Engineer will scan for targets within its detection range * and trigger events to change its state accordingly. This task must be called once the Engineer has * appropiately moved into position. */ public class BombshipCombatTask extends DefaultTask implements PriorityTask { - private static final int INTERVAL = 1; // The time interval for each target scan from the Engineer. private static final int PRIORITY = 3; // Default priority of the combat task when mobs are in range. - private static final short TARGET1 = PhysicsLayer.BOSS; // The type of targets that the Engineer will detect. - private static final short TARGET2 = PhysicsLayer.XENO; // Animation event names for the Engineer's state machine. private static final String START = "start"; @@ -34,18 +24,11 @@ public class BombshipCombatTask extends DefaultTask implements PriorityTask { // The Engineer's attributes. private final float maxRange; // The maximum range of the bombship. - - private Vector2 bombShipPosition = new Vector2(0, 0); // Placeholder value for the Bombship's position. private final Vector2 maxRangePosition = new Vector2(); private PhysicsEngine physics; private GameTime timeSource; private long endTime; - private long reloadTime; -/** - private ArrayList hits = new ArrayList<>(); - private final RaycastHit hit = new RaycastHit(); - private ArrayList targets = new ArrayList<>(); -*/ + /** The Engineer's states. */ private enum STATE { IDLE, START , DESTROY @@ -64,8 +47,11 @@ public BombshipCombatTask(float maxRange) { @Override public void start() { super.start(); - this.bombShipPosition = owner.getEntity().getCenterPosition(); + + // Placeholder value for the Bombship's position. + Vector2 bombShipPosition = owner.getEntity().getCenterPosition(); this.maxRangePosition.set(bombShipPosition.x + maxRange, bombShipPosition.y); + // Default to idle mode owner.getEntity().getEvents().trigger(IDLE); endTime = timeSource.getTime() + (INTERVAL * 500); @@ -104,9 +90,7 @@ public void updateBombshipState() { owner.getEntity().getEvents().trigger(START); } } - case DESTROY -> { - owner.getEntity().getEvents().trigger(DESTROY); - } + default -> owner.getEntity().getEvents().trigger(DESTROY); // DESTROY } } @@ -117,13 +101,6 @@ private void combatState() { owner.getEntity().getEvents().trigger(START); bombshipState = STATE.START; } - /** - * For stopping the running task - */ - @Override - public void stop() { - super.stop(); - } /** * Simplified getPriority function, returns the priority of the task @@ -141,32 +118,7 @@ public int getPriority() { * @return true if a target is detected, false otherwise */ public boolean isEngineerDied() { - //if (engineerCount < maxEngineers) { - return true; - //} - } - - /** - * Fetches the nearest target from the array of detected target positions created during the last call of - * this could be done in the next sprint , the scan doesnt work as of now ! - * @return a Vector2 position of the nearest mob detected. - */ - /** public Vector2 fetchTarget() { - // Initial nearest position for comparison - int lowest = 10; - - Vector2 nearest = new Vector2(owner.getEntity().getCenterPosition().x, - owner.getEntity().getCenterPosition().y); - - // Find the nearest target from the array of targets - for (Vector2 tgt : targets){ - if (Math.abs(tgt.y - nearest.y) < lowest) { - lowest = (int)Math.abs(tgt.y - nearest.y); - nearest = tgt; - } - } - return nearest; + return true; } - */ } diff --git a/source/core/src/main/com/csse3200/game/components/tasks/bombship/BombshipWanderTask.java b/source/core/src/main/com/csse3200/game/components/tasks/bombship/BombshipWanderTask.java index 63c611885..faf62ba6c 100644 --- a/source/core/src/main/com/csse3200/game/components/tasks/bombship/BombshipWanderTask.java +++ b/source/core/src/main/com/csse3200/game/components/tasks/bombship/BombshipWanderTask.java @@ -4,12 +4,10 @@ import com.csse3200.game.ai.tasks.DefaultTask; import com.csse3200.game.ai.tasks.PriorityTask; import com.csse3200.game.ai.tasks.Task; -import com.csse3200.game.components.CombatStatsComponent; import com.csse3200.game.physics.PhysicsLayer; import com.csse3200.game.physics.components.ColliderComponent; import com.csse3200.game.physics.components.HitboxComponent; import com.csse3200.game.rendering.AnimationRenderComponent; -import com.csse3200.game.services.ServiceLocator; /** * BombshipWanderTask is the entry point for the engineer entity's behaviour. Instantiates subtasks HumanWaitTask, @@ -17,7 +15,6 @@ * handled in this class. */ public class BombshipWanderTask extends DefaultTask implements PriorityTask { - private static final int TOLERANCE = 1; private static final float STOP_DISTANCE = 0.5f; private static final int DEFAULT_PRIORITY = 1; private static final String START = "start"; @@ -85,28 +82,18 @@ public void start() { */ @Override public void update() { - if (!isDestroyed) { - startDestroying(); - } - // Check if bombship has destroyed since last update if (!isDestroyed) { - startDestroying(); - } else if (isDestroyed && animator.isFinished()) { - owner.getEntity().setFlagForDelete(true); - } - - // otherwise doing engineer things since engineer is alive - else if (!isDestroyed){ doBombshipThings(); - + startDestroying();; currentTask.update(); + } else if (isDestroyed && animator.isFinished()) { + owner.getEntity().setFlagForDelete(true); } } private void doBombshipThings() { if (currentTask.getStatus() != Status.ACTIVE) { - // if the engineer is in move state and update has been called, engineer has arrived at destination if (currentTask == movementTask) { startWaiting(); @@ -117,6 +104,7 @@ private void doBombshipThings() { } } } + /** * Handle the dying phase of the entity. Triggers an event to play the appropriate media, * sets HitBox and Collider components to ignore contact (stops the body being pushed around) @@ -137,22 +125,6 @@ private void startWaiting() { swapTask(waitTask); } - /** - * Starts the movement task, to a particular destination - * @param destination the Vector2 position to which the entity needs to move - */ - private void startMoving(Vector2 destination) { - movementTask.setTarget(destination); - swapTask(movementTask); - } - - /** - * Starts the combat task. - */ - private void startCombat() { - swapTask(combatTask); - } - /** * Allows manual switching of tasks, from the current task to the supplied newTask. * @param newTask the task being switched to. @@ -161,6 +133,7 @@ private void swapTask(Task newTask) { if (currentTask != null) { currentTask.stop(); } + currentTask = newTask; currentTask.start(); } diff --git a/source/core/src/main/com/csse3200/game/components/tasks/bosstask/DemonBossTask.java b/source/core/src/main/com/csse3200/game/components/tasks/bosstask/DemonBossTask.java index 38896b2b4..a861a3a5c 100644 --- a/source/core/src/main/com/csse3200/game/components/tasks/bosstask/DemonBossTask.java +++ b/source/core/src/main/com/csse3200/game/components/tasks/bosstask/DemonBossTask.java @@ -10,6 +10,7 @@ import com.csse3200.game.components.ProjectileEffects; import com.csse3200.game.components.tasks.MovementTask; import com.csse3200.game.entities.Entity; +import com.csse3200.game.entities.factories.DropFactory; import com.csse3200.game.entities.factories.MobBossFactory; import com.csse3200.game.entities.factories.ProjectileFactory; import com.csse3200.game.physics.PhysicsLayer; @@ -140,6 +141,8 @@ public void update() { slimey.setScale(5f, 5f); ServiceLocator.getEntityService().register(slimey); demon.setFlagForDelete(true); + dropCurrency(); + } // detect half health @@ -495,4 +498,19 @@ public void run() { }, (float) (i + 1) * 2); } } + + private void dropCurrency() { + // Create and register 5 crystal drops around the bossPosition + for (int i = 0; i < 5; i++) { + Entity crystal = DropFactory.createCrystalDrop(); + + // Calculate positions around the bossPosition + float offsetX = MathUtils.random(-1f, 1f); // Adjust the range as needed + float offsetY = MathUtils.random(-1f, 1f); + float dropX = owner.getEntity().getPosition().x + offsetX; + float dropY = owner.getEntity().getPosition().y + offsetY; + crystal.setPosition(dropX, dropY); + ServiceLocator.getEntityService().register(crystal); + } + } } \ No newline at end of file diff --git a/source/core/src/main/com/csse3200/game/components/tasks/bosstask/IceBabyTask.java b/source/core/src/main/com/csse3200/game/components/tasks/bosstask/IceBabyTask.java index 4b298c059..39270ec91 100644 --- a/source/core/src/main/com/csse3200/game/components/tasks/bosstask/IceBabyTask.java +++ b/source/core/src/main/com/csse3200/game/components/tasks/bosstask/IceBabyTask.java @@ -9,6 +9,7 @@ import com.csse3200.game.components.CombatStatsComponent; import com.csse3200.game.components.tasks.MovementTask; import com.csse3200.game.entities.Entity; +import com.csse3200.game.entities.factories.DropFactory; import com.csse3200.game.entities.factories.NPCFactory; import com.csse3200.game.physics.PhysicsLayer; import com.csse3200.game.physics.components.HitboxComponent; @@ -38,8 +39,8 @@ public class IceBabyTask extends DefaultTask implements PriorityTask { private Vector2 currentPos; private Vector2 walkPos; private MovementTask walkTask; - private static int xRightBoundary = 17; - private static int xLeftBoundary = 12; + private int xRightBoundary = 17; + private int xLeftBoundary = 12; private boolean startFlag = false; private boolean isWalking; /** Animation constants */ @@ -71,6 +72,7 @@ public void start() { animation = iceBaby.getComponent(AnimationRenderComponent.class); currentPos = iceBaby.getPosition(); iceBaby.getComponent(PhysicsMovementComponent.class).setSpeed(ICEBABY_SPEED); + iceBaby.getComponent(PhysicsMovementComponent.class).setNormalSpeed(ICEBABY_SPEED); Timer.schedule(new Timer.Task() { @Override public void run() { @@ -114,6 +116,7 @@ public void update() { animate(); if (animation.isFinished()) { iceBaby.setFlagForDelete(true); + dropCurrency(); } } @@ -318,4 +321,20 @@ public int getPriority() { return PRIORITY; } + + private void dropCurrency() { + // Create and register 5 crystal drops around the bossPosition + for (int i = 0; i < 5; i++) { + Entity crystal = DropFactory.createCrystalDrop(); + + // Calculate positions around the bossPosition + float offsetX = MathUtils.random(-1f, 1f); // Adjust the range as needed + float offsetY = MathUtils.random(-1f, 1f); + float dropX = owner.getEntity().getPosition().x + offsetX; + float dropY = owner.getEntity().getPosition().y + offsetY; + crystal.setPosition(dropX, dropY); + ServiceLocator.getEntityService().register(crystal); + } + } + } diff --git a/source/core/src/main/com/csse3200/game/components/tasks/bosstask/MobBossDeathTask.java b/source/core/src/main/com/csse3200/game/components/tasks/bosstask/MobBossDeathTask.java index 719c2436a..1bc2db0c3 100644 --- a/source/core/src/main/com/csse3200/game/components/tasks/bosstask/MobBossDeathTask.java +++ b/source/core/src/main/com/csse3200/game/components/tasks/bosstask/MobBossDeathTask.java @@ -1,5 +1,6 @@ package com.csse3200.game.components.tasks.bosstask; +import com.badlogic.gdx.math.MathUtils; import com.badlogic.gdx.math.Vector2; import com.csse3200.game.ai.tasks.DefaultTask; import com.csse3200.game.ai.tasks.PriorityTask; @@ -9,6 +10,8 @@ import com.csse3200.game.services.GameTime; import com.csse3200.game.services.ServiceLocator; +//CODE IS REDUNDANT /// + /** * Task that prints a message to the terminal whenever it is called. */ @@ -93,10 +96,18 @@ private void killboss() { } private void dropCurrency() { + // Create and register 5 crystal drops around the bossPosition + for (int i = 0; i < 5; i++) { + Entity crystal = DropFactory.createCrystalDrop(); + + // Calculate positions around the bossPosition + float offsetX = MathUtils.random(-1f, 1f); // Adjust the range as needed - Entity scrap = DropFactory.createScrapDrop(); - scrap.setPosition(bossPosition.x,bossPosition.y); - ServiceLocator.getEntityService().register(scrap.setScale(2,2)); + float dropX = bossPosition.x + offsetX; + crystal.setPosition(dropX, bossPosition.y); + ServiceLocator.getEntityService().register(crystal); + } } + } diff --git a/source/core/src/main/com/csse3200/game/components/tasks/bosstask/PatrickDeathTask.java b/source/core/src/main/com/csse3200/game/components/tasks/bosstask/PatrickDeathTask.java index 97f7dacc4..92da2af17 100644 --- a/source/core/src/main/com/csse3200/game/components/tasks/bosstask/PatrickDeathTask.java +++ b/source/core/src/main/com/csse3200/game/components/tasks/bosstask/PatrickDeathTask.java @@ -1,8 +1,12 @@ package com.csse3200.game.components.tasks.bosstask; +import com.badlogic.gdx.math.MathUtils; import com.csse3200.game.ai.tasks.DefaultTask; import com.csse3200.game.ai.tasks.PriorityTask; +import com.csse3200.game.entities.Entity; +import com.csse3200.game.entities.factories.DropFactory; import com.csse3200.game.rendering.AnimationRenderComponent; +import com.csse3200.game.services.ServiceLocator; public class PatrickDeathTask extends DefaultTask implements PriorityTask { @@ -27,6 +31,8 @@ public void update() { if (startFlag && owner.getEntity().getComponent(AnimationRenderComponent.class). isFinished()) { owner.getEntity().setFlagForDelete(true); + ServiceLocator.getGameEndService().updateEngineerCount(); + dropCurrency(); } } @@ -37,4 +43,19 @@ public void update() { public int getPriority() { return PRIORITY; } + + private void dropCurrency() { + // Create and register 5 crystal drops around the bossPosition + for (int i = 0; i < 5; i++) { + Entity crystal = DropFactory.createCrystalDrop(); + + // Calculate positions around the bossPosition + float offsetX = MathUtils.random(-1f, 1f); // Adjust the range as needed + float offsetY = MathUtils.random(-1f, 1f); + float dropX = owner.getEntity().getPosition().x + offsetX; + float dropY = owner.getEntity().getPosition().y + offsetY; + crystal.setPosition(dropX, dropY); + ServiceLocator.getEntityService().register(crystal); + } + } } diff --git a/source/core/src/main/com/csse3200/game/components/tasks/bosstask/PatrickTask.java b/source/core/src/main/com/csse3200/game/components/tasks/bosstask/PatrickTask.java index bf5e6c30c..cb3f27b8f 100644 --- a/source/core/src/main/com/csse3200/game/components/tasks/bosstask/PatrickTask.java +++ b/source/core/src/main/com/csse3200/game/components/tasks/bosstask/PatrickTask.java @@ -64,6 +64,7 @@ public void start() { patrick = owner.getEntity(); animation = owner.getEntity().getComponent(AnimationRenderComponent.class); // get animation patrick.getComponent(PhysicsMovementComponent.class).setSpeed(PATRICK_SPEED); // set speed + patrick.getComponent(PhysicsMovementComponent.class).setNormalSpeed(PATRICK_SPEED); // give game time to load Timer.schedule(new Timer.Task() { diff --git a/source/core/src/main/com/csse3200/game/components/tasks/bosstask/SlimeyBoyTask.java b/source/core/src/main/com/csse3200/game/components/tasks/bosstask/SlimeyBoyTask.java index fe8cefae4..bf607a123 100644 --- a/source/core/src/main/com/csse3200/game/components/tasks/bosstask/SlimeyBoyTask.java +++ b/source/core/src/main/com/csse3200/game/components/tasks/bosstask/SlimeyBoyTask.java @@ -1,5 +1,6 @@ package com.csse3200.game.components.tasks.bosstask; +import com.badlogic.gdx.math.MathUtils; import com.badlogic.gdx.math.Vector2; import com.badlogic.gdx.utils.Array; import com.badlogic.gdx.utils.Timer; @@ -8,6 +9,7 @@ import com.csse3200.game.components.CombatStatsComponent; import com.csse3200.game.components.tasks.MovementTask; import com.csse3200.game.entities.Entity; +import com.csse3200.game.entities.factories.DropFactory; import com.csse3200.game.physics.PhysicsLayer; import com.csse3200.game.physics.components.PhysicsMovementComponent; import com.csse3200.game.rendering.AnimationRenderComponent; @@ -186,4 +188,6 @@ private void applyAoeDamage(Array targets, int damage) { public int getPriority() { return PRIORITY; } + + } diff --git a/source/core/src/main/com/csse3200/game/components/tasks/human/EngineerCombatTask.java b/source/core/src/main/com/csse3200/game/components/tasks/human/EngineerCombatTask.java index 3598a245a..f4a803893 100644 --- a/source/core/src/main/com/csse3200/game/components/tasks/human/EngineerCombatTask.java +++ b/source/core/src/main/com/csse3200/game/components/tasks/human/EngineerCombatTask.java @@ -10,6 +10,8 @@ import com.csse3200.game.physics.raycast.RaycastHit; import com.csse3200.game.services.GameTime; import com.csse3200.game.services.ServiceLocator; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; import java.util.ArrayList; @@ -32,7 +34,7 @@ public class EngineerCombatTask extends DefaultTask implements PriorityTask { // The Engineer's attributes. private final float maxRange; // The maximum range of the Engineer's weapon. // weaponCapacity is the number of shots fired before the engineer has to reload - private static final int weaponCapacity = 10; + private static final int WEAPON_CAPACITY = 10; private int shotsFired = 0; // Tracks the number of shots fired in the current cycle private final Vector2 maxRangePosition = new Vector2(); @@ -41,6 +43,8 @@ public class EngineerCombatTask extends DefaultTask implements PriorityTask { private long endTime; private long reloadTime; + private static final Logger logger = LoggerFactory.getLogger(EngineerCombatTask.class); + private ArrayList hits = new ArrayList<>(); private final RaycastHit hit = new RaycastHit(); private ArrayList targets = new ArrayList<>(); @@ -106,7 +110,7 @@ public void updateEngineerState() { owner.getEntity().getEvents().trigger(IDLE_RIGHT); engineerState = STATE.IDLE_RIGHT; } else { - if (shotsFired <= weaponCapacity) { + if (shotsFired <= WEAPON_CAPACITY) { owner.getEntity().getEvents().trigger(FIRING); owner.getEntity().getEvents().trigger(ENGINEER_PROJECTILE_FIRED); // this might be changed to an event which gets triggered everytime the tower enters the firing state @@ -166,6 +170,7 @@ public boolean isTargetVisible() { // If there is an obstacle in the path to the max range point, mobs visible. Vector2 position = owner.getEntity().getCenterPosition(); hits.clear(); + targets.clear(); for (int i = 5; i > -5; i--) { if (physics.raycast(position, new Vector2(position.x + maxRange, position.y + i), TARGET, hit)) { hits.add(hit); @@ -182,16 +187,17 @@ public boolean isTargetVisible() { */ public Vector2 fetchTarget() { // Initial nearest position for comparison - int lowest = 10; + float currentClosest = Float.MAX_VALUE; Vector2 nearest = new Vector2(owner.getEntity().getCenterPosition().x, owner.getEntity().getCenterPosition().y); + Vector2 enggPosition = owner.getEntity().getCenterPosition(); - // Find the nearest target from the array of targets - for (Vector2 tgt : targets){ - if (Math.abs(tgt.y - nearest.y) < lowest) { - lowest = (int)Math.abs(tgt.y - nearest.y); - nearest = tgt; + for (Vector2 target : targets) { + float distance = enggPosition.dst(target); // euclidean distance + if (distance < currentClosest) { + currentClosest = distance; + nearest = target; } } return nearest; diff --git a/source/core/src/main/com/csse3200/game/components/tasks/human/HumanWanderTask.java b/source/core/src/main/com/csse3200/game/components/tasks/human/HumanWanderTask.java index 421fbf6ce..f6b055873 100644 --- a/source/core/src/main/com/csse3200/game/components/tasks/human/HumanWanderTask.java +++ b/source/core/src/main/com/csse3200/game/components/tasks/human/HumanWanderTask.java @@ -124,14 +124,14 @@ private void doEngineerThings() { float engY = owner.getEntity().getCenterPosition().y; float targetY = combatTask.fetchTarget().y; // if the engineer is positioned within the tolerance range of the mob's y position, enter combat state - if (engY < targetY + TOLERANCE && + if (engY < targetY + TOLERANCE && engY > targetY - TOLERANCE) { startCombat(); // move into position for targeting mob } else if (!this.isSelected()) { - Vector2 newPos = new Vector2(owner.getEntity().getPosition().x, combatTask.fetchTarget().y); - startMoving(newPos); + Vector2 newPos = new Vector2(owner.getEntity().getPosition().x, combatTask.fetchTarget().y); + startMoving(newPos); } } } diff --git a/source/core/src/main/com/csse3200/game/components/tasks/scanner/ScannerTask.java b/source/core/src/main/com/csse3200/game/components/tasks/scanner/ScannerTask.java index 3405cecd0..f04b7f2c9 100644 --- a/source/core/src/main/com/csse3200/game/components/tasks/scanner/ScannerTask.java +++ b/source/core/src/main/com/csse3200/game/components/tasks/scanner/ScannerTask.java @@ -8,6 +8,7 @@ import com.csse3200.game.physics.PhysicsEngine; import com.csse3200.game.physics.PhysicsLayer; import com.csse3200.game.physics.raycast.RaycastHit; +import com.csse3200.game.services.GameEndService; import com.csse3200.game.services.GameTime; import com.csse3200.game.services.ServiceLocator; @@ -31,8 +32,8 @@ public class ScannerTask extends DefaultTask implements PriorityTask { private boolean mobs = false; // track the number of engineers spawned. - private static final int maxEngineers = 3; - private int engineerCount = 0; + private static final int MAX_ENGINEERS = ServiceLocator.getGameEndService().getEngineerCount(); + /** * ScannerTask Constructor @@ -68,12 +69,12 @@ public void update() { scan(); if (!towers && !engineers && mobs) { // spawn engineers now - if (engineerCount < maxEngineers) { + if (ServiceLocator.getGameEndService().getNumSpawnedEngineers() < MAX_ENGINEERS) { Entity engineer = EngineerFactory.createEngineer(); engineer.setPosition(new Vector2((int)(selfPosition.x + 1),(int) selfPosition.y)); ServiceLocator.getEntityService().register(engineer); - engineerCount += 1; + ServiceLocator.getGameEndService().incrementNumSpawnedEngineers(); } } endTime = timeSource.getTime() + SCAN_INTERVAL; diff --git a/source/core/src/main/com/csse3200/game/components/tasks/waves/LevelWaves.java b/source/core/src/main/com/csse3200/game/components/tasks/waves/LevelWaves.java index cca65b2e5..19d6e910c 100644 --- a/source/core/src/main/com/csse3200/game/components/tasks/waves/LevelWaves.java +++ b/source/core/src/main/com/csse3200/game/components/tasks/waves/LevelWaves.java @@ -1,6 +1,7 @@ package com.csse3200.game.components.tasks.waves; import com.badlogic.gdx.math.GridPoint2; +import com.badlogic.gdx.math.MathUtils; import com.csse3200.game.entities.Entity; import com.csse3200.game.services.GameTime; import com.csse3200.game.services.ServiceLocator; @@ -14,7 +15,6 @@ public class LevelWaves extends Entity { private float spawnDelay; private GameTime gameTime; private long startTime; - private Random rand = new Random(); private int currentRandom = 0; private int previousRandom = 0; private int mobIndex; @@ -63,7 +63,7 @@ public WaveClass getWave(int index) { public void spawnWave() { if (gameTime.getTime() >= startTime + spawnDelay * 1000) { do { - currentRandom = rand.nextInt(0, ServiceLocator.getMapService().getHeight()); + currentRandom = MathUtils.random(0, ServiceLocator.getMapService().getHeight()); } while (currentRandom == previousRandom); ServiceLocator.getWaveService().setNextLane(currentRandom); GridPoint2 randomPos = new GridPoint2(19, currentRandom); diff --git a/source/core/src/main/com/csse3200/game/components/tower/DroidAnimationController.java b/source/core/src/main/com/csse3200/game/components/tower/DroidAnimationController.java index efd00a117..8ed009a12 100644 --- a/source/core/src/main/com/csse3200/game/components/tower/DroidAnimationController.java +++ b/source/core/src/main/com/csse3200/game/components/tower/DroidAnimationController.java @@ -1,5 +1,6 @@ package com.csse3200.game.components.tower; +import com.badlogic.gdx.audio.Sound; import com.badlogic.gdx.math.Vector2; import com.csse3200.game.components.Component; import com.csse3200.game.components.ProjectileEffects; @@ -16,6 +17,11 @@ public class DroidAnimationController extends Component { private AnimationRenderComponent animator; + private static final String FIRE_SINGLE_SFX = "sounds/towers/5.56_single_shot.mp3"; + + private final Sound fireSingleSound = ServiceLocator.getResourceService().getAsset( + FIRE_SINGLE_SFX, Sound.class); + /** * Creation call for a DroidAnimationController, fetches the animationRenderComponent that this controller will * be attached to and registers all the event listeners required to trigger the animations and sounds. @@ -66,6 +72,7 @@ void animateGoDown() { */ void animateAttackUp() { animator.startAnimation("attackUp"); + fireSingleSound.play(); } /** @@ -74,6 +81,7 @@ void animateAttackUp() { */ void animateAttackDown() { animator.startAnimation("attackDown"); + fireSingleSound.play(); } /** diff --git a/source/core/src/main/com/csse3200/game/components/tower/EconTowerAnimationController.java b/source/core/src/main/com/csse3200/game/components/tower/EconTowerAnimationController.java index 3fcd566f5..dd6a1a07f 100644 --- a/source/core/src/main/com/csse3200/game/components/tower/EconTowerAnimationController.java +++ b/source/core/src/main/com/csse3200/game/components/tower/EconTowerAnimationController.java @@ -1,7 +1,9 @@ package com.csse3200.game.components.tower; +import com.badlogic.gdx.audio.Sound; import com.csse3200.game.components.Component; import com.csse3200.game.rendering.AnimationRenderComponent; +import com.csse3200.game.services.ServiceLocator; /** * Listens for events relevant to a weapon tower state. @@ -15,6 +17,11 @@ public class EconTowerAnimationController extends Component { private static final String ECO_MOVE = "move1"; private static final String ECO_IDLE = "idle"; + private static final String PING_PATH = "sounds/towers/eco_tower_ping.mp3"; + + private final Sound ping = ServiceLocator.getResourceService().getAsset( + PING_PATH, Sound.class); + AnimationRenderComponent animator; /** @@ -36,7 +43,8 @@ void animateIdle() { animator.startAnimation(ECO_IDLE); } - void animateMove() { animator.startAnimation(ECO_MOVE); } + void animateMove() { animator.startAnimation(ECO_MOVE); + ping.play();} } \ No newline at end of file diff --git a/source/core/src/main/com/csse3200/game/components/tower/FireTowerAnimationController.java b/source/core/src/main/com/csse3200/game/components/tower/FireTowerAnimationController.java index 0bfe1fe11..a2cd29c89 100644 --- a/source/core/src/main/com/csse3200/game/components/tower/FireTowerAnimationController.java +++ b/source/core/src/main/com/csse3200/game/components/tower/FireTowerAnimationController.java @@ -21,7 +21,11 @@ public class FireTowerAnimationController extends Component{ private static final String PREP_ATTACK_ANIM = "prepAttack"; private static final String ATTACK_ANIM = "attack"; private static final String DEATH_ANIM = "death"; - //here we can add the sounds for the implemented animations + + private static final String FIRE_SINGLE_SFX = "sounds/towers/Desert-Eagle-Far-Single-Gunshot.mp3"; + + private final Sound fireSingleSound = ServiceLocator.getResourceService().getAsset( + FIRE_SINGLE_SFX, Sound.class); AnimationRenderComponent animator; @@ -57,6 +61,7 @@ void animatePrepAttack() { */ void animateAttack() { animator.startAnimation(ATTACK_ANIM); + fireSingleSound.play(); } void animateDeath() { diff --git a/source/core/src/main/com/csse3200/game/components/tower/FireworksTowerAnimationController.java b/source/core/src/main/com/csse3200/game/components/tower/FireworksTowerAnimationController.java index cd701b0db..e9ec5c653 100644 --- a/source/core/src/main/com/csse3200/game/components/tower/FireworksTowerAnimationController.java +++ b/source/core/src/main/com/csse3200/game/components/tower/FireworksTowerAnimationController.java @@ -1,7 +1,9 @@ package com.csse3200.game.components.tower; +import com.badlogic.gdx.audio.Sound; import com.csse3200.game.components.Component; import com.csse3200.game.rendering.AnimationRenderComponent; +import com.csse3200.game.services.ServiceLocator; /** * This class listens to events relevant to DroidTower entity's state and plays the animation when one @@ -10,6 +12,11 @@ public class FireworksTowerAnimationController extends Component { private AnimationRenderComponent animator; + private static final String FIRE_SINGLE_SFX = "sounds/towers/5.56_single_shot.mp3"; + + private final Sound fireSingleSound = ServiceLocator.getResourceService().getAsset( + FIRE_SINGLE_SFX, Sound.class); + /** * Creation call for a DroidAnimationController, fetches the animationRenderComponent that this controller will * be attached to and registers all the event listeners required to trigger the animations and sounds. @@ -30,6 +37,7 @@ public void create() { */ void animateAttack() { animator.startAnimation("Attack"); + fireSingleSound.play(); } diff --git a/source/core/src/main/com/csse3200/game/components/tower/PierceTowerAnimationController.java b/source/core/src/main/com/csse3200/game/components/tower/PierceTowerAnimationController.java index 9daa39acb..f9018e0af 100644 --- a/source/core/src/main/com/csse3200/game/components/tower/PierceTowerAnimationController.java +++ b/source/core/src/main/com/csse3200/game/components/tower/PierceTowerAnimationController.java @@ -1,7 +1,9 @@ package com.csse3200.game.components.tower; +import com.badlogic.gdx.audio.Sound; import com.csse3200.game.components.Component; import com.csse3200.game.rendering.AnimationRenderComponent; +import com.csse3200.game.services.ServiceLocator; /** * Listens to triggers phrases and executes the required animations. @@ -19,7 +21,10 @@ public class PierceTowerAnimationController extends Component { private static final String ALERT_ANIM = "Warning"; //further sounds can be added for the tower attacks/movement + private static final String FIRE_SINGLE_SFX = "sounds/towers/5.56_single_shot.mp3"; + private final Sound fireSingleSound = ServiceLocator.getResourceService().getAsset( + FIRE_SINGLE_SFX, Sound.class); AnimationRenderComponent animator; /** @@ -48,6 +53,7 @@ void animateIdle() { */ void animateAttack() { animator.startAnimation(ATTACK_ANIM); + fireSingleSound.play(); } /** diff --git a/source/core/src/main/com/csse3200/game/components/tower/RicochetTowerAnimationController.java b/source/core/src/main/com/csse3200/game/components/tower/RicochetTowerAnimationController.java index 1a197868b..8210e7d73 100644 --- a/source/core/src/main/com/csse3200/game/components/tower/RicochetTowerAnimationController.java +++ b/source/core/src/main/com/csse3200/game/components/tower/RicochetTowerAnimationController.java @@ -1,13 +1,19 @@ package com.csse3200.game.components.tower; +import com.badlogic.gdx.audio.Sound; import com.csse3200.game.components.Component; import com.csse3200.game.rendering.AnimationRenderComponent; +import com.csse3200.game.services.ServiceLocator; /** * This class listens to events relevant to DroidTower entity's state and plays the animation when one * of the events is triggered. */ public class RicochetTowerAnimationController extends Component { + private static final String FIRE_SINGLE_SFX = "sounds/towers/5.56_single_shot.mp3"; + + private final Sound fireSingleSound = ServiceLocator.getResourceService().getAsset( + FIRE_SINGLE_SFX, Sound.class); private AnimationRenderComponent animator; /** @@ -30,6 +36,7 @@ public void create() { */ void animateAttack() { animator.startAnimation("Attack"); + fireSingleSound.play(); } diff --git a/source/core/src/main/com/csse3200/game/components/tower/StunTowerAnimationController.java b/source/core/src/main/com/csse3200/game/components/tower/StunTowerAnimationController.java index 846d344dd..1fc59068a 100644 --- a/source/core/src/main/com/csse3200/game/components/tower/StunTowerAnimationController.java +++ b/source/core/src/main/com/csse3200/game/components/tower/StunTowerAnimationController.java @@ -18,7 +18,10 @@ public class StunTowerAnimationController extends Component { private static final String ATTACK_ANIM = "attack"; private static final String DEATH_ANIM = "death"; - //further sounds can be added for the tower attacks/movement + private static final String FIRE_SINGLE_SFX = "sounds/towers/ar15_single_shot_far.mp3"; + + private final Sound fireSingleSound = ServiceLocator.getResourceService().getAsset( + FIRE_SINGLE_SFX, Sound.class); AnimationRenderComponent animator; @@ -47,6 +50,7 @@ void animateIdle() { */ void animateAttack() { animator.startAnimation(ATTACK_ANIM); + fireSingleSound.play(); } /** diff --git a/source/core/src/main/com/csse3200/game/components/tower/TNTAnimationController.java b/source/core/src/main/com/csse3200/game/components/tower/TNTAnimationController.java index 1f092dab5..595bc6930 100644 --- a/source/core/src/main/com/csse3200/game/components/tower/TNTAnimationController.java +++ b/source/core/src/main/com/csse3200/game/components/tower/TNTAnimationController.java @@ -1,7 +1,9 @@ package com.csse3200.game.components.tower; +import com.badlogic.gdx.audio.Sound; import com.csse3200.game.components.Component; import com.csse3200.game.rendering.AnimationRenderComponent; +import com.csse3200.game.services.ServiceLocator; /** * This class listens to events relevant to TNTTower entity's state and plays the animation when one @@ -9,6 +11,10 @@ */ public class TNTAnimationController extends Component { private AnimationRenderComponent animator; + private static final String EXPLOSION_PATH = "sounds/towers/explosion.mp3"; + + private final Sound explosion = ServiceLocator.getResourceService().getAsset( + EXPLOSION_PATH, Sound.class); /** * Creation call for a TNTAnimationController, fetches the animationRenderComponent that this controller will @@ -43,7 +49,7 @@ void animateDig() { * This method should be invoked when the entity enters the explosion state. */ void animateExplode() { animator.startAnimation("explode"); - + explosion.play(); } } diff --git a/source/core/src/main/com/csse3200/game/components/tower/TowerUpgraderComponent.java b/source/core/src/main/com/csse3200/game/components/tower/TowerUpgraderComponent.java index 4394d6309..a7d07d076 100644 --- a/source/core/src/main/com/csse3200/game/components/tower/TowerUpgraderComponent.java +++ b/source/core/src/main/com/csse3200/game/components/tower/TowerUpgraderComponent.java @@ -38,7 +38,6 @@ public void upgradeTower(UPGRADE upgradeType, int value) { case MAXHP -> upgradeTowerMaxHealth( value); case FIRERATE -> getEntity().getEvents().trigger("addFireRate", value); case REPAIR -> repairTower(); - } } diff --git a/source/core/src/main/com/csse3200/game/components/tower/WallTowerAnimationController.java b/source/core/src/main/com/csse3200/game/components/tower/WallTowerAnimationController.java index 4f22bfaf5..69624ad41 100644 --- a/source/core/src/main/com/csse3200/game/components/tower/WallTowerAnimationController.java +++ b/source/core/src/main/com/csse3200/game/components/tower/WallTowerAnimationController.java @@ -14,7 +14,7 @@ public class WallTowerAnimationController extends Component{ //animation name constants private static final String DEATH_ANIM = "Death"; - private static final String Idle_ANIM = "Idle"; + private static final String IDLE_ANIM = "Idle"; //here we can add the sounds for the implemented animations AnimationRenderComponent animator; @@ -29,13 +29,15 @@ public void create() { entity.getEvents().addListener(DEATH, this::animateDeath); entity.getEvents().addListener(IDLE, this::animateIdle); } + /** * Starts the idle animation. */ void animateDeath() { animator.startAnimation(DEATH_ANIM); } + void animateIdle(){ - animator.startAnimation(Idle_ANIM); + animator.startAnimation(IDLE_ANIM); } } diff --git a/source/core/src/main/com/csse3200/game/currency/Currency.java b/source/core/src/main/com/csse3200/game/currency/Currency.java index 81820ad5f..d3d974a26 100644 --- a/source/core/src/main/com/csse3200/game/currency/Currency.java +++ b/source/core/src/main/com/csse3200/game/currency/Currency.java @@ -4,7 +4,6 @@ import org.slf4j.LoggerFactory; public abstract class Currency { - private static final Logger logger = LoggerFactory.getLogger(Currency.class); // The logo of the currency @@ -53,7 +52,7 @@ public String getTexture() { * @param addedAmount The amount the currency will be incremented by. */ public void modify(int addedAmount) { - logger.debug("Modifying " + this.getClass().getSimpleName() + "by " + addedAmount); + logger.debug(String.format("Modifying %s by %d", this.getClass().getSimpleName(), addedAmount)); this.amount += addedAmount; } diff --git a/source/core/src/main/com/csse3200/game/entities/configs/BaseTowerConfigs.java b/source/core/src/main/com/csse3200/game/entities/configs/BaseTowerConfigs.java new file mode 100644 index 000000000..03530fef3 --- /dev/null +++ b/source/core/src/main/com/csse3200/game/entities/configs/BaseTowerConfigs.java @@ -0,0 +1,105 @@ +package com.csse3200.game.entities.configs; + +/** + * Defines all tower configs to be loaded by the Tower Factory. + */ +public class BaseTowerConfigs { + private WeaponTowerConfig weapon = new WeaponTowerConfig(); + private WallTowerConfig wall = new WallTowerConfig(); + private IncomeTowerConfig income = new IncomeTowerConfig(); + private FireTowerConfig fireTower = new FireTowerConfig(); + private StunTowerConfig stunTower = new StunTowerConfig(); + private TNTTowerConfigs TNTTower = new TNTTowerConfigs(); + private DroidTowerConfig DroidTower = new DroidTowerConfig(); + private FireworksTowerConfig fireworksTower = new FireworksTowerConfig(); + private PierceTowerConfig pierceTower = new PierceTowerConfig(); + private RicochetTowerConfig ricochetTower = new RicochetTowerConfig(); + private HealTowerConfig healTower = new HealTowerConfig(); + + /** + * Function for getting the wall tower's config + * @return The config of wall tower + */ + public WallTowerConfig getWall() { + return wall; + } + + /** + * Function for getting the weapon tower's config + * @return The config of weapon tower + */ + public WeaponTowerConfig getWeapon() { + return weapon; + } + + /** + * Function for getting the income tower's config + * @return The config of income tower + */ + public IncomeTowerConfig getIncome() { + return income; + } + + /** + * Function for getting the fire tower's config + * @return The config of fire tower + */ + public FireTowerConfig getFireTower() { + return fireTower; + } + + /** + * Function for getting the stun tower's config + * @return The config of stun tower + */ + public StunTowerConfig getStunTower() { + return stunTower; + } + + /** + * Function for getting the TNT tower's config + * @return The config of TNT tower */ + public TNTTowerConfigs getTNTTower() { + return TNTTower; + } + + /** + * Function for getting the droid tower's config + * @return The config of droid tower + */ + public DroidTowerConfig getDroidTower() { + return DroidTower; + } + + /** + * Function for getting the fireworks tower's config + * @return The config of fireworks tower + */ + public FireworksTowerConfig getFireworksTower() { + return fireworksTower; + } + + /** + * Function for getting the pierce tower's config + * @return The config of pierce tower + */ + public PierceTowerConfig getPierceTower() { + return pierceTower; + } + + /** + * Function for getting the ricochet tower's config + * @return The config of ricochet tower + */ + public RicochetTowerConfig getRicochetTower() { + return ricochetTower; + } + + /** + * Function for getting the heal tower's config + * @return The config of heal tower + */ + public HealTowerConfig getHealTower() { + return healTower; + } +} \ No newline at end of file diff --git a/source/core/src/main/com/csse3200/game/entities/configs/DroidTowerConfig.java b/source/core/src/main/com/csse3200/game/entities/configs/DroidTowerConfig.java index 3349a149c..804e72d7d 100644 --- a/source/core/src/main/com/csse3200/game/entities/configs/DroidTowerConfig.java +++ b/source/core/src/main/com/csse3200/game/entities/configs/DroidTowerConfig.java @@ -4,9 +4,40 @@ * Defines a basic set of properties stored in entities config files to be loaded by Entity Factories. */ public class DroidTowerConfig { - public int health = 1; - public int baseAttack = 0; - public int cost = 1; + private int health = 1; + private int baseAttack = 0; + private int cost = 1; + private float attackRate = 1; - public float attackRate = 1; + /** + * Function for getting tower's health + * @return The health of this tower + */ + public int getHealth() { + return this.health; + } + + /** + * Function for getting tower's base attack + * @return The base attach of this tower + */ + public int getBaseAttack() { + return this.baseAttack; + } + + /** + * Function for getting tower's cost + * @return The cost of this tower + */ + public int getCost() { + return this.cost; + } + + /** + * Function for getting tower's attack rate + * @return The attack rate of this tower + */ + public float getAttackRate() { + return this.attackRate; + } } diff --git a/source/core/src/main/com/csse3200/game/entities/configs/EngineerConfig.java b/source/core/src/main/com/csse3200/game/entities/configs/EngineerConfig.java index 121fb0b9c..37ab5489e 100644 --- a/source/core/src/main/com/csse3200/game/entities/configs/EngineerConfig.java +++ b/source/core/src/main/com/csse3200/game/entities/configs/EngineerConfig.java @@ -1,6 +1,6 @@ package com.csse3200.game.entities.configs; /** Defines the basic set of properties for an Engineer entity to be loaded by EngineerFactory */ -public class EngineerConfig extends BaseEntityConfig { +public class EngineerConfig { public int health = 1; public int baseAttack = 0; } \ No newline at end of file diff --git a/source/core/src/main/com/csse3200/game/entities/configs/EngineerConfigs.java b/source/core/src/main/com/csse3200/game/entities/configs/EngineerConfigs.java index 21113b7a0..5e6bacdbf 100644 --- a/source/core/src/main/com/csse3200/game/entities/configs/EngineerConfigs.java +++ b/source/core/src/main/com/csse3200/game/entities/configs/EngineerConfigs.java @@ -3,7 +3,7 @@ /** * Defines the properties stored in Engineer config files to be loaded by the Engineer Factory. */ -public class EngineerConfigs extends BaseEntityConfig { +public class EngineerConfigs { public BaseEntityConfig engineer = new BaseEntityConfig(); public int health = 10; public int baseAttack = 1; diff --git a/source/core/src/main/com/csse3200/game/entities/configs/FireTowerConfig.java b/source/core/src/main/com/csse3200/game/entities/configs/FireTowerConfig.java index cc5b75284..48af9db9d 100644 --- a/source/core/src/main/com/csse3200/game/entities/configs/FireTowerConfig.java +++ b/source/core/src/main/com/csse3200/game/entities/configs/FireTowerConfig.java @@ -1,9 +1,40 @@ package com.csse3200.game.entities.configs; public class FireTowerConfig { - public int health = 1; - public int baseAttack = 0; - public int cost = 1; + private int health = 1; + private int baseAttack = 0; + private int cost = 1; + private float attackRate = 1; - public float attackRate = 1; + /** + * Function for getting tower's health + * @return The health of this tower + */ + public int getHealth() { + return this.health; + } + + /** + * Function for getting tower's base attack + * @return The base attach of this tower + */ + public int getBaseAttack() { + return this.baseAttack; + } + + /** + * Function for getting tower's cost + * @return The cost of this tower + */ + public int getCost() { + return this.cost; + } + + /** + * Function for getting tower's attack rate + * @return The attack rate of this tower + */ + public float getAttackRate() { + return this.attackRate; + } } diff --git a/source/core/src/main/com/csse3200/game/entities/configs/FireworksTowerConfig.java b/source/core/src/main/com/csse3200/game/entities/configs/FireworksTowerConfig.java index c1bde5dd3..dd345afae 100644 --- a/source/core/src/main/com/csse3200/game/entities/configs/FireworksTowerConfig.java +++ b/source/core/src/main/com/csse3200/game/entities/configs/FireworksTowerConfig.java @@ -1,9 +1,49 @@ package com.csse3200.game.entities.configs; public class FireworksTowerConfig { - public int health = 1; - public int baseAttack = 0; - public int cost = 1; - public int attackRate =0; - public int incomeRate =0; + private int health = 1; + private int baseAttack = 1; + private int cost = 1; + private int attackRate = 1; + private int incomeRate = 0; + + /** + * Function for getting tower's health + * @return The health of this tower + */ + public int getHealth() { + return this.health; + } + + /** + * Function for getting tower's base attack + * @return The base attach of this tower + */ + public int getBaseAttack() { + return this.baseAttack; + } + + /** + * Function for getting tower's cost + * @return The cost of this tower + */ + public int getCost() { + return this.cost; + } + + /** + * Function for getting tower's attack rate + * @return The attack rate of this tower + */ + public int getAttackRate() { + return this.attackRate; + } + + /** + * Function for getting tower's income rate + * @return The income rate of this tower + */ + public int getIncomeRate() { + return this.incomeRate; + } } diff --git a/source/core/src/main/com/csse3200/game/entities/configs/HealTowerConfig.java b/source/core/src/main/com/csse3200/game/entities/configs/HealTowerConfig.java index 0eb826bcc..c3e5c6f22 100644 --- a/source/core/src/main/com/csse3200/game/entities/configs/HealTowerConfig.java +++ b/source/core/src/main/com/csse3200/game/entities/configs/HealTowerConfig.java @@ -1,8 +1,31 @@ package com.csse3200.game.entities.configs; public class HealTowerConfig { - public int health = 1; - public int baseAttack = 0; - public int cost = 1; - //comment to commit message + private int health = 1; + private int baseAttack = 0; + private int cost = 1; + + /** + * Function for getting tower's health + * @return The health of this tower + */ + public int getHealth() { + return this.health; + } + + /** + * Function for getting tower's base attack + * @return The base attach of this tower + */ + public int getBaseAttack() { + return this.baseAttack; + } + + /** + * Function for getting tower's cost + * @return The cost of this tower + */ + public int getCost() { + return this.cost; + } } diff --git a/source/core/src/main/com/csse3200/game/entities/configs/IncomeTowerConfig.java b/source/core/src/main/com/csse3200/game/entities/configs/IncomeTowerConfig.java index bdcf77c06..ed1500176 100644 --- a/source/core/src/main/com/csse3200/game/entities/configs/IncomeTowerConfig.java +++ b/source/core/src/main/com/csse3200/game/entities/configs/IncomeTowerConfig.java @@ -4,11 +4,49 @@ * Defines a basic set of properties stored in entities config files to be loaded by Entity Factories. */ public class IncomeTowerConfig { - public int health = 1; - public int baseAttack = 0; - public int cost = 1; + private int health = 1; + private int baseAttack = 0; + private int cost = 1; + private float attackRate = 0; + public float incomeRate = 3; - public float attackRate = 0; - public float incomeRate = 10; + /** + * Function for getting tower's health + * @return The health of this tower + */ + public int getHealth() { + return this.health; + } + /** + * Function for getting tower's base attack + * @return The base attach of this tower + */ + public int getBaseAttack() { + return this.baseAttack; + } + + /** + * Function for getting tower's cost + * @return The cost of this tower + */ + public int getCost() { + return this.cost; + } + + /** + * Function for getting tower's attack rate + * @return The attack rate of this tower + */ + public float getAttackRate() { + return this.attackRate; + } + + /** + * Function for getting tower's income rate + * @return The income rate of this tower + */ + public float getIncomeRate() { + return this.incomeRate; + } } diff --git a/source/core/src/main/com/csse3200/game/entities/configs/MobBossConfigs.java b/source/core/src/main/com/csse3200/game/entities/configs/MobBossConfigs.java index 1ae0fc4af..602fbab59 100644 --- a/source/core/src/main/com/csse3200/game/entities/configs/MobBossConfigs.java +++ b/source/core/src/main/com/csse3200/game/entities/configs/MobBossConfigs.java @@ -4,5 +4,4 @@ * Defines the properties stored in mob boss config files to be loaded by the Mob Boss Factory. */ public class MobBossConfigs extends BaseEntityConfig { - public int baseAttack = 0; } diff --git a/source/core/src/main/com/csse3200/game/entities/configs/PierceTowerConfig.java b/source/core/src/main/com/csse3200/game/entities/configs/PierceTowerConfig.java index 8baf1cb52..329b226c2 100644 --- a/source/core/src/main/com/csse3200/game/entities/configs/PierceTowerConfig.java +++ b/source/core/src/main/com/csse3200/game/entities/configs/PierceTowerConfig.java @@ -1,7 +1,40 @@ package com.csse3200.game.entities.configs; public class PierceTowerConfig { - public int health = 1; - public int baseAttack = 0; - public int cost = 1; + private int health = 1; + private int baseAttack = 1; + private int cost = 1; + private float attackRate = 1; + + /** + * Function for getting tower's health + * @return The health of this tower + */ + public int getHealth() { + return this.health; + } + + /** + * Function for getting tower's base attack + * @return The base attach of this tower + */ + public int getBaseAttack() { + return this.baseAttack; + } + + /** + * Function for getting tower's cost + * @return The cost of this tower + */ + public int getCost() { + return this.cost; + } + + /** + * Functino for getting tower's attack rate + * @return The attack rate of this tower + */ + public float getAttackRate() { + return this.attackRate; + } } diff --git a/source/core/src/main/com/csse3200/game/entities/configs/ProjectileConfig.java b/source/core/src/main/com/csse3200/game/entities/configs/ProjectileConfig.java index 757129687..925864cac 100644 --- a/source/core/src/main/com/csse3200/game/entities/configs/ProjectileConfig.java +++ b/source/core/src/main/com/csse3200/game/entities/configs/ProjectileConfig.java @@ -6,8 +6,6 @@ * Configuration for projectiles. */ public class ProjectileConfig extends BaseEntityConfig implements Weapon { - public int health = 1; - public int baseAttack = 0; public int getDamage() { return baseAttack; diff --git a/source/core/src/main/com/csse3200/game/entities/configs/RicochetTowerConfig.java b/source/core/src/main/com/csse3200/game/entities/configs/RicochetTowerConfig.java index 3a637c7e2..07946bc4f 100644 --- a/source/core/src/main/com/csse3200/game/entities/configs/RicochetTowerConfig.java +++ b/source/core/src/main/com/csse3200/game/entities/configs/RicochetTowerConfig.java @@ -1,7 +1,40 @@ package com.csse3200.game.entities.configs; public class RicochetTowerConfig { - public int health = 1; - public int baseAttack = 0; - public int cost = 1; + private int health = 1; + private int baseAttack = 1; + private int cost = 1; + private float attackRate = 1; + + /** + * Function for getting tower's health + * @return The health of this tower + */ + public int getHealth() { + return this.health; + } + + /** + * Function for getting tower's base attack + * @return The base attach of this tower + */ + public int getBaseAttack() { + return this.baseAttack; + } + + /** + * Function for getting tower's cost + * @return The cost of this tower + */ + public int getCost() { + return this.cost; + } + + /** + * Functino for getting tower's attack rate + * @return The attack rate of this tower + */ + public float getAttackRate() { + return this.attackRate; + } } diff --git a/source/core/src/main/com/csse3200/game/entities/configs/StunTowerConfig.java b/source/core/src/main/com/csse3200/game/entities/configs/StunTowerConfig.java index adf01c4f4..a84a1bb9e 100644 --- a/source/core/src/main/com/csse3200/game/entities/configs/StunTowerConfig.java +++ b/source/core/src/main/com/csse3200/game/entities/configs/StunTowerConfig.java @@ -1,9 +1,40 @@ package com.csse3200.game.entities.configs; public class StunTowerConfig { - public int health = 1; - public int baseAttack = 0; - public int cost = 1; + private int health = 1; + private int baseAttack = 0; + private int cost = 1; + private float attackRate = 1; - public float attackRate = 1; + /** + * Function for getting tower's health + * @return The health of this tower + */ + public int getHealth() { + return this.health; + } + + /** + * Function for getting tower's base attack + * @return The base attach of this tower + */ + public int getBaseAttack() { + return this.baseAttack; + } + + /** + * Function for getting tower's cost + * @return The cost of this tower + */ + public int getCost() { + return this.cost; + } + + /** + * Function for getting tower's attack rate + * @return The attack rate of this tower + */ + public float getAttackRate() { + return this.attackRate; + } } diff --git a/source/core/src/main/com/csse3200/game/entities/configs/TNTTowerConfigs.java b/source/core/src/main/com/csse3200/game/entities/configs/TNTTowerConfigs.java index 9385fc512..c6d82c1a1 100644 --- a/source/core/src/main/com/csse3200/game/entities/configs/TNTTowerConfigs.java +++ b/source/core/src/main/com/csse3200/game/entities/configs/TNTTowerConfigs.java @@ -4,9 +4,40 @@ * Defines a basic set of properties stored in entities config files to be loaded by Entity Factories. */ public class TNTTowerConfigs { - public int health = 1; - public int baseAttack = 0; - public int cost = 1; + private int health = 1; + private int baseAttack = 0; + private int cost = 1; + private float attackRate = 1; - public float attackRate = 1; + /** + * Function for getting tower's health + * @return The health of this tower + */ + public int getHealth() { + return this.health; + } + + /** + * Function for getting tower's base attack + * @return The base attach of this tower + */ + public int getBaseAttack() { + return this.baseAttack; + } + + /** + * Function for getting tower's cost + * @return The cost of this tower + */ + public int getCost() { + return this.cost; + } + + /** + * Function for getting tower's attack rate + * @return The attack rate of this tower + */ + public float getAttackRate() { + return this.attackRate; + } } diff --git a/source/core/src/main/com/csse3200/game/entities/configs/WallTowerConfig.java b/source/core/src/main/com/csse3200/game/entities/configs/WallTowerConfig.java index 59c830d4f..82d1f00ee 100644 --- a/source/core/src/main/com/csse3200/game/entities/configs/WallTowerConfig.java +++ b/source/core/src/main/com/csse3200/game/entities/configs/WallTowerConfig.java @@ -4,9 +4,40 @@ * Defines a basic set of properties stored in entities config files to be loaded by Entity Factories. */ public class WallTowerConfig { - public int health = 1; - public int baseAttack = 0; - public int cost = 1; + private int health = 1; + private int baseAttack = 0; + private int cost = 1; + private float attackRate = 0; - public float attackRate = 0; + /** + * Function for getting tower's health + * @return The health of this tower + */ + public int getHealth() { + return this.health; + } + + /** + * Function for getting tower's base attack + * @return The base attach of this tower + */ + public int getBaseAttack() { + return this.baseAttack; + } + + /** + * Function for getting tower's cost + * @return The cost of this tower + */ + public int getCost() { + return this.cost; + } + + /** + * Function for getting tower's attack rate + * @return The attack rate of this tower + */ + public float getAttackRate() { + return this.attackRate; + } } diff --git a/source/core/src/main/com/csse3200/game/entities/configs/WeaponTowerConfig.java b/source/core/src/main/com/csse3200/game/entities/configs/WeaponTowerConfig.java index 7f0547769..b1c123f39 100644 --- a/source/core/src/main/com/csse3200/game/entities/configs/WeaponTowerConfig.java +++ b/source/core/src/main/com/csse3200/game/entities/configs/WeaponTowerConfig.java @@ -4,8 +4,40 @@ * Defines a basic set of properties stored in entities config files to be loaded by Entity Factories. */ public class WeaponTowerConfig { - public int health = 1; - public int baseAttack = 0; - public int cost = 1; - public float attackRate = 1; + private int health = 1; + private int baseAttack = 0; + private int cost = 1; + private float attackRate = 1; + + /** + * Function for getting tower's health + * @return The health of this tower + */ + public int getHealth() { + return this.health; + } + + /** + * Function for getting tower's base attack + * @return The base attach of this tower + */ + public int getBaseAttack() { + return this.baseAttack; + } + + /** + * Function for getting tower's cost + * @return The cost of this tower + */ + public int getCost() { + return this.cost; + } + + /** + * Function for getting tower's attack rate + * @return The attack rate of this tower + */ + public float getAttackRate() { + return this.attackRate; + } } diff --git a/source/core/src/main/com/csse3200/game/entities/configs/baseTowerConfigs.java b/source/core/src/main/com/csse3200/game/entities/configs/baseTowerConfigs.java deleted file mode 100644 index c1b3f997a..000000000 --- a/source/core/src/main/com/csse3200/game/entities/configs/baseTowerConfigs.java +++ /dev/null @@ -1,18 +0,0 @@ -package com.csse3200.game.entities.configs; - -/** - * Defines all tower configs to be loaded by the Tower Factory. - */ -public class baseTowerConfigs { - public WeaponTowerConfig weapon = new WeaponTowerConfig(); - public WallTowerConfig wall = new WallTowerConfig(); - public IncomeTowerConfig income = new IncomeTowerConfig(); - public FireTowerConfig fireTower = new FireTowerConfig(); - public StunTowerConfig stunTower = new StunTowerConfig(); - public TNTTowerConfigs TNTTower = new TNTTowerConfigs(); - public DroidTowerConfig DroidTower = new DroidTowerConfig(); - public FireworksTowerConfig fireworksTower = new FireworksTowerConfig(); - public PierceTowerConfig pierceTower = new PierceTowerConfig(); - public RicochetTowerConfig ricochetTower = new RicochetTowerConfig(); - public HealTowerConfig HealTower = new HealTowerConfig(); -} \ No newline at end of file diff --git a/source/core/src/main/com/csse3200/game/entities/factories/BombshipFactory.java b/source/core/src/main/com/csse3200/game/entities/factories/BombshipFactory.java index b96ad1e59..f2e9a50a1 100644 --- a/source/core/src/main/com/csse3200/game/entities/factories/BombshipFactory.java +++ b/source/core/src/main/com/csse3200/game/entities/factories/BombshipFactory.java @@ -11,7 +11,6 @@ import com.csse3200.game.entities.configs.*; import com.csse3200.game.files.FileLoader; import com.csse3200.game.physics.PhysicsLayer; -import com.csse3200.game.physics.PhysicsUtils; import com.csse3200.game.physics.components.ColliderComponent; import com.csse3200.game.physics.components.HitboxComponent; import com.csse3200.game.physics.components.PhysicsComponent; @@ -20,7 +19,6 @@ /** * Factory to create non-playable human character (NPC) entities with predefined components. - * * These may be modified to become controllable characters in future sprints. * *

Each NPC entity type should have a creation method that returns a corresponding entity. @@ -31,19 +29,16 @@ * similar characteristics. */ public class BombshipFactory { - private static final int COMBAT_TASK_PRIORITY = 2; private static final int BOMBSHIP_RANGE = 30; private static final BombshipConfigs configs = FileLoader.readClass(BombshipConfigs.class, "configs/Bombship.json"); - private static final float HUMAN_SCALE_X = 1f; private static final float HUMAN_SCALE_Y = 0.8f; /** * Creates an Engineer entity, based on a base Human entity, with the appropriate components and animations * - * * @return entity */ public static Entity createBombship() { @@ -75,17 +70,12 @@ public static Entity createBombship() { * @return entity */ public static Entity createBaseshipNPC() { - - - Entity ship = - new Entity() - .addComponent(new PhysicsComponent()) - .addComponent(new PhysicsMovementComponent()) - .addComponent(new ColliderComponent()) - .addComponent(new HitboxComponent().setLayer(PhysicsLayer.BOMBSHIP)) - .addComponent(new TouchAttackComponent(PhysicsLayer.NPC, 1.5f)); - - return ship; + return new Entity() + .addComponent(new PhysicsComponent()) + .addComponent(new PhysicsMovementComponent()) + .addComponent(new ColliderComponent()) + .addComponent(new HitboxComponent().setLayer(PhysicsLayer.BOMBSHIP)) + .addComponent(new TouchAttackComponent(PhysicsLayer.NPC, 1.5f)); } private BombshipFactory() { diff --git a/source/core/src/main/com/csse3200/game/entities/factories/EngineerFactory.java b/source/core/src/main/com/csse3200/game/entities/factories/EngineerFactory.java index 593c12de4..55188a494 100644 --- a/source/core/src/main/com/csse3200/game/entities/factories/EngineerFactory.java +++ b/source/core/src/main/com/csse3200/game/entities/factories/EngineerFactory.java @@ -6,7 +6,6 @@ import com.csse3200.game.components.CombatStatsComponent; import com.csse3200.game.components.EffectComponent; import com.csse3200.game.components.TouchAttackComponent; -import com.csse3200.game.components.npc.EngineerMenuComponent; import com.csse3200.game.components.player.HumanAnimationController; import com.csse3200.game.components.tasks.human.HumanWanderTask; import com.csse3200.game.entities.Entity; @@ -59,7 +58,6 @@ public static Entity createEngineer() { .addComponent(new CombatStatsComponent(config.health, config.baseAttack)) .addComponent(animator) .addComponent(new HumanAnimationController()) - .addComponent(new EngineerMenuComponent()) .addComponent(aiComponent); engineer.getComponent(AITaskComponent.class).addTask(new HumanWanderTask(COMBAT_TASK_PRIORITY, ENGINEER_RANGE)); diff --git a/source/core/src/main/com/csse3200/game/entities/factories/MobBossFactory.java b/source/core/src/main/com/csse3200/game/entities/factories/MobBossFactory.java index f42babb8a..136dba625 100644 --- a/source/core/src/main/com/csse3200/game/entities/factories/MobBossFactory.java +++ b/source/core/src/main/com/csse3200/game/entities/factories/MobBossFactory.java @@ -216,6 +216,7 @@ public static Entity createIceBoss(int health) { * @return a base mob boss entity */ public static Entity createBaseBoss() { + return new Entity() .addComponent(new PhysicsComponent()) // .addComponent(new ColliderComponent()) diff --git a/source/core/src/main/com/csse3200/game/entities/factories/NPCFactory.java b/source/core/src/main/com/csse3200/game/entities/factories/NPCFactory.java index e8038c978..f4dab8d2b 100644 --- a/source/core/src/main/com/csse3200/game/entities/factories/NPCFactory.java +++ b/source/core/src/main/com/csse3200/game/entities/factories/NPCFactory.java @@ -622,10 +622,10 @@ public static Entity createSplittingNightBorne(int health) { public static Entity createDodgingDragonKnight(int health) { Entity dodgeKnight = createDragonKnight(health); - dodgeKnight.addComponent(new DodgingComponent(PhysicsLayer.PROJECTILE, 0.25f)); + dodgeKnight.addComponent(new DodgingComponent(PhysicsLayer.PROJECTILE, 0.25f, 5f)); dodgeKnight.getComponent(AITaskComponent.class).getTask(MobTask.class).setDodge(true); - PhysicsUtils.setScaledCollider(dodgeKnight, 0.3f, 0.7f); - dodgeKnight.setScale(0.3f, 0.7f); + // PhysicsUtils.setScaledCollider(dodgeKnight, 0.3f, 1f); + dodgeKnight.setScale(0.5f, 1.2f); return dodgeKnight; } diff --git a/source/core/src/main/com/csse3200/game/entities/factories/ProjectileFactory.java b/source/core/src/main/com/csse3200/game/entities/factories/ProjectileFactory.java index ce160b65b..b64959976 100644 --- a/source/core/src/main/com/csse3200/game/entities/factories/ProjectileFactory.java +++ b/source/core/src/main/com/csse3200/game/entities/factories/ProjectileFactory.java @@ -343,7 +343,7 @@ public static Entity createBaseProjectile(short targetLayer, Vector2 // This is the component that allows the projectile to damage a // specified target. // Original knockback value: 1.5f - .addComponent(new TouchAttackComponent(targetLayer, 1.5f, true)) + .addComponent(new TouchAttackComponent(targetLayer, 0f, true)) .addComponent(new CombatStatsComponent(config.health, config.baseAttack)) // *TEMPORARY .addComponent(new DeleteOnMapEdgeComponent()); diff --git a/source/core/src/main/com/csse3200/game/entities/factories/TowerFactory.java b/source/core/src/main/com/csse3200/game/entities/factories/TowerFactory.java index b98c5bde0..387d9c59e 100644 --- a/source/core/src/main/com/csse3200/game/entities/factories/TowerFactory.java +++ b/source/core/src/main/com/csse3200/game/entities/factories/TowerFactory.java @@ -1,9 +1,7 @@ package com.csse3200.game.entities.factories; import com.badlogic.gdx.math.Vector2; -import com.badlogic.gdx.physics.box2d.Filter; import com.csse3200.game.components.EffectComponent; -import com.csse3200.game.components.EffectsComponent; import com.csse3200.game.components.tasks.DroidCombatTask; import com.csse3200.game.components.tasks.TNTTowerCombatTask; import com.csse3200.game.components.tasks.*; @@ -23,28 +21,24 @@ import com.csse3200.game.physics.components.PhysicsComponent; import com.csse3200.game.files.FileLoader; import com.csse3200.game.rendering.AnimationRenderComponent; -import com.csse3200.game.rendering.TextureRenderComponent; import com.csse3200.game.services.ServiceLocator; -import com.csse3200.game.input.UpgradeUIComponent;import java.util.HashSet; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import java.util.HashSet; import java.util.Set; - /** * Factory to create a tower entity. * * Predefined tower properties are loaded from a config stored as a json file and should have - * the properties stores in 'baseTowerConfigs'. + * the properties stores in 'BaseTowerConfigs'. */ public class TowerFactory { - // Define a set to keep track of occupied lanes - private static final Set occupiedLanes = new HashSet<>(); private static final int COMBAT_TASK_PRIORITY = 2; private static final int WEAPON_TOWER_MAX_RANGE = 40; private static final int TNT_TOWER_MAX_RANGE = 6; private static final int TNT_TOWER_RANGE = 6; private static final int TNT_KNOCK_BACK_FORCE = 10; - private static final String WALL_IMAGE = "images/towers/wall_tower.png"; - private static final String RESOURCE_TOWER = "images/towers/mine_tower.png"; private static final String TURRET_ATLAS = "images/towers/turret01.atlas"; private static final String FIRE_TOWER_ATLAS = "images/towers/fire_tower_atlas.atlas"; private static final String STUN_TOWER_ATLAS = "images/towers/stun_tower.atlas"; @@ -105,29 +99,29 @@ public class TowerFactory { private static final float RICOCHET_TOWER_ANIM_ATTACK_SPEED = 0.12f; private static final String PIERCE_TOWER_ALERT_ANIM ="Warning"; private static final float PIERCE_TOWER_ANIM_ATTACK_SPEED = 0.12f; - private static final int INCOME_INTERVAL = 300; + // private static final int INCOME_INTERVAL = 300; private static final int INCOME_TASK_PRIORITY = 1; private static final String ECO_ATLAS = "images/economy/econ-tower.atlas"; private static final String ECO_MOVE = "move1"; private static final String ECO_IDLE = "idle"; private static final float ECO_IDLE_SPEED = 0.3f; - private static final baseTowerConfigs configs = - FileLoader.readClass(baseTowerConfigs.class, "configs/tower.json"); + private static final BaseTowerConfigs configs = + FileLoader.readClass(BaseTowerConfigs.class, "configs/tower.json"); + /** * Creates an income tower that generates scrap * @return income */ public static Entity createIncomeTower() { Entity income = createBaseTower(); - IncomeTowerConfig config = configs.income; + IncomeTowerConfig config = configs.getIncome(); // Create the CurrencyIncomeTask and add it to the AITaskComponent - CurrencyTask currencyTask = new CurrencyTask(INCOME_TASK_PRIORITY, (int) config.incomeRate); + CurrencyTask currencyTask = new CurrencyTask(INCOME_TASK_PRIORITY, (int) config.getIncomeRate()); AITaskComponent aiTaskComponent = new AITaskComponent().addTask(currencyTask); - // Contains all the animations that the tower will have AnimationRenderComponent animator = new AnimationRenderComponent( @@ -137,10 +131,10 @@ public static Entity createIncomeTower() { animator.addAnimation(ECO_MOVE, ECO_IDLE_SPEED, Animation.PlayMode.NORMAL); income - .addComponent(new CombatStatsComponent(config.health, config.baseAttack)) - .addComponent(new UpgradableStatsComponent(config.attackRate)) - .addComponent(new CostComponent(config.cost)) - .addComponent(new IncomeUpgradeComponent(config.incomeRate)) + .addComponent(new CombatStatsComponent(config.getHealth(), config.getBaseAttack())) + .addComponent(new UpgradableStatsComponent(config.getAttackRate())) + .addComponent(new CostComponent(config.getCost())) + .addComponent(new IncomeUpgradeComponent(config.getIncomeRate())) .addComponent(aiTaskComponent) .addComponent(animator) .addComponent(new EconTowerAnimationController()); @@ -148,9 +142,13 @@ public static Entity createIncomeTower() { return income; } + /** + * Create wall tower + * @return entity + */ public static Entity createWallTower() { Entity wall = createBaseTower(); - WallTowerConfig config = configs.wall; + WallTowerConfig config = configs.getWall(); AITaskComponent aiTaskComponent = new AITaskComponent() .addTask(new WallTowerDestructionTask(COMBAT_TASK_PRIORITY,TNT_TOWER_MAX_RANGE)); @@ -164,9 +162,9 @@ public static Entity createWallTower() { wall .addComponent(aiTaskComponent) - .addComponent(new CombatStatsComponent(config.health, config.baseAttack)) - .addComponent(new UpgradableStatsComponent(config.attackRate)) - .addComponent(new CostComponent(config.cost)) + .addComponent(new CombatStatsComponent(config.getHealth(), config.getBaseAttack())) + .addComponent(new UpgradableStatsComponent(config.getAttackRate())) + .addComponent(new CostComponent(config.getCost())) .addComponent(animator) .addComponent(new WallTowerAnimationController()); @@ -175,7 +173,6 @@ public static Entity createWallTower() { return wall; } - /** * Create a type of TNT that explodes once it detects a mob within a certain range. * Upon detonation, the TNT will apply both knock-back and health damage to the affected mobs @@ -187,7 +184,7 @@ public static Entity createTNTTower() { .setLayer(PhysicsLayer.NONE) .setSensor(true); TNTTower.getComponent(ColliderComponent.class).setSensor(true); - TNTTowerConfigs config = configs.TNTTower; + TNTTowerConfigs config = configs.getTNTTower(); AITaskComponent aiTaskComponent = new AITaskComponent() .addTask(new TNTTowerCombatTask(COMBAT_TASK_PRIORITY,TNT_TOWER_MAX_RANGE)); @@ -202,15 +199,16 @@ public static Entity createTNTTower() { animator.addAnimation(EXPLODE_ANIM,EXPLODE_SPEED, Animation.PlayMode.NORMAL); TNTTower - .addComponent(new CombatStatsComponent(config.health, config.baseAttack)) - .addComponent(new UpgradableStatsComponent(config.attackRate)) - .addComponent(new CostComponent(config.cost)) + .addComponent(new CombatStatsComponent(config.getHealth(), config.getBaseAttack())) + .addComponent(new UpgradableStatsComponent(config.getAttackRate())) + .addComponent(new CostComponent(config.getCost())) .addComponent(new TNTDamageComponent(PhysicsLayer.NPC,TNT_KNOCK_BACK_FORCE,TNT_TOWER_RANGE)) .addComponent(aiTaskComponent) .addComponent(animator) .addComponent(new TNTAnimationController()); TNTTower.getComponent(AnimationRenderComponent.class).scaleEntity(); + TNTTower.removeComponent(TowerUpgraderComponent.class); return TNTTower; } @@ -222,8 +220,8 @@ public static Entity createTNTTower() { * @return entity */ public static Entity createDroidTower() { - Entity DroidTower = createBaseTower(); - DroidTowerConfig config = configs.DroidTower; + Entity droidTower = createBaseTower(); + DroidTowerConfig config = configs.getDroidTower(); AITaskComponent aiTaskComponent = new AITaskComponent() .addTask(new DroidCombatTask(COMBAT_TASK_PRIORITY, WEAPON_TOWER_MAX_RANGE)); @@ -241,17 +239,15 @@ public static Entity createDroidTower() { animator.addAnimation(GO_UP,DROID_SPEED, Animation.PlayMode.NORMAL); animator.addAnimation(GO_DOWN,DROID_SPEED, Animation.PlayMode.NORMAL); - - - DroidTower - .addComponent(new CombatStatsComponent(config.health, config.baseAttack)) - .addComponent(new UpgradableStatsComponent(config.attackRate)) - .addComponent(new CostComponent(config.cost)) + droidTower + .addComponent(new CombatStatsComponent(config.getHealth(), config.getBaseAttack())) + .addComponent(new UpgradableStatsComponent(config.getAttackRate())) + .addComponent(new CostComponent(config.getCost())) .addComponent(new DroidAnimationController()) .addComponent(animator) .addComponent(aiTaskComponent); - return DroidTower; + return droidTower; } @@ -262,7 +258,7 @@ public static Entity createDroidTower() { */ public static Entity createWeaponTower() { Entity weapon = createBaseTower(); - WeaponTowerConfig config = configs.weapon; + WeaponTowerConfig config = configs.getWeapon(); // AiTaskComponent will run the tower task which carries out detection of targets and trigger events AITaskComponent aiTaskComponent = new AITaskComponent() @@ -279,16 +275,14 @@ public static Entity createWeaponTower() { animator.addAnimation(FIRE_ANIM, (2*FIRE_SPEED), Animation.PlayMode.LOOP); weapon - .addComponent(new CombatStatsComponent(config.health, config.baseAttack)) - .addComponent(new UpgradableStatsComponent(config.attackRate)) - .addComponent(new CostComponent(config.cost)) - .addComponent(new UpgradableStatsComponent(config.attackRate)) + .addComponent(new CombatStatsComponent(config.getHealth(), config.getBaseAttack())) + .addComponent(new UpgradableStatsComponent(config.getAttackRate())) + .addComponent(new CostComponent(config.getCost())) .addComponent(aiTaskComponent) .addComponent(animator) .addComponent(new TowerAnimationController()); return weapon; - } /** @@ -297,7 +291,7 @@ public static Entity createWeaponTower() { */ public static Entity createFireTower() { Entity fireTower = createBaseTower(); - FireTowerConfig config = configs.fireTower; + FireTowerConfig config = configs.getFireTower(); //Component that handles triggering events and animations AITaskComponent aiTaskComponent = new AITaskComponent() @@ -313,12 +307,13 @@ public static Entity createFireTower() { animator.addAnimation(FIRE_TOWER_DEATH_ANIM, FIRE_TOWER_DEATH_SPEED, Animation.PlayMode.NORMAL); fireTower - .addComponent(new CombatStatsComponent(config.health, config.baseAttack)) - .addComponent(new UpgradableStatsComponent(config.attackRate)) - .addComponent(new CostComponent(config.cost)) + .addComponent(new CombatStatsComponent(config.getHealth(), config.getBaseAttack())) + .addComponent(new UpgradableStatsComponent(config.getAttackRate())) + .addComponent(new CostComponent(config.getCost())) .addComponent(aiTaskComponent) .addComponent(animator) .addComponent(new FireTowerAnimationController()); + fireTower.setScale(1.25f, 1.25f); return fireTower; } @@ -329,7 +324,7 @@ public static Entity createFireTower() { */ public static Entity createStunTower() { Entity stunTower = createBaseTower(); - StunTowerConfig config = configs.stunTower; + StunTowerConfig config = configs.getStunTower(); AITaskComponent aiTaskComponent = new AITaskComponent() .addTask(new StunTowerCombatTask(COMBAT_TASK_PRIORITY, WEAPON_TOWER_MAX_RANGE)); @@ -339,13 +334,13 @@ public static Entity createStunTower() { ServiceLocator.getResourceService() .getAsset(STUN_TOWER_ATLAS, TextureAtlas.class)); animator.addAnimation(STUN_TOWER_IDLE_ANIM, STUN_TOWER_IDLE_SPEED, Animation.PlayMode.LOOP); - animator.addAnimation(STUN_TOWER_ATTACK_ANIM, ((STUN_TOWER_ATTACK_SPEED+ 0.20f)), Animation.PlayMode.LOOP); + animator.addAnimation(STUN_TOWER_ATTACK_ANIM, (STUN_TOWER_ATTACK_SPEED+ 0.20f), Animation.PlayMode.LOOP); animator.addAnimation(STUN_TOWER_DEATH_ANIM, STUN_TOWER_DEATH_SPEED, Animation.PlayMode.NORMAL); stunTower - .addComponent(new CombatStatsComponent(config.health, config.baseAttack)) - .addComponent(new UpgradableStatsComponent(config.attackRate)) - .addComponent((new CostComponent(config.cost))) + .addComponent(new CombatStatsComponent(config.getHealth(), config.getBaseAttack())) + .addComponent(new UpgradableStatsComponent(config.getAttackRate())) + .addComponent(new CostComponent(config.getCost())) .addComponent(aiTaskComponent) .addComponent(animator) .addComponent(new StunTowerAnimationController()); @@ -360,12 +355,11 @@ public static Entity createStunTower() { */ public static Entity createFireworksTower() { Entity fireworksTower = createBaseTower(); - FireworksTowerConfig config = configs.fireworksTower; + FireworksTowerConfig config = configs.getFireworksTower(); AITaskComponent aiTaskComponent = new AITaskComponent() .addTask(new FireworksTowerCombatTask(COMBAT_TASK_PRIORITY, WEAPON_TOWER_MAX_RANGE)); - AnimationRenderComponent animator = new AnimationRenderComponent( ServiceLocator.getResourceService() @@ -375,8 +369,9 @@ public static Entity createFireworksTower() { animator.addAnimation(FIREWORKS_TOWER_DEATH_ANIM, FIREWORKS_TOWER_ANIM_SPEED, Animation.PlayMode.NORMAL); fireworksTower - .addComponent(new CombatStatsComponent(config.health, config.baseAttack)) - .addComponent((new CostComponent(config.cost))) + .addComponent(new CombatStatsComponent(config.getHealth(), config.getBaseAttack())) + .addComponent(new UpgradableStatsComponent(config.getAttackRate())) + .addComponent(new CostComponent(config.getCost())) .addComponent(aiTaskComponent) .addComponent(animator) .addComponent(new FireworksTowerAnimationController()); @@ -390,7 +385,7 @@ public static Entity createFireworksTower() { */ public static Entity createPierceTower() { Entity pierceTower = createBaseTower(); - PierceTowerConfig config = configs.pierceTower; + PierceTowerConfig config = configs.getPierceTower(); AITaskComponent aiTaskComponent = new AITaskComponent() .addTask(new PierceTowerCombatTask(COMBAT_TASK_PRIORITY, WEAPON_TOWER_MAX_RANGE)); @@ -404,12 +399,12 @@ public static Entity createPierceTower() { animator.addAnimation(PIERCE_TOWER_DEATH_ANIM, PIERCE_TOWER_ANIM_ATTACK_SPEED, Animation.PlayMode.NORMAL); animator.addAnimation(PIERCE_TOWER_ALERT_ANIM, PIERCE_TOWER_ANIM_ATTACK_SPEED, Animation.PlayMode.NORMAL); - pierceTower .addComponent(animator) .addComponent(new PierceTowerAnimationController()) - .addComponent(new CombatStatsComponent(config.health, config.baseAttack)) - .addComponent((new CostComponent(config.cost))) + .addComponent(new CombatStatsComponent(config.getHealth(), config.getBaseAttack())) + .addComponent(new CostComponent(config.getCost())) + .addComponent(new UpgradableStatsComponent(config.getAttackRate())) .addComponent(aiTaskComponent); pierceTower.setScale(1.5f, 1.5f); @@ -422,7 +417,7 @@ public static Entity createPierceTower() { */ public static Entity createRicochetTower() { Entity ricochetTower = createBaseTower(); - RicochetTowerConfig config = configs.ricochetTower; + RicochetTowerConfig config = configs.getRicochetTower(); AITaskComponent aiTaskComponent = new AITaskComponent() .addTask(new RicochetTowerCombatTask(COMBAT_TASK_PRIORITY, WEAPON_TOWER_MAX_RANGE)); @@ -432,13 +427,14 @@ public static Entity createRicochetTower() { animator.addAnimation(RICOCHET_TOWER_ATTACK_ANIM,(2*RICOCHET_TOWER_ANIM_ATTACK_SPEED),Animation.PlayMode.LOOP); animator.addAnimation(RICOCHET_TOWER_DEATH_ANIM,RICOCHET_TOWER_ANIM_ATTACK_SPEED,Animation.PlayMode.NORMAL); animator.addAnimation(RICOCHET_TOWER_IDLE_ANIM,RICOCHET_TOWER_ANIM_ATTACK_SPEED,Animation.PlayMode.LOOP); + ricochetTower .addComponent(animator) .addComponent(new RicochetTowerAnimationController()) - .addComponent(new CombatStatsComponent(config.health, config.baseAttack)) - .addComponent((new CostComponent(config.cost))) + .addComponent(new CombatStatsComponent(config.getHealth(), config.getBaseAttack())) + .addComponent(new CostComponent(config.getCost())) + .addComponent(new UpgradableStatsComponent(config.getAttackRate())) .addComponent(aiTaskComponent); - // ADD ANIMATION COMPONENTS ricochetTower.setScale(1.5f, 1.5f); return ricochetTower; @@ -451,7 +447,6 @@ public static Entity createRicochetTower() { public static Entity createBaseTower() { // we're going to add more components later on - Entity tower = new Entity() .addComponent(new ColliderComponent()) .addComponent(new EffectComponent(false)) @@ -466,27 +461,4 @@ public static Entity createBaseTower() { tower.getComponent(ColliderComponent.class).setAsBoxAligned(new Vector2(1f, 1f), PhysicsComponent.AlignX.CENTER, PhysicsComponent.AlignY.CENTER); return tower; } - public static Entity createAndPlaceTower(int lane) { - if (isLaneOccupied(lane)) { - System.out.println("Lane " + lane + " is already occupied by a tower"); - return null; - } - - Entity tower = createBaseTower(); - // Customize the tower creation here based on the chosen tower type - - // Add the lane to the set of occupied lanes - occupiedLanes.add(lane); - - return tower; - } - - /** - * Checks if a lane is already occupied by a tower. - * @param lane The lane to check. - * @return True if the lane is occupied, false otherwise. - */ - public static boolean isLaneOccupied(int lane) { - return occupiedLanes.contains(lane); - } } \ No newline at end of file diff --git a/source/core/src/main/com/csse3200/game/entities/factories/WaveFactory.java b/source/core/src/main/com/csse3200/game/entities/factories/WaveFactory.java index 53ea2007a..2bcd6af06 100644 --- a/source/core/src/main/com/csse3200/game/entities/factories/WaveFactory.java +++ b/source/core/src/main/com/csse3200/game/entities/factories/WaveFactory.java @@ -1,5 +1,6 @@ package com.csse3200.game.entities.factories; +import com.badlogic.gdx.math.MathUtils; import com.csse3200.game.ai.tasks.AITaskComponent; import com.csse3200.game.components.tasks.waves.LevelWaves; import com.csse3200.game.components.tasks.waves.WaveClass; @@ -15,70 +16,73 @@ import java.util.HashMap; import java.util.Random; - public class WaveFactory { - /** - * Create a Wave entity. - * Each wave class represents a single wave, then they are appended to a level. - * Cases can be written in here to set what happens for each level. - * - * @return entity - */ + // CONSTANTS + private static final String SKELETON = "Skeleton"; + private static final String COAT = "Coat"; + private static final String WATER_QUEEN = "WaterQueen"; + private static final String SPLITTING_WATER_SLIME = "SplittingWaterSlime"; + private static final String ARCANE_ARCHER = "ArcaneArcher"; + private static final String WIZARD = "Wizard"; + private static final String SPLITTING_NIGHT_BORNE = "SplittingNightBorne"; + private static final String DRAGON_KNIGHT = "DragonKnight"; + private static final String NECROMANCER = "Necromancer"; + private static final String SPLITTING_ROCKY = "SplittingRocky"; + private static final String FIRE_WORM = "FireWorm"; + private static final String DODGING_DRAGON = "DodgingDragon"; + private static final String DEFLECT_FIRE_WIZARD = "DeflectFireWizard"; + private static final String ICE_BOSS = "IceBoss"; + private static final String PATRICK_BOSS = "PatrickBoss"; + private static final String FIRE_BOSS = "FireBoss"; private static final Logger logger = LoggerFactory.getLogger(WaveFactory.class); - private static Random rand = new Random(); private static final ArrayList MELEE_MOBS = new ArrayList<>(Arrays.asList( - "Skeleton", "Coat", "DragonKnight", "Necromancer" + SKELETON, COAT, DRAGON_KNIGHT, NECROMANCER )); - private static final ArrayList> lvl1Structure = new ArrayList<>(Arrays.asList( - new ArrayList<>(Arrays.asList("Coat" - )), new ArrayList<>(Arrays.asList("Coat", "WaterQueen" - )), new ArrayList<>(Arrays.asList("WaterQueen", "SplittingWaterSlime" - )), new ArrayList<>(Arrays.asList("Coat", "WaterQueen", "SplittingWaterSlime" - )) + private static final ArrayList> LVL1_STRUCTURE = new ArrayList<>(Arrays.asList( + new ArrayList<>(Arrays.asList(COAT + )), new ArrayList<>(Arrays.asList(COAT, WATER_QUEEN + )), new ArrayList<>(Arrays.asList(WATER_QUEEN, SPLITTING_WATER_SLIME + )), new ArrayList<>(Arrays.asList(COAT, WATER_QUEEN, SPLITTING_WATER_SLIME + )) )); - private static final ArrayList> lvl2Structure = new ArrayList<>(Arrays.asList( - new ArrayList<>(Arrays.asList("Skeleton" - )), new ArrayList<>(Arrays.asList("Skeleton", "ArcaneArcher" - )), new ArrayList<>(Arrays.asList("Skeleton", "Wizard" - )), new ArrayList<>(Arrays.asList("Skeleton", "SplittingNightBorne" - )), new ArrayList<>(Arrays.asList("Wizard", "SplittingNightBorne" - )), new ArrayList<>(Arrays.asList("SplittingNightBorne", "Skeleton" - )), new ArrayList<>(Arrays.asList("Wizard", "SplittingNightBorne" - )), new ArrayList<>(Arrays.asList("ArcaneArcher", "SplittingNightBorne", "Wizard" - )), new ArrayList<>(Arrays.asList("Skeleton", "ArcaneArcher", "Wizard", "SplittingNightBorne" - )) + private static final ArrayList> LVL2_STRUCTURE = new ArrayList<>(Arrays.asList( + new ArrayList<>(Arrays.asList(SKELETON + )), new ArrayList<>(Arrays.asList(SKELETON, ARCANE_ARCHER + )), new ArrayList<>(Arrays.asList(SKELETON, WIZARD + )), new ArrayList<>(Arrays.asList(SKELETON, SPLITTING_NIGHT_BORNE + )), new ArrayList<>(Arrays.asList(WIZARD, SPLITTING_NIGHT_BORNE + )), new ArrayList<>(Arrays.asList(SPLITTING_NIGHT_BORNE, SKELETON + )), new ArrayList<>(Arrays.asList(WIZARD, SPLITTING_NIGHT_BORNE + )), new ArrayList<>(Arrays.asList(ARCANE_ARCHER, SPLITTING_NIGHT_BORNE, WIZARD + )), new ArrayList<>(Arrays.asList(SKELETON, ARCANE_ARCHER, WIZARD, SPLITTING_NIGHT_BORNE + )) )); - private static final ArrayList> lvl3Structure = new ArrayList<>(Arrays.asList( - new ArrayList<>(Arrays.asList("Necromancer" - )), new ArrayList<>(Arrays.asList("Necromancer", "DodgingDragon" - )), new ArrayList<>(Arrays.asList("Necromancer", "FireWorm" - )), new ArrayList<>(Arrays.asList("Necromancer", "DeflectFireWizard" - )), new ArrayList<>(Arrays.asList("DeflectFireWizard", "FireWorm" - )), new ArrayList<>(Arrays.asList("DodgingDragon", "FireWorm" - )), new ArrayList<>(Arrays.asList("DodgingDragon", "Necromancer" - )), new ArrayList<>(Arrays.asList("FireWorm", "Necromancer" - )), new ArrayList<>(Arrays.asList("DeflectFireWizard", "Necromancer" - )), new ArrayList<>(Arrays.asList("DodgingDragon", "DeflectFireWizard", "Necromancer" - )), new ArrayList<>(Arrays.asList("FireWorm", "Necromancer", "DodgingDragon" - )), new ArrayList<>(Arrays.asList("FireWorm", "SplittingRocky", "Necromancer" - )), new ArrayList<>(Arrays.asList("SplittingRocky", "DeflectFireWizard", "FireWorm" - )), new ArrayList<>(Arrays.asList("DeflectFireWizard", "SplittingRocky", "Necromancer", "DodgingDragon", "FireWorm" - )) + private static final ArrayList> LVL3_STRUCTURE = new ArrayList<>(Arrays.asList( + new ArrayList<>(Arrays.asList(NECROMANCER + )), new ArrayList<>(Arrays.asList(NECROMANCER, DODGING_DRAGON + )), new ArrayList<>(Arrays.asList(NECROMANCER, FIRE_WORM + )), new ArrayList<>(Arrays.asList(NECROMANCER, DEFLECT_FIRE_WIZARD + )), new ArrayList<>(Arrays.asList(DEFLECT_FIRE_WIZARD, FIRE_WORM + )), new ArrayList<>(Arrays.asList(DODGING_DRAGON, FIRE_WORM + )), new ArrayList<>(Arrays.asList(DODGING_DRAGON, NECROMANCER + )), new ArrayList<>(Arrays.asList(FIRE_WORM, NECROMANCER + )), new ArrayList<>(Arrays.asList(DEFLECT_FIRE_WIZARD, NECROMANCER + )), new ArrayList<>(Arrays.asList(DODGING_DRAGON, DEFLECT_FIRE_WIZARD, NECROMANCER + )), new ArrayList<>(Arrays.asList(FIRE_WORM, NECROMANCER, DODGING_DRAGON + )), new ArrayList<>(Arrays.asList(FIRE_WORM, SPLITTING_ROCKY, NECROMANCER + )), new ArrayList<>(Arrays.asList(SPLITTING_ROCKY, DEFLECT_FIRE_WIZARD, FIRE_WORM + )), new ArrayList<>(Arrays.asList(DEFLECT_FIRE_WIZARD, SPLITTING_ROCKY, NECROMANCER, DODGING_DRAGON, FIRE_WORM + )) )); - private static final String BOSS_1 = "IceBoss"; - private static final String BOSS_2 = "PatrickBoss"; - private static final String BOSS_3 = "FireBoss"; - /** * The function will create the waves depending on the level selected by the user. - * */ + */ public static Entity createWaves() { - int chosenLevel = GameLevelData.getSelectedLevel(); int difficulty; switch (chosenLevel) { @@ -93,9 +97,7 @@ public static Entity createWaves() { } LevelWaves level = createLevel(difficulty); - AITaskComponent aiComponent = - new AITaskComponent() - .addTask(new WaveTask()); + AITaskComponent aiComponent = new AITaskComponent().addTask(new WaveTask()); return level.addComponent(aiComponent); } @@ -130,32 +132,31 @@ public static LevelWaves createLevel(int chosenLevel) { int bossHealth; int minMobs; // Base health of the bosses - int LVL1_BOSS_BASE_HEALTH = 500; - int LVL2_BOSS_BASE_HEALTH = 1000; - int LVL3_BOSS_BASE_HEALTH = 2000; + final int LVL1_BOSS_BASE_HEALTH = 500; + final int LVL2_BOSS_BASE_HEALTH = 1000; + final int LVL3_BOSS_BASE_HEALTH = 2000; switch (chosenLevel) { case 2: - boss = BOSS_2; + boss = PATRICK_BOSS; bossHealth = LVL2_BOSS_BASE_HEALTH; - possibleMobs = lvl2Structure; + possibleMobs = LVL2_STRUCTURE; minMobs = 6; break; case 3: - boss = BOSS_3; + boss = FIRE_BOSS; bossHealth = LVL3_BOSS_BASE_HEALTH; - possibleMobs = lvl3Structure; + possibleMobs = LVL3_STRUCTURE; minMobs = 8; break; default: - boss = BOSS_1; + boss = ICE_BOSS; bossHealth = LVL1_BOSS_BASE_HEALTH; - possibleMobs = lvl1Structure; + possibleMobs = LVL1_STRUCTURE; minMobs = 5; break; } -// int totalMobs = 0; // Create mxWaves number of waves with mob stats increasing int atWave = 1; for (ArrayList wave : possibleMobs) { @@ -171,26 +172,22 @@ public static LevelWaves createLevel(int chosenLevel) { // Calculate the number of mobs for the wave if (leftToSort == 0) { num = minMobs - currentMobs; - System.out.println(num + " for " + mob + " at wave " + atWave); } else { - num = rand.nextInt(minMobs - currentMobs - (2 * leftToSort) - 2) + 2; + num = MathUtils.random(minMobs - currentMobs - (2 * leftToSort) - 2) + 2; System.out.println(num + " for " + mob + " at wave " + atWave); currentMobs += num; } // Calculate the health - int RANGE_BASE_HEALTH = 60; - int health = RANGE_BASE_HEALTH; + int health = 60; if (MELEE_MOBS.contains(mob)) { // The base health for the different mobs - int MELEE_BASE_HEALTH = 80; - health = MELEE_BASE_HEALTH; + health = 80; } int[] mobStats = {num, health + (atWave * chosenLevel)}; mobs.put(mob, mobStats); leftToSort --; -// totalMobs += num; } minMobs ++; level.addWave(new WaveClass(mobs)); @@ -200,9 +197,7 @@ public static LevelWaves createLevel(int chosenLevel) { // Add boss wave HashMap bossMob = new HashMap<>(); bossMob.put(boss, new int[]{1, bossHealth}); -// totalMobs ++; -// ServiceLocator.getWaveService().setTotalMobs(totalMobs); level.addWave(new WaveClass(bossMob)); @@ -214,3 +209,4 @@ private WaveFactory() { throw new IllegalStateException("Instantiating static util class"); } } + diff --git a/source/core/src/main/com/csse3200/game/input/DropInputComponent.java b/source/core/src/main/com/csse3200/game/input/DropInputComponent.java index df9ed2146..6ea786348 100644 --- a/source/core/src/main/com/csse3200/game/input/DropInputComponent.java +++ b/source/core/src/main/com/csse3200/game/input/DropInputComponent.java @@ -3,27 +3,21 @@ import com.badlogic.gdx.graphics.Camera; import com.badlogic.gdx.math.Vector2; import com.badlogic.gdx.math.Vector3; -import com.csse3200.game.areas.ForestGameArea; import com.csse3200.game.components.npc.DropComponent; import com.csse3200.game.entities.Entity; import com.csse3200.game.entities.EntityService; import com.csse3200.game.services.ServiceLocator; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; import java.util.Objects; public class DropInputComponent extends InputComponent { - private static final Logger logger = LoggerFactory.getLogger(ForestGameArea.class); private final EntityService entityService; private final Camera camera; - int value; /** * Constructor for the DropInputComponent * @param camera the camera to be used, this is the camera that the game is rendered with */ public DropInputComponent(Camera camera) { - this.value = ServiceLocator.getCurrencyService().getScrap().getAmount(); this.entityService = ServiceLocator.getEntityService(); this.camera = camera; } @@ -45,15 +39,14 @@ public Camera getCamera() { * @param screenY The y coordinate, origin is in the upper left corner * @param pointer the pointer for the event. * @param button the button - * @return + * @return true if the event was handled; false otherwise */ @Override public boolean touchDown(int screenX, int screenY, int pointer, int button) { - Vector3 worldCoordinates = new Vector3((float) screenX , (float) screenY, 0); + Vector3 worldCoordinates = new Vector3(screenX , screenY, 0); getCamera().unproject(worldCoordinates); // translate from screen to world coordinates Vector2 cursorPosition = new Vector2(worldCoordinates.x, worldCoordinates.y); Entity clickedEntity = entityService.getEntityAtPosition(cursorPosition.x, cursorPosition.y); - //logger.info("Clicked entity: " + clickedEntity); if (clickedEntity != null && clickedEntity.getComponent(DropComponent.class) != null) { int value = clickedEntity.getComponent(DropComponent.class).getValue(); @@ -69,15 +62,13 @@ public boolean touchDown(int screenX, int screenY, int pointer, int button) { ServiceLocator.getCurrencyService().getDisplay().updateCrystalsStats(); } - float X = clickedEntity.getCenterPosition().x; - float Y = clickedEntity.getCenterPosition().y; + float xValue = clickedEntity.getCenterPosition().x; + float yValue = clickedEntity.getCenterPosition().y; // remove the entity from the game EntityService.removeEntity(clickedEntity); // display a visual indication that currency has been picked up - ServiceLocator.getCurrencyService().getDisplay().currencyPopUp(X, Y, value, 10); - - //logger.info("Scrap amount: " + ServiceLocator.getCurrencyService().getScrap().getAmount()); + ServiceLocator.getCurrencyService().getDisplay().currencyPopUp(xValue, yValue, value, 10); return true; } return false; diff --git a/source/core/src/main/com/csse3200/game/input/EngineerInputComponent.java b/source/core/src/main/com/csse3200/game/input/EngineerInputComponent.java index 4b2388878..75a37bb07 100644 --- a/source/core/src/main/com/csse3200/game/input/EngineerInputComponent.java +++ b/source/core/src/main/com/csse3200/game/input/EngineerInputComponent.java @@ -6,15 +6,10 @@ import com.badlogic.gdx.math.Vector2; import com.badlogic.gdx.math.Vector3; import com.csse3200.game.ai.tasks.AITaskComponent; -import com.csse3200.game.ai.tasks.PriorityTask; -import com.csse3200.game.ai.tasks.Task; -import com.csse3200.game.components.npc.EngineerMenuComponent; import com.csse3200.game.components.player.HumanAnimationController; -import com.csse3200.game.components.tasks.human.HumanMovementTask; import com.csse3200.game.components.tasks.human.HumanWanderTask; import com.csse3200.game.entities.Entity; import com.csse3200.game.entities.EntityService; -import com.csse3200.game.entities.factories.EngineerFactory; import com.csse3200.game.physics.PhysicsLayer; import com.csse3200.game.rendering.AnimationRenderComponent; import com.csse3200.game.services.ServiceLocator; @@ -28,8 +23,9 @@ public class EngineerInputComponent extends InputComponent { private Camera camera; private EntityService entityService; - private Entity selectedEngineer = null; - private boolean moveClicked = false; + public Entity selectedEngineer = null; + + private final String OUTLINE_STRING = "_outline"; public EngineerInputComponent(Game game, Camera camera) { this.game = game; @@ -37,8 +33,9 @@ public EngineerInputComponent(Game game, Camera camera) { this.entityService = ServiceLocator.getEntityService(); } + @Override public boolean touchDown(int screenX, int screenY, int pointer, int button) { - Vector3 worldCoordinates = new Vector3((float) screenX , (float) screenY, 0); + Vector3 worldCoordinates = new Vector3(screenX , screenY, 0); camera.unproject(worldCoordinates); Vector2 cursorPosition = new Vector2(worldCoordinates.x, worldCoordinates.y); camera.project(worldCoordinates); @@ -47,43 +44,17 @@ public boolean touchDown(int screenX, int screenY, int pointer, int button) { // Case when engineer is not clicked if (engineer == null || engineer.getComponent(HumanAnimationController.class) == null) { - if (selectedEngineer != null && moveClicked) { + if (selectedEngineer != null) { + // Clicked a tile with an engineer selected and clicked on not an engineer moveEngineer(cursorPosition); - selectedEngineer = null; - moveClicked = false; return true; } else { + // Clicked a tile with no engineer selected or engineer on the tile return false; } } // Case when engineer is clicked - AnimationRenderComponent animator = engineer.getComponent(AnimationRenderComponent.class); - String currentAnimation = animator.getCurrentAnimation(); - HumanAnimationController controller = engineer.getComponent(HumanAnimationController.class); - EngineerMenuComponent menu = engineer.getComponent(EngineerMenuComponent.class); - - if (engineer.equals(selectedEngineer)) { - // Deselect the engineer by clicking on itself - this.getWanderTask().setSelected(false); - selectedEngineer = null; - moveClicked = false; - if (currentAnimation.contains("_outline")) { - controller.deselectEngineer(currentAnimation); - //logger.info("Engineer deselected"); - } - } else { - this.selectedEngineer = engineer; - this.getWanderTask().setSelected(true); - moveClicked = false; - logger.info("Engineer size: {}", engineer.getScale()); - - // outline image if it is not already outlined and vice versa - if (!currentAnimation.contains("_outline")) { - animator.startAnimation(currentAnimation + "_outline"); - menu.createMenu(cursorPosition.x, cursorPosition.y, camera); - controller.setClicked(true); - } - } + switchEngineer(engineer); return true; } @@ -106,12 +77,65 @@ private void manualShoot() { wander.startCombat(); } + /** + * Switches the specified engineer + * If the given engineer is already selected, deselect it + * If another engineer is selected, deselect it and select the new given engineer + * @param engineer (Entity) the specified engineer + */ + private void switchEngineer(Entity engineer) { + if (engineer.equals(this.selectedEngineer)) { + this.getWanderTask().setSelected(false); + this.selectedEngineer = null; + switchOutline(engineer); + } + else if (selectedEngineer == null) { + this.selectedEngineer = engineer; + switchOutline(engineer); + this.getWanderTask().setSelected(true); + + } else { + this.getWanderTask().setSelected(false); + switchOutline(this.selectedEngineer); + switchOutline(engineer); + this.selectedEngineer = engineer; + this.getWanderTask().setSelected(true); + + } + } + + /** + * Switches the outline of the given engineer and deselects/selects engineer as needed + * If outlined -> remove outline, deselect engineer and vice versa + * @param engineer (Entity) the specified engineer + */ + private void switchOutline(Entity engineer) { + AnimationRenderComponent animator = engineer.getComponent(AnimationRenderComponent.class); + String currentAnimation = animator.getCurrentAnimation(); + HumanAnimationController controller = engineer.getComponent(HumanAnimationController.class); + if (currentAnimation.contains(OUTLINE_STRING)) { + animator.startAnimation(currentAnimation.substring(0, currentAnimation.lastIndexOf('_'))); + controller.setClicked(false); + } else { + animator.startAnimation(currentAnimation + OUTLINE_STRING); + controller.setClicked(true); + } + } + + /** + * Returns the wander task of the selected engineer + * @return (HumanWanderTask) the wander task of the selected engineer + */ private HumanWanderTask getWanderTask() { AITaskComponent movementTask = selectedEngineer.getComponent(AITaskComponent.class); return movementTask.getTask(HumanWanderTask.class); } - private void moveEngineer(Vector2 cursorPosition) { + /** + * Moves the selected engineer to the given cursor position + * @param cursorPosition (Vector2) the cursor position + */ + public void moveEngineer(Vector2 cursorPosition) { if (selectedEngineer == null) { logger.info("Trying to move an engineer that is not selected"); } @@ -123,9 +147,4 @@ private void moveEngineer(Vector2 cursorPosition) { Vector2 dest = cursorPosition.add(offset); wander.startMoving(dest); } - - public void setMoveClicked(boolean moveClicked) { - this.moveClicked = moveClicked; - } - } diff --git a/source/core/src/main/com/csse3200/game/input/UpgradeUIComponent.java b/source/core/src/main/com/csse3200/game/input/UpgradeUIComponent.java index 9d4c42f50..f58199c4a 100644 --- a/source/core/src/main/com/csse3200/game/input/UpgradeUIComponent.java +++ b/source/core/src/main/com/csse3200/game/input/UpgradeUIComponent.java @@ -1,5 +1,6 @@ package com.csse3200.game.input; +import com.badlogic.gdx.Gdx; import com.badlogic.gdx.graphics.Camera; import com.badlogic.gdx.graphics.Color; import com.badlogic.gdx.graphics.Texture; @@ -19,7 +20,7 @@ import com.badlogic.gdx.utils.Scaling; import com.csse3200.game.ai.tasks.AITaskComponent; import com.csse3200.game.ai.tasks.PriorityTask; -import com.csse3200.game.areas.ForestGameArea; +import com.csse3200.game.areas.*; import com.csse3200.game.components.CombatStatsComponent; import com.csse3200.game.components.tasks.TowerCombatTask; import com.csse3200.game.components.tower.*; @@ -33,7 +34,25 @@ import java.util.Iterator; import java.util.Map; +import static java.lang.Math.round; + public class UpgradeUIComponent extends InputComponent { + + // CONSTANTS + + /** + * The cost for all upgrades are 10 crystals + */ + private static final int UPGRADE_COST = 10; // Crystal + + /** + * The cost for repairing a turret is 50 scrap + */ + private static final int REPAIR_COST = 50; // Scrap + private static final float ATTACK_RATE_INCREASE = 0.2f; + private static final int ATTACK_INCREASE = 10; // Damage + private static final int HEALTH_INCREASE = 10; // Health + private static final int TIME_DECREASE = 5; // Scrap private static final Logger logger = LoggerFactory.getLogger(ForestGameArea.class); private final EntityService entityService; private final Camera camera; @@ -85,21 +104,10 @@ public boolean touchDown(int screenX, int screenY, int pointer, int button) { Vector2 cursorPosition = new Vector2(worldCoordinates.x, worldCoordinates.y); Entity clickedEntity = entityService.getEntityAtPosition(cursorPosition.x, cursorPosition.y); - //temp fix to prevent upgrading of new towers - if (clickedEntity!= null && (clickedEntity.getComponent(RicochetTowerAnimationController.class) != null || - clickedEntity.getComponent(PierceTowerAnimationController.class) != null || - clickedEntity.getComponent(FireworksTowerAnimationController.class) != null)) { - return false; - } - // - // If the clicked position contains a turret, and the turret is upgradable and not a TNT tower if (clickedEntity != null && clickedEntity.getComponent(TowerUpgraderComponent.class) != null - && clickedEntity.getComponent(TNTDamageComponent.class) == null) { - // TNT TowerUpgraderComponent can be removed later, but possibly useful for future sprint. - // Clear all existing upgrade tables - logger.info("clickedEntity: " + clickedEntity); - + ) { +// clearUpgradeTables(); // Check if there is an existing upgrade table for this turret entity Table existingUpgradeTable = upgradeTables.get(clickedEntity); @@ -110,8 +118,6 @@ public boolean touchDown(int screenX, int screenY, int pointer, int button) { } else { // If no upgrade table exists, create and store a new one Table newUpgradeTable = createUpgradeTable(clickedEntity); - Vector2 UICoordinates = stage.screenToStageCoordinates(new Vector2(screenX, screenY)); - newUpgradeTable.setPosition(UICoordinates.x, UICoordinates.y); stage.addActor(newUpgradeTable); // Store the new upgrade table in the map @@ -149,9 +155,11 @@ private void clearUpgradeTables() { private Table createUpgradeTable(Entity turretEntity) { // This is the overarching table that contains the close button, the inner table, and the cost display Table upgradeTable = new Table(); - upgradeTable.top(); + upgradeTable.top().left(); upgradeTable.defaults().pad(0).space(0); upgradeTable.setSize(60, 60); + upgradeTable.padTop(85f).padLeft(5f); + upgradeTable.setPosition(0, round((float) Gdx.graphics.getHeight() / 1.3f)); // The inner table contains the upgrade buttons and the stats display Table innerUpgradeTable = new Table(); @@ -180,11 +188,12 @@ private Table createUpgradeTable(Entity turretEntity) { costDisplay.setWidth(0); costDisplay.setBackground(drawableBackground); // Create an Image for the scrap icon - Drawable costDrawable = new TextureRegionDrawable(new TextureRegion(new Texture("images/economy/scrap.png"))); + Drawable costDrawable = new TextureRegionDrawable(new TextureRegion(new Texture("images/economy/crystal.png"))); + Drawable costDrawableScrap = new TextureRegionDrawable(new TextureRegion(new Texture("images/economy/scrap.png"))); Image costImage = new Image(costDrawable); costDisplay.add(costImage).center(); costImage.setScaling(Scaling.none); - Label costDisplayLabel = new Label("100", createLabelStyle()); + Label costDisplayLabel = new Label("You shouldn't see this", createLabelStyle()); costDisplay.add(costDisplayLabel).padLeft(0); TextButton closeButton = new TextButton("X", style); @@ -217,14 +226,15 @@ public void clicked(InputEvent event, float x, float y) { upgradeHealth.addListener(new ClickListener() { @Override public void clicked(InputEvent event, float x, float y) { - value = ServiceLocator.getCurrencyService().getScrap().getAmount(); + value = ServiceLocator.getCurrencyService().getCrystal().getAmount(); logger.info("clicked"); - if (value >= 100) { - value -= 100; - ServiceLocator.getCurrencyService().getScrap().setAmount(value); - ServiceLocator.getCurrencyService().getDisplay().updateScrapsStats(); + if (value >= UPGRADE_COST) { + value -= UPGRADE_COST; + ServiceLocator.getCurrencyService().getCrystal().setAmount(value); + ServiceLocator.getCurrencyService().getDisplay().updateCrystalsStats(); - turretEntity.getComponent(TowerUpgraderComponent.class).upgradeTower(TowerUpgraderComponent.UPGRADE.MAXHP, 10); + turretEntity.getComponent(TowerUpgraderComponent.class) + .upgradeTower(TowerUpgraderComponent.UPGRADE.MAXHP, HEALTH_INCREASE); int currentHealth = turretEntity.getComponent(CombatStatsComponent.class).getHealth(); int maxHealth = turretEntity.getComponent(CombatStatsComponent.class).getMaxHealth(); healthLabel.setText(String.format("%d/%d", currentHealth, maxHealth)); @@ -232,7 +242,7 @@ public void clicked(InputEvent event, float x, float y) { } @Override public void enter(InputEvent event, float x, float y, int pointer, Actor fromActor) { - costDisplayLabel.setText("10"); + costDisplayLabel.setText(String.format("%d", UPGRADE_COST)); costDisplay.setVisible(true); } @Override @@ -247,12 +257,13 @@ public void exit(InputEvent event, float x, float y, int pointer, Actor toActor) upgradeAttack.addListener(new ClickListener() { @Override public void clicked(InputEvent event, float x, float y) { - value = ServiceLocator.getCurrencyService().getScrap().getAmount(); - if (value >= 10) { - value -= 10; - ServiceLocator.getCurrencyService().getScrap().setAmount(value); - ServiceLocator.getCurrencyService().getDisplay().updateScrapsStats(); - turretEntity.getComponent(TowerUpgraderComponent.class).upgradeTower(TowerUpgraderComponent.UPGRADE.ATTACK, 5); + value = ServiceLocator.getCurrencyService().getCrystal().getAmount(); + if (value >= UPGRADE_COST) { + value -= UPGRADE_COST; + ServiceLocator.getCurrencyService().getCrystal().setAmount(value); + ServiceLocator.getCurrencyService().getDisplay().updateCrystalsStats(); + turretEntity.getComponent(TowerUpgraderComponent.class) + .upgradeTower(TowerUpgraderComponent.UPGRADE.ATTACK, ATTACK_INCREASE); int attack = turretEntity.getComponent(CombatStatsComponent.class).getBaseAttack(); attackLabel.setText(String.format("%d", attack)); @@ -260,7 +271,7 @@ public void clicked(InputEvent event, float x, float y) { } @Override public void enter(InputEvent event, float x, float y, int pointer, Actor fromActor) { - costDisplayLabel.setText("10"); + costDisplayLabel.setText(String.format("%d", UPGRADE_COST)); costDisplay.setVisible(true); } @Override @@ -276,14 +287,18 @@ public void exit(InputEvent event, float x, float y, int pointer, Actor toActor) upgradeFireRate.addListener(new ClickListener() { @Override public void clicked(InputEvent event, float x, float y) { - value = ServiceLocator.getCurrencyService().getScrap().getAmount(); - if (value >= 10) { - value -= 10; - ServiceLocator.getCurrencyService().getScrap().setAmount(value); - ServiceLocator.getCurrencyService().getDisplay().updateScrapsStats(); - float newFireRate = turretEntity.getComponent(UpgradableStatsComponent.class).getAttackRate() + 0.2f; - turretEntity.getComponent(UpgradableStatsComponent.class).setAttackRate(newFireRate); - turretEntity.getComponent(TowerUpgraderComponent.class).upgradeTower(TowerUpgraderComponent.UPGRADE.FIRERATE, (int) newFireRate * 5); + value = ServiceLocator.getCurrencyService().getCrystal().getAmount(); + if (value >= UPGRADE_COST) { + value -= UPGRADE_COST; + ServiceLocator.getCurrencyService().getCrystal().setAmount(value); + ServiceLocator.getCurrencyService().getDisplay().updateCrystalsStats(); + float newFireRate = turretEntity.getComponent(UpgradableStatsComponent.class) + .getAttackRate() + ATTACK_RATE_INCREASE; + turretEntity.getComponent(UpgradableStatsComponent.class) + .setAttackRate(newFireRate); + turretEntity.getComponent(TowerUpgraderComponent.class) + .upgradeTower(TowerUpgraderComponent.UPGRADE.FIRERATE, + (int) newFireRate * 5); float fireRate = turretEntity.getComponent(UpgradableStatsComponent.class).getAttackRate(); fireRateLabel.setText(String.format("%.2f", fireRate)); @@ -293,7 +308,7 @@ public void clicked(InputEvent event, float x, float y) { @Override public void enter(InputEvent event, float x, float y, int pointer, Actor fromActor) { - costDisplayLabel.setText("10"); + costDisplayLabel.setText(String.format("%d", UPGRADE_COST)); costDisplay.setVisible(true); } @Override @@ -309,11 +324,12 @@ public void exit(InputEvent event, float x, float y, int pointer, Actor toActor) @Override public void clicked(InputEvent event, float x, float y) { value = ServiceLocator.getCurrencyService().getScrap().getAmount(); - if (value >= 100) { - value -= 100; + if (value >= REPAIR_COST) { + value -= REPAIR_COST; ServiceLocator.getCurrencyService().getScrap().setAmount(value); ServiceLocator.getCurrencyService().getDisplay().updateScrapsStats(); - turretEntity.getComponent(TowerUpgraderComponent.class).upgradeTower(TowerUpgraderComponent.UPGRADE.REPAIR, 0); + turretEntity.getComponent(TowerUpgraderComponent.class) + .upgradeTower(TowerUpgraderComponent.UPGRADE.REPAIR, 0); int currentHealth = turretEntity.getComponent(CombatStatsComponent.class).getHealth(); healthLabel.setText(String.format("%d/%d", currentHealth, maxHealth)); } @@ -321,11 +337,13 @@ public void clicked(InputEvent event, float x, float y) { @Override public void enter(InputEvent event, float x, float y, int pointer, Actor fromActor) { - costDisplayLabel.setText("10"); + costDisplayLabel.setText(String.format("%d", REPAIR_COST)); + costImage.setDrawable(costDrawableScrap); costDisplay.setVisible(true); } @Override public void exit(InputEvent event, float x, float y, int pointer, Actor toActor) { + costImage.setDrawable(costDrawable); costDisplay.setVisible(false); } @@ -353,12 +371,12 @@ public void exit(InputEvent event, float x, float y, int pointer, Actor toActor) upgradeIncome.addListener(new ClickListener() { @Override public void clicked(InputEvent event, float x, float y) { - value = ServiceLocator.getCurrencyService().getScrap().getAmount(); - if (value >= 10 && turretEntity.getComponent(IncomeUpgradeComponent.class).getIncomeRate() >= 10) { - value -= 10; - ServiceLocator.getCurrencyService().getScrap().setAmount(value); - ServiceLocator.getCurrencyService().getDisplay().updateScrapsStats(); - float newIncome = turretEntity.getComponent(IncomeUpgradeComponent.class).getIncomeRate() - 5; + value = ServiceLocator.getCurrencyService().getCrystal().getAmount(); + if (value >= UPGRADE_COST && turretEntity.getComponent(IncomeUpgradeComponent.class).getIncomeRate() >= 10) { + value -= UPGRADE_COST; + ServiceLocator.getCurrencyService().getCrystal().setAmount(value); + ServiceLocator.getCurrencyService().getDisplay().updateCrystalsStats(); + float newIncome = turretEntity.getComponent(IncomeUpgradeComponent.class).getIncomeRate() - TIME_DECREASE; turretEntity.getComponent(IncomeUpgradeComponent.class).setIncomeRate(newIncome); turretEntity.getComponent(TowerUpgraderComponent.class).upgradeTower(TowerUpgraderComponent.UPGRADE.INCOME, (int) newIncome); incomeLabel.setText(String.format("%.2f", newIncome)); @@ -369,7 +387,7 @@ public void clicked(InputEvent event, float x, float y) { @Override public void enter(InputEvent event, float x, float y, int pointer, Actor fromActor) { - costDisplayLabel.setText("10"); + costDisplayLabel.setText(String.format("%d", UPGRADE_COST)); costDisplay.setVisible(true); } @Override @@ -378,6 +396,7 @@ public void exit(InputEvent event, float x, float y, int pointer, Actor toActor) } }); } + logger.info(String.valueOf(attack)); if (attack != 0) { innerUpgradeTable.add(attackIconImage).padRight(5).width(32).height(32); // Add attack icon innerUpgradeTable.add(attackLabel).expandX().left(); diff --git a/source/core/src/main/com/csse3200/game/screens/AnimationTexturePair.java b/source/core/src/main/com/csse3200/game/screens/AnimationTexturePair.java new file mode 100644 index 000000000..1aebf5c26 --- /dev/null +++ b/source/core/src/main/com/csse3200/game/screens/AnimationTexturePair.java @@ -0,0 +1,45 @@ +/** + * Represents a pair of animation and texture for rendering in a game screen. + * Provides easy access to the animation and its associated texture. + * + * @param The type of texture region used in the animation. + */ +package com.csse3200.game.screens; + +import com.badlogic.gdx.graphics.Texture; +import com.badlogic.gdx.graphics.g2d.Animation; +import com.badlogic.gdx.graphics.g2d.TextureRegion; + +public class AnimationTexturePair { + public final Animation animation; + public final Texture texture; + + /** + * Creates an {@code AnimationTexturePair} with the specified animation and texture. + * + * @param animation The animation to be associated with this pair. + * @param texture The texture to be associated with this pair. + */ + public AnimationTexturePair(Animation animation, Texture texture) { + this.animation = animation; + this.texture = texture; + } + + /** + * Gets the animation associated with this pair. + * + * @return The animation. + */ + public Animation getAnimation() { + return animation; + } + + /** + * Gets the texture associated with this pair. + * + * @return The texture. + */ + public Texture getTexture() { + return texture; + } +} diff --git a/source/core/src/main/com/csse3200/game/screens/AssetLoader.java b/source/core/src/main/com/csse3200/game/screens/AssetLoader.java index 396f933e5..d224c8e78 100644 --- a/source/core/src/main/com/csse3200/game/screens/AssetLoader.java +++ b/source/core/src/main/com/csse3200/game/screens/AssetLoader.java @@ -1,13 +1,17 @@ package com.csse3200.game.screens; + import com.badlogic.gdx.audio.Music; -import com.badlogic.gdx.graphics.g2d.TextureAtlas; -import com.badlogic.gdx.graphics.Texture; import com.badlogic.gdx.audio.Sound; -import com.csse3200.game.areas.ForestGameArea; +import com.badlogic.gdx.graphics.Texture; +import com.badlogic.gdx.graphics.g2d.TextureAtlas; import com.csse3200.game.services.ResourceService; import com.csse3200.game.services.ServiceLocator; +import java.util.Arrays; +import java.util.HashSet; +import java.util.Set; public class AssetLoader { + private static final Set loadedAssets = new HashSet<>(); // Define your asset file paths here public static final String[] textures = { "images/desert_bg.png", @@ -85,7 +89,8 @@ public class AssetLoader { "images/highlight_tile.png", "images/ui/Sprites/UI_Glass_Toggle_Bar_01a.png", "images/green_tile.png", - "images/red_tile.png" + "images/red_tile.png", + "images/HelpScreen/hs.jpg" }; public static final String[] textureAtlases = { @@ -136,6 +141,7 @@ public class AssetLoader { }; public static final String[] Sounds = { + "sounds/ui/Open_Close/NA_SFUI_Vol1_Open_01.ogg", "sounds/Impact4.ogg", "sounds/economy/click.wav", "sounds/economy/click_1.wav", @@ -235,5 +241,13 @@ public static Sound getSound(String assetPath) { public static Music getMusic(String assetPath) { return ServiceLocator.getResourceService().getAsset(assetPath, Music.class); } -} + public static boolean areAllAssetsLoaded() { + ResourceService resourceService = ServiceLocator.getResourceService(); + + return loadedAssets.containsAll(Arrays.asList(textures)) && + loadedAssets.containsAll(Arrays.asList(textureAtlases)) && + loadedAssets.containsAll(Arrays.asList(Sounds)) && + loadedAssets.containsAll(Arrays.asList(music)); + } +} diff --git a/source/core/src/main/com/csse3200/game/screens/HelpScreen.java b/source/core/src/main/com/csse3200/game/screens/HelpScreen.java deleted file mode 100644 index 14c8cd5dc..000000000 --- a/source/core/src/main/com/csse3200/game/screens/HelpScreen.java +++ /dev/null @@ -1,85 +0,0 @@ -package com.csse3200.game.screens; - -import com.badlogic.gdx.Gdx; -import com.badlogic.gdx.ScreenAdapter; -import com.badlogic.gdx.graphics.Texture; -import com.badlogic.gdx.graphics.g2d.SpriteBatch; -import com.badlogic.gdx.scenes.scene2d.Stage; -import com.badlogic.gdx.scenes.scene2d.ui.Image; -import com.badlogic.gdx.scenes.scene2d.ui.Skin; -import com.badlogic.gdx.scenes.scene2d.ui.Table; -import com.badlogic.gdx.scenes.scene2d.ui.TextButton; -import com.badlogic.gdx.scenes.scene2d.utils.ClickListener; -import com.badlogic.gdx.utils.viewport.FitViewport; -import com.csse3200.game.GdxGame; -import com.csse3200.game.ui.ButtonFactory; - -public class HelpScreen extends ScreenAdapter { - private final GdxGame game; - private Stage stage; - private SpriteBatch spriteBatch; - - - public HelpScreen(GdxGame game) { - this.game = game; - stage = new Stage(new FitViewport(Gdx.graphics.getWidth(), Gdx.graphics.getHeight())); - spriteBatch = new SpriteBatch(); - - // Create a table to organize the image placeholder - Table table = new Table(); - table.setFillParent(true); // Makes the table the size of the stage - - // Create one image placeholder - Image image = new Image(new Texture("images/background/HelpScreenBG.png")); - - // Add the image placeholder to the table - table.add(image).expand().fill(); - - // Add the table to the stage - stage.addActor(table); - - TextButton BackButton = ButtonFactory.createButton("Back"); - BackButton.addListener(new ClickListener() { - @Override - public void clicked(com.badlogic.gdx.scenes.scene2d.InputEvent event, float x, float y) { - game.setScreen(GdxGame.ScreenType.MAIN_MENU); - - } - }); - Table buttonTable = new Table(); - buttonTable.add(BackButton).padRight(10); - Table table1 = new Table(); - table1.setFillParent(true); - table1.top().right(); // Align to the top-right corner - table1.pad(20); // Add padding to the top-right corner - table1.add(buttonTable).row(); // Add button table and move to the next row - stage.addActor(table1); - } - @Override - public void show() { - // Set this screen as the input processor - Gdx.input.setInputProcessor(stage); - } - - @Override - public void render(float delta) { - // Clear the screen - spriteBatch.begin(); - spriteBatch.end(); - - // Draw the stage - stage.act(Math.min(Gdx.graphics.getDeltaTime(), 1 / 30f)); - stage.draw(); - } - - @Override - public void resize(int width, int height) { - stage.getViewport().update(width, height, true); - } - - @Override - public void dispose() { - stage.dispose(); - spriteBatch.dispose(); - } -} \ No newline at end of file diff --git a/source/core/src/main/com/csse3200/game/screens/HelpScreen/BossDescriptionHelpScreen.java b/source/core/src/main/com/csse3200/game/screens/HelpScreen/BossDescriptionHelpScreen.java new file mode 100644 index 000000000..772c72e32 --- /dev/null +++ b/source/core/src/main/com/csse3200/game/screens/HelpScreen/BossDescriptionHelpScreen.java @@ -0,0 +1,164 @@ +package com.csse3200.game.screens.HelpScreen; + +import com.badlogic.gdx.Gdx; +import com.badlogic.gdx.ScreenAdapter; +import com.badlogic.gdx.graphics.Texture; +import com.badlogic.gdx.graphics.g2d.BitmapFont; +import com.badlogic.gdx.graphics.g2d.SpriteBatch; +import com.badlogic.gdx.graphics.g2d.TextureRegion; +import com.badlogic.gdx.scenes.scene2d.Stage; +import com.badlogic.gdx.scenes.scene2d.ui.*; +import com.badlogic.gdx.scenes.scene2d.utils.ClickListener; +import com.badlogic.gdx.scenes.scene2d.utils.Drawable; +import com.badlogic.gdx.scenes.scene2d.utils.TextureRegionDrawable; +import com.badlogic.gdx.utils.viewport.FitViewport; +import com.csse3200.game.GdxGame; +/** + * A screen that displays information about boss characters in the game. + */ +public class BossDescriptionHelpScreen extends ScreenAdapter { + private final GdxGame game; + private Stage stage; + private SpriteBatch spriteBatch; + + /** + * Creates a new BossDescriptionHelpScreen. + * + * @param game The main game instance. + */ + public BossDescriptionHelpScreen(GdxGame game) { + this.game = game; + stage = new Stage(new FitViewport(Gdx.graphics.getWidth(), Gdx.graphics.getHeight())); + spriteBatch = new SpriteBatch(); + + // Create a table to organize the image placeholder + Table table = new Table(); + table.setFillParent(true); // Makes the table the size of the stage + + // Create one image placeholder + Image image = new Image(new Texture("images/lose-screen/desktop-wallpaper-simple-stars-video-background-loop-black-and-white-aesthetic-space.jpg")); + + // Add the image placeholder to the table + table.add(image).expand().fill(); + + // Add the table to the stage + stage.addActor(table); + + Skin skin = new Skin(Gdx.files.internal("images/ui/buttons/glass.json")); + TextButton BackButton = new TextButton("Back", skin); + BackButton.addListener(new ClickListener() { + @Override + public void clicked(com.badlogic.gdx.scenes.scene2d.InputEvent event, float x, float y) { + game.setScreen(GdxGame.ScreenType.MAIN_MENU); + + } + }); + + Texture imageTexture = new Texture("images/ui/Sprites/UI_Glass_Arrow_Large_01a - Copy.png"); + Drawable drawable = new TextureRegionDrawable(new TextureRegion(imageTexture)); + ImageButton MobsButton = new ImageButton(drawable); + MobsButton.addListener(new ClickListener() { + @Override + public void clicked(com.badlogic.gdx.scenes.scene2d.InputEvent event, float x, float y) { + game.setScreen(GdxGame.ScreenType.HELP_MOBS_SCREEN); + + } + }); + + Table buttonTable = new Table(); + buttonTable.add(BackButton).padRight(10); + Table table1 = new Table(); + table1.setFillParent(true); + table1.top().right(); // Align to the top-right corner + table1.pad(20); // Add padding to the top-right corner + table1.add(buttonTable).row(); // Add button table and move to the next row + stage.addActor(table1); + + Table buttonTable1 = new Table(); + buttonTable1.add(MobsButton).padRight(10); + Table table2 = new Table(); + table2.setFillParent(true); + table2.center().top(); // Align to the middle-right corner + table2.pad(20); // Add padding to the middle-right corner + table2.add(buttonTable1).row(); // Add button table and move to the next row + stage.addActor(table2); + + Table imageTextTable = new Table(); + imageTextTable.setFillParent(true); + imageTextTable.center(); + + float leftPadding = 30f; + imageTextTable.padLeft(leftPadding); + + // Create an array of image file names + String[] imageFileNames = { + "images/HelpScreen/Patrickbosshelp.png", + "images/HelpScreen/IceBosshelp.png", + "images/HelpScreen/Demonbosshelp.png" + }; + + // Create an array of text descriptions + String[] textDescriptions = { + "Patrick is able to teleport to the closest entity and do damage as well as shoot projectiles and melee attack. Once Patrick hits half health, it will shoot 5 projectiles of random effect.", + "The Ice Baby is able to spawn water mobs and do AOE damage.", + "Demon boss is able to shoot burn projectiles and do AOE damage. Demon boss is also able to regenerate health once it hits half health.", + }; + + // Add images and text to the table + for (int i = 0; i < imageFileNames.length; i++) { + // Create an image from the file + Image imager = new Image(new Texture(imageFileNames[i])); + float imageSize = 200f; + imager.setSize(imageSize, imageSize); + + // Create a label for text description + Label.LabelStyle labelStyle = new Label.LabelStyle(); + BitmapFont customFont = new BitmapFont(Gdx.files.internal("images/ui/buttons/dot_gothic_16.fnt")); // Replace "your-font.fnt" with your font file path + customFont.getData().setScale(1.2f); // Adjust the scale factor to change the text size + labelStyle.font = customFont; + + Label label = new Label(textDescriptions[i], labelStyle); + label.setWrap(true); + + // Add the image and label to the table in two columns + Cell imageCell = imageTextTable.add(imager).expandX().pad(10); // Add spacing around the image + Cell

+ * @param towerButton the ImageButton which is being set to on, all others will be toggled off. If null, all buttons + * will be set to off + */ + private void towerToggle(ImageButton towerButton) { + if (towerButton == null) { + // set all buttons to off, disable if isDisabled + for (ImageButton button : towerButtons) { + button.setChecked(false); + } + } else { + // set the button corresponding to towerButton to on, all others to off + for (ImageButton button : towerButtons) { + button.setChecked(button == towerButton); + } + } + } + @Override + public float getZIndex() { + return Z_INDEX; + } + + @Override + public void dispose() { + buttonTable.clear(); + towerTable.clear(); + unloadSounds(); + super.dispose(); + } + + /** + * Loads sound assets for use in the class + */ + public void loadSounds() { + ServiceLocator.getResourceService().loadSounds(sounds); + ServiceLocator.getResourceService().loadAll(); + click = ServiceLocator.getResourceService().getAsset(sounds[0], Sound.class); + openSound = ServiceLocator.getResourceService().getAsset(sounds[1], Sound.class); + } + + public void unloadSounds() { + ServiceLocator.getResourceService().unloadAssets(sounds); + } + + public void animateTowerButton(ImageButton button, float x, float y) { + button.setPosition(Gdx.graphics.getWidth() - x, Gdx.graphics.getHeight()); + button.addAction(new SequenceAction(Actions.moveTo(Gdx.graphics.getWidth() - x, + Gdx.graphics.getHeight() - y, 1f, Interpolation.fastSlow))); + } +} diff --git a/source/core/src/main/com/csse3200/game/screens/HelpScreen/HowToPlay.java b/source/core/src/main/com/csse3200/game/screens/HelpScreen/HowToPlay.java new file mode 100644 index 000000000..625ab1bd9 --- /dev/null +++ b/source/core/src/main/com/csse3200/game/screens/HelpScreen/HowToPlay.java @@ -0,0 +1,132 @@ +package com.csse3200.game.screens.HelpScreen; + +import com.badlogic.gdx.Gdx; +import com.badlogic.gdx.ScreenAdapter; +import com.badlogic.gdx.graphics.Texture; +import com.badlogic.gdx.graphics.g2d.BitmapFont; +import com.badlogic.gdx.graphics.g2d.SpriteBatch; +import com.badlogic.gdx.graphics.g2d.TextureRegion; +import com.badlogic.gdx.scenes.scene2d.Stage; +import com.badlogic.gdx.scenes.scene2d.ui.*; +import com.badlogic.gdx.scenes.scene2d.utils.ClickListener; +import com.badlogic.gdx.scenes.scene2d.utils.Drawable; +import com.badlogic.gdx.scenes.scene2d.utils.TextureRegionDrawable; +import com.badlogic.gdx.utils.Align; +import com.badlogic.gdx.utils.viewport.FitViewport; +import com.csse3200.game.GdxGame; +/** + * A screen that provides information on how to play the game. + */ +public class HowToPlay extends ScreenAdapter { + private final GdxGame game; + private Stage stage; + private SpriteBatch spriteBatch; + /** + * Creates a new HowToPlay screen. + * + * @param game The main game instance. + */ + + public HowToPlay(GdxGame game) { + this.game = game; + stage = new Stage(new FitViewport(Gdx.graphics.getWidth(), Gdx.graphics.getHeight())); + spriteBatch = new SpriteBatch(); + Skin skin = new Skin(Gdx.files.internal("images/ui/buttons/glass.json")); + + Label.LabelStyle labelStyle = new Label.LabelStyle(); + BitmapFont customFont = new BitmapFont(Gdx.files.internal("images/ui/buttons/dot_gothic_16.fnt")); + labelStyle.font = customFont; + + Texture backgroundTexture = new Texture("images/lose-screen/desktop-wallpaper-simple-stars-video-background-loop-black-and-white-aesthetic-space.jpg"); + Drawable backgroundDrawable = new TextureRegionDrawable(new TextureRegion(backgroundTexture)); + + // Create a table to hold the background image + Table backgroundTable = new Table(); + backgroundTable.setFillParent(true); + backgroundTable.add(new Image(backgroundDrawable)).expand().fill(); + + // Add the background table to the stage first + stage.addActor(backgroundTable); + // Create a table to organize the image placeholder + Table table = new Table(); + table.setFillParent(true); // Makes the table the size of the stage + + // Create one image placeholder + Image image = new Image(new Texture("images/HelpScreen/HTP.png")); + + // Add the image placeholder to the table + table.add(image).expand().fill(); + table.pad(60f); + + // Add the table to the stage + stage.addActor(table); + + + TextButton BackButton = new TextButton("Back", skin); + BackButton.addListener(new ClickListener() { + @Override + public void clicked(com.badlogic.gdx.scenes.scene2d.InputEvent event, float x, float y) { + game.setScreen(GdxGame.ScreenType.MAIN_MENU); + + } + }); + + Texture imageTexture = new Texture("images/ui/Sprites/UI_Glass_Arrow_Large_01a - Copy.png"); + Drawable drawable = new TextureRegionDrawable(new TextureRegion(imageTexture)); + ImageButton HelpButton = new ImageButton(drawable); + HelpButton.addListener(new ClickListener() { + @Override + public void clicked(com.badlogic.gdx.scenes.scene2d.InputEvent event, float x, float y) { + game.setScreen(GdxGame.ScreenType.HELP_SCREEN); + + } + }); + + + Table buttonTable = new Table(); + buttonTable.add(BackButton).padRight(10); + Table table1 = new Table(); + table1.setFillParent(true); + table1.top().right(); // Align to the top-right corner + table1.pad(20); // Add padding to the top-right corner + table1.add(buttonTable).row(); // Add button table and move to the next row + stage.addActor(table1); + + Table buttonTable1 = new Table(); + buttonTable1.add(HelpButton).padRight(10); + Table table2 = new Table(); + table2.setFillParent(true); + table2.center().top(); // Align to the middle-right corner + table2.pad(20); // Add padding to the middle-right corner + table2.add(buttonTable1).row(); // Add button table and move to the next row + stage.addActor(table2); + + } + @Override + public void show() { + // Set this screen as the input processor + Gdx.input.setInputProcessor(stage); + } + + @Override + public void render(float delta) { + // Clear the screen + spriteBatch.begin(); + spriteBatch.end(); + + // Draw the stage + stage.act(Math.min(Gdx.graphics.getDeltaTime(), 1 / 30f)); + stage.draw(); + } + + @Override + public void resize(int width, int height) { + stage.getViewport().update(width, height, true); + } + + @Override + public void dispose() { + stage.dispose(); + spriteBatch.dispose(); + } +} \ No newline at end of file diff --git a/source/core/src/main/com/csse3200/game/screens/HelpScreen/MobsDescriptionHelpScreen.java b/source/core/src/main/com/csse3200/game/screens/HelpScreen/MobsDescriptionHelpScreen.java new file mode 100644 index 000000000..1d89925fa --- /dev/null +++ b/source/core/src/main/com/csse3200/game/screens/HelpScreen/MobsDescriptionHelpScreen.java @@ -0,0 +1,205 @@ +package com.csse3200.game.screens.HelpScreen; + +import com.badlogic.gdx.Gdx; +import com.badlogic.gdx.ScreenAdapter; +import com.badlogic.gdx.graphics.Texture; +import com.badlogic.gdx.graphics.g2d.BitmapFont; +import com.badlogic.gdx.graphics.g2d.SpriteBatch; +import com.badlogic.gdx.graphics.g2d.TextureRegion; +import com.badlogic.gdx.scenes.scene2d.Stage; +import com.badlogic.gdx.scenes.scene2d.ui.*; +import com.badlogic.gdx.scenes.scene2d.utils.ClickListener; +import com.badlogic.gdx.scenes.scene2d.utils.Drawable; +import com.badlogic.gdx.scenes.scene2d.utils.TextureRegionDrawable; +import com.badlogic.gdx.utils.viewport.FitViewport; +import com.csse3200.game.GdxGame; +/** + * A screen that provides descriptions of different enemy mobs in the game. + */ + +public class MobsDescriptionHelpScreen extends ScreenAdapter { + private final GdxGame game; + private Stage stage; + private SpriteBatch spriteBatch; + + /** + * Creates a new MobsDescriptionHelpScreen. + * + * @param game The main game instance. + */ + public MobsDescriptionHelpScreen(GdxGame game) { + this.game = game; + stage = new Stage(new FitViewport(Gdx.graphics.getWidth(), Gdx.graphics.getHeight())); + spriteBatch = new SpriteBatch(); + + // Create a table to organize the image placeholder + Table table = new Table(); + table.setFillParent(true); // Makes the table the size of the stage + + // Create one image placeholder + Image image = new Image(new Texture("images/lose-screen/desktop-wallpaper-simple-stars-video-background-loop-black-and-white-aesthetic-space.jpg")); + + // Add the image placeholder to the table + table.add(image).expand().fill(); + + // Add the table to the stage + stage.addActor(table); + + Skin skin = new Skin(Gdx.files.internal("images/ui/buttons/glass.json")); + TextButton BackButton = new TextButton("Back", skin); + BackButton.addListener(new ClickListener() { + @Override + public void clicked(com.badlogic.gdx.scenes.scene2d.InputEvent event, float x, float y) { + game.setScreen(GdxGame.ScreenType.MAIN_MENU); + } + }); + + TextButton TowersButton = new TextButton("Towers", skin); + TowersButton.addListener(new ClickListener() { + @Override + public void clicked(com.badlogic.gdx.scenes.scene2d.InputEvent event, float x, float y) { + game.setScreen(GdxGame.ScreenType.HELP_TOWER_SCREEN); + } + }); + + TextButton BossButton = new TextButton("BossMobs", skin); + BossButton.addListener(new ClickListener() { + @Override + public void clicked(com.badlogic.gdx.scenes.scene2d.InputEvent event, float x, float y) { + game.setScreen(GdxGame.ScreenType.HELP_BOSS_SCREEN); + } + }); + + Texture imageTexture = new Texture("images/ui/Sprites/UI_Glass_Arrow_Large_01a.png"); + Drawable drawable = new TextureRegionDrawable(new TextureRegion(imageTexture)); + ImageButton GameDescButton = new ImageButton(drawable); + GameDescButton.addListener(new ClickListener() { + @Override + public void clicked(com.badlogic.gdx.scenes.scene2d.InputEvent event, float x, float y) { + game.setScreen(GdxGame.ScreenType.HELP_SCREEN); + } + }); + + Table buttonTable = new Table(); + buttonTable.add(BackButton).padRight(10); + Table table1 = new Table(); + table1.setFillParent(true); + table1.top().right(); // Align to the top-right corner + table1.pad(20); // Add padding to the top-right corner + table1.add(buttonTable).row(); // Add button table and move to the next row + stage.addActor(table1); + + Table buttonTable2 = new Table(); + buttonTable2.add(TowersButton).padRight(10); + Table table2 = new Table(); + table2.setFillParent(true); + table2.center().right(); // Align to the top-right corner + table2.pad(20); // Add padding to the top-right corner + table2.add(buttonTable2).row(); // Add button table and move to the next row + stage.addActor(table2); + + Table buttonTable3 = new Table(); + buttonTable3.add(GameDescButton).padRight(10); + Table table3 = new Table(); + table3.setFillParent(true); + table3.center().left(); // Align to the top-right corner + table3.pad(20); // Add padding to the top-right corner + table3.add(buttonTable3).row(); // Add button table and move to the next row + stage.addActor(table3); + + Table buttonTable4 = new Table(); + buttonTable4.add(BossButton).padRight(10); + Table table4 = new Table(); + table4.setFillParent(true); + table4.center().bottom(); // Align to the top-right corner + table4.pad(20); // Add padding to the top-right corner + table4.add(buttonTable4).row(); // Add button table and move to the next row + stage.addActor(table4); + + Table imageTextTable = new Table(); + imageTextTable.setFillParent(true); + imageTextTable.center(); + + float leftPadding = 30f; + imageTextTable.padLeft(leftPadding); + + // Create an array of image file names + String[] imageFileNames = { + "images/HelpScreen/Firewormhelp.png", + "images/HelpScreen/Dragonknighthelp.png", + "images/HelpScreen/Waterslimehelp.png", + "images/HelpScreen/WaterQueenhelp.png", + "images/HelpScreen/Skeletonhelp.png", + "images/HelpScreen/Wizardhelp.png", + }; + + // Create an array of text descriptions + String[] textDescriptions = { + "Fire Worm is able to shoot projectiles towards the towers.", + "DragonKnight is able to dodge projectiles.", + "Water slimes are able to split once dead", + "Water Queens are able to shoot towards the towers", + "Skeletons are able to do close range attack damage.", + "Wizards are able to deflect projectiles.", + }; + + // Add images and text to the table + for (int i = 0; i < imageFileNames.length; i++) { + // Create an image from the file + Image imager = new Image(new Texture( imageFileNames[i])); + float imageSize = 200f; + imager.setSize(imageSize, imageSize); + + // Create a label for text description + Label.LabelStyle labelStyle = new Label.LabelStyle(); + BitmapFont customFont = new BitmapFont(Gdx.files.internal("images/ui/buttons/dot_gothic_16.fnt")); // Replace "your-font.fnt" with your font file path + customFont.getData().setScale(1.2f); // Adjust the scale factor to change the text size + labelStyle.font = customFont; + + Label label = new Label(textDescriptions[i], labelStyle); + label.setWrap(true); + + // Add the image and label to the table in two columns + Cell imageCell = imageTextTable.add(imager).expandX().pad(10); // Add spacing around the image + Cell