Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/slideshow776/terfenstein-3d
:star: My first 3D shooter game, made with LibGDX :star:
https://github.com/slideshow776/terfenstein-3d
3d-game libgdx pixel-art shooter-game women-in-tech
Last synced: 2 months ago
JSON representation
:star: My first 3D shooter game, made with LibGDX :star:
- Host: GitHub
- URL: https://github.com/slideshow776/terfenstein-3d
- Owner: Slideshow776
- Created: 2022-07-13T10:10:39.000Z (over 2 years ago)
- Default Branch: master
- Last Pushed: 2024-11-06T07:45:00.000Z (3 months ago)
- Last Synced: 2024-11-06T08:31:45.993Z (3 months ago)
- Topics: 3d-game, libgdx, pixel-art, shooter-game, women-in-tech
- Language: Java
- Homepage: https://sandramoen.no
- Size: 170 MB
- Stars: 22
- Watchers: 2
- Forks: 2
- Open Issues: 1
-
Metadata Files:
- Readme: README.md
- Funding: .github/FUNDING.yml
Awesome Lists containing this project
README
# Terfenstein 3D
[Official Trailer](https://youtu.be/BUj2IYbL-zI)
[![Trailer thumbnaik](https://user-images.githubusercontent.com/4059636/201704064-5a339898-6c23-4ee2-8d4f-5ac62d32c9a0.png)](https://youtu.be/BUj2IYbL-zI)
[Buy it on Steam?](https://store.steampowered.com/app/2192840/Terfenstein_3D/?beta=0)
[![Main Capsule](https://user-images.githubusercontent.com/4059636/200133991-84551e8a-26b2-4126-8695-40a6689ac5de.png)](https://store.steampowered.com/app/2192840/Terfenstein_3D/?beta=0)
![image](https://user-images.githubusercontent.com/4059636/184866741-be475fb3-9945-4740-a251-5781a98a6eea.gif)
Read on below for project specifics.
1. [Game Design Document](#game-design-document-sparkling_heart)
2. [Credits](#credits)
3. [Project Comments](#project-comments)
4. [Other](#other)
5. [Project Kanban Board](https://github.com/Slideshow776/3D-shooting-game/projects/1)## Game Design Document :sparkling_heart:
1. :blue_heart: Overall Vision
* **Write a short paragraph explaining the game:**
This is a first person shooter (fps) about infiltrating gender fascists facility and disabling their weapons of mass destruction.
* **Describe the genre:**> First-person shooter (FPS) is a sub-genre of shooter video games centered on gun and other weapon-based combat in a first-person perspective, with the player experiencing the action through the eyes of the protagonist and controlling the player character in a three-dimensional space. --[wikipedia](https://en.wikipedia.org/wiki/First-person_shooter)
* **What is the target audience?**
The average player is assumed to be aware of the transphobic pressure that exists to transgendered peope by the media and TERFs
* **Why play this game?**
2. :purple_heart: Mechanics: the rules of the game world
* **What are the character's goals?**
* **What abilities does the character have?**
* **What obstacles or difficulties will the character face?**
* **What items can the character obtain**
* **What resources must be managed?**
3. :heart: Dynamics: the interaction between the player and the game mechanics
* **What hardware is required by the game?*** Desktop needs to have a functional mouse, keyboard, and screen. This game will not require a powerful computer.
* **What type of proficiency will the player need to develop to become proficient at the game?**
* **What gameplay data is displayed during the game?**
* **What menus, screens, or overlays will there be?**
* **How does the player interact with the game at the software level?**
4. :green_heart: Aesthetics: the visual, audio, narrative, and psychological aspects of the game
* **Describe the style and feel of the game.**
* **Does the game use pixel art, line art, or realistic graphics?**
* **What style of background music, ambient sounds will the game use?**
* **What is the relevant backstory for the game?**
* **What emotional state(s) does the game try to provoke?**
* **What makes the game fun?**
5. :yellow_heart: Development
* **List the team members and their roles, responsibilities, and skills.**
This project will be completed individually; graphics and audio will be obtained from third-party websites that make their assets available under the Creative Commons license, and so the main task will be programming and creating some graphics.
* **What equipment is needed for this project?**A computer (with keyboard, mouse, and speakers) and internet access will be necessary to complete this project.
* **What are the tasks that need to be accomplished to create this game?**
This project will use a simple Kanban board hosted on the project's GitHub page.
The main sequence of steps to complete this project is as follows:
* Setting up a project scaffold
* **Programming game mechanics and UI**
* **Creating and obtaining graphical assets**
* Obtaining audio assets
* Controller support
* **Polishing**
* Deployment* **What points in the development process are suitable for playtesting?**
The main points for playtesting are when the basic game mechanics of the level screen are implemented, and when it is visualised. The questions that will be asked are:
* Is the gameplay and UI understandable?
* Is the gameplay interesting?
* How do the controls feel?
* How is the pace of the game?
* Are there any improvement suggestions?
* **What are the plans for publication?**## Credits
## Project comments
### Wall clipping
To stop the wall from clipping out of sight it is necessary to set the `camera.near` to a very tiny amount, like so:
```
camera = new PerspectiveCamera(67, Gdx.graphics.getWidth(), Gdx.graphics.getHeight());
camera.lookAt(0, 0, 0);
camera.near = .01f;
camera.far = 100f;
camera.update();
```
### Player movement
The cursor can be locked to the window like this line of code: `Gdx.input.setCursorCatched(true);`. In HTML this needs to be triggered multiple times to ensure the correct application behaviour.Keyboard movement is as simple as this:
```
private void keyboardPolling(float dt) {
if (Gdx.input.isKeyPressed(Input.Keys.W))
stage.moveCameraForward(speed * dt);
if (Gdx.input.isKeyPressed(Input.Keys.A))
stage.moveCameraRight(speed * dt);
if (Gdx.input.isKeyPressed(Input.Keys.S))
stage.moveCameraForward(-speed * dt);
if (Gdx.input.isKeyPressed(Input.Keys.D))
stage.moveCameraRight(-speed * dt);
}
```Mouse movement is this simple line:
```
private void mousePolling() {
turnBy(rotateSpeed * Gdx.input.getDeltaX() * BaseGame.mouseMovementSensitivity);
stage.turnCamera(rotateSpeed * Gdx.input.getDeltaX() * BaseGame.mouseMovementSensitivity);
}
```### Collisions
While the game is 3D viewed from a birds perspective the area of movement is 2D, and thus we can implement a simple/standard 2D collision detection system using these functions found in [`BaseActor3D.java`](https://github.com/Slideshow776/Terfenstein-3D/blob/master/source%20code/core/src/no/sandramoen/commanderqueen/actors/utils/baseActors/BaseActor3D.java):
* `setBaseRectangle()`
* `setBasePolygon()`
* `getBoundaryPolygon()`
* `overlaps(BaseActor3D other)`
* `preventOverlap(BaseActor3D other)`### Frustum Culling
![demo](https://user-images.githubusercontent.com/4059636/179392342-22c26f1b-775d-49b6-8f44-ed7992cc1aa7.gif)As explained in [this blog](https://xoppa.github.io/blog/3d-frustum-culling-with-libgdx/) one can save computer resources by only drawing the 3D world entities that are visible, this is called frustum culling. Feature added in [this commit](https://github.com/Slideshow776/3D-shooting-game/commit/b8b4f9183c6923a61a60fabbc4b729be8523ebe2).
### Ray Picking
![image](https://user-images.githubusercontent.com/4059636/179480413-eb9409c7-0b68-4168-aa83-17a3e187834a.png)
> Mathematically, a ray is the portion of a line that originates from a fixed point and extends indefinitely in a particular direction. A ray always travels in a straight line in a medium until it hits the boundary of the medium that it is traveling in.
>
>Once it hits that boundary, it can either get reflected, refracted, absorbed, or undergo all three operations partially. >-- [techcenturion.com](https://www.techcenturion.com/ray-tracing)Ray picking was implemented following [this tutorial](https://xoppa.github.io/blog/interacting-with-3d-objects/) and can be seen in [this commit](https://github.com/Slideshow776/3D-shooting-game/commit/55d86d4240be404148fd7d3fe5502e6591f9f13b).
### Tile Maps
Tile maps were implemented [like so](https://github.com/Slideshow776/3D-shooting-game/commit/edddec5d5e640447599fe48c0d940af20ee898e7).![image](https://user-images.githubusercontent.com/4059636/179554378-55590c98-aa44-45e3-a6ce-30da743743ed.png)
### Directional Sprites
Eight directional sprites were added in [this commit](https://github.com/Slideshow776/3D-shooting-game/commit/1191866abdba55cc8dbda692d030d859626b2033).
The play can now circle an enemy and see it from behind, etc. This allows for gameplay where the enemy is not always facing the player ready to attack.![eight directional sprites example](https://user-images.githubusercontent.com/4059636/181704832-6e015a4c-68b3-4573-94fe-c12480758664.gif)
### Enemy Ray Picking
You don't need a camera to ray pick. The algorithm was expanded to support simple non-camera rays too =>
```
public static int getRayPickedListIndex(Vector3 origin, Vector3 direction, Array list) {
Ray ray = new Ray(origin, direction);
return getClosestListIndex(ray, list);
}public static int getRayPickedListIndex(int screenX, int screenY, Array list, PerspectiveCamera camera) {
Ray ray = camera.getPickRay(screenX, screenY);
return getClosestListIndex(ray, list);
}private static int getClosestListIndex(Ray ray, Array list) {
int index = -1;
float distance = -1;
for (int i = 0; i < list.size; ++i) {
final float dist2 = list.get(i).modelData.intersects(ray);
if (dist2 >= 0f && (distance < 0f || dist2 <= distance)) {
index = i;
distance = dist2;
}
}
return index;
}
```
This allows for the enemy to scan for the player to detect them, and also to be able to shoot them.
Raypicking should be used sparingly, this game runs the algorithm for all enemies not dead, facing the player, at it's own interval.### Pathfinding
LibGDX's [AI library](https://github.com/libgdx/gdx-ai) was included in [this](https://github.com/Slideshow776/3D-shooting-game/commit/85b5b903a4aa80ca0621bc0a6462f40481b43761) and [this](https://github.com/Slideshow776/3D-shooting-game/commit/7aaa1c3a9080e70785d45faa4b5848deb7654dfc) commits following [this excellent tutorial](https://happycoding.io/tutorials/libgdx/pathfinding).
The pathfinding is a directional improvement upon [Dijkstra's algorithm](https://en.wikipedia.org/wiki/Dijkstra%27s_algorithm) and enables us to tell the enemies where to go.
This algorithm is used sparingly whenever there is a sound the enemy should respond to or get the last position of the player's last sighting. This opens up more interesting gameplay.![Dijkstra's improved algorithm A*](https://happycoding.io/tutorials/libgdx/images/dijkstra-search.gif)
### Decals
A [decal](https://libgdx.com/wiki/graphics/3d/decals) is a 2D image in a 3D world. In this project it's used to generate an effect where the bullet hits stuff, e.g. a wall.
This project implements this with a parent [class manager](https://github.com/Slideshow776/Terfenstein-3D/blob/master/source%20code/core/src/no/sandramoen/commanderqueen/actors/decals/DecalsManager.java) that can be [extended](https://github.com/Slideshow776/Terfenstein-3D/blob/master/source%20code/core/src/no/sandramoen/commanderqueen/actors/decals/BulletDecals.java), and then used [like so](https://github.com/Slideshow776/Terfenstein-3D/blob/master/source%20code/core/src/no/sandramoen/commanderqueen/screens/gameplay/LevelScreen.java).
```
Vector3 temp = new Vector3().set(ray.direction).scl(player.distanceBetween(shootable.get(i)) - (Tile.diagonalLength / 2)).add(ray.origin);
bulletDecals.addDecal(temp.x, temp.y, temp.z);
```
This code will get the wall's position, and create a new `Decal` at roughly the wall's facing surface.### Camera Rolling
For a lean-into-walking-direction effect:
![roll demo](https://user-images.githubusercontent.com/4059636/188260161-cc00ca97-c036-4eb5-8391-6f4f9b4064d1.gif)
![camera-movement](https://user-images.githubusercontent.com/4059636/188260208-71410b8b-cc0d-4922-868f-d2067d1bebdd.png)
```
private void keyboardPolling(float dt) {
if (Gdx.input.isKeyPressed(Keys.A))
rollAngle = MathUtils.clamp(rollAngle -= ROLL_INCREMENT, -ROLL_ANGLE_MAX, ROLL_ANGLE_MAX);if (Gdx.input.isKeyPressed(Keys.D))
rollAngle = MathUtils.clamp(rollAngle += ROLL_INCREMENT, -ROLL_ANGLE_MAX, ROLL_ANGLE_MAX);if (!Gdx.input.isKeyPressed(Keys.A) && !Gdx.input.isKeyPressed(Keys.D)) {
if (rollAngle > 0)
rollAngle -= ROLL_INCREMENT;
else if (rollAngle < 0)
rollAngle += ROLL_INCREMENT;
}
}
```
```
public void rollCamera(float angle) {
camera.up.set(Vector3.X);
camera.rotate(camera.direction, angle);
}
```### Level Design
Some level design facets that are used are:
* lights
* affordances
* visual guidance
* pickups
* pacing (remember downbeats)
* leading lines
* local landmarks
* avoid straights
* junctions
* choices of traversal
* gating
* bread crumbs
* negative space
* readability (is it easy to know where to go?)
* environmental storytelling
* illusion of spaceThese, and more, are covered in [this book](https://www.lulu.com/shop/max-pears/lets-design-exploration/paperback/product-zerwr7.html?page=1&pageSize=4).
[![image](https://user-images.githubusercontent.com/4059636/194706249-e244466e-0cb2-41e9-9599-3f78370f1d86.png)](https://www.lulu.com/shop/max-pears/lets-design-exploration/paperback/product-zerwr7.html?page=1&pageSize=4)
### Convert `.jar` to `.exe`
Follow [this tutorial](https://fullstackdeveloper.guru/2020/06/17/how-to-create-a-windows-native-java-application-generating-exe-file/) to do so.
To automate this process checkout the [desktop build file](https://github.com/Slideshow776/Terfenstein-3D/blob/master/source%20code/desktop/build.gradle).Alternatively use [packr](https://github.com/libgdx/packr#usage) with the following commands
```
java -jar packr-all-4.0.0.jar
--platform "windows64"
--jdk "C:\Program Files\Java\jdk-15.0.1"
--useZgcIfSupportedOs
--executable "Terfenstein 3D"
--classpath "C:\Users\Sandra Moen\dev\Terfenstein 3D\source code\desktop\build\lib\Terfenstein 3D.jar"
--mainclass "no.sandramoen.terfenstein3D.DesktopLauncher"
--resources "C:\Users\Sandra Moen\dev\Terfenstein 3D\source code\assets"
--output "C:\Users\Sandra Moen\Desktop\Terfenstein 3D\packr build files"
```## Other
For other project specifics check out the [commits](https://github.com/Slideshow776/3D-shooting-game/commits/master).[Go back to the top](https://github.com/Slideshow776/Terfenstein-3D#terfenstein-3d).