al2f's website

Hiding walls which cover the player in a 3D RPG

2025-03-03

Introduction

This year I started a 3D rpg project in godot. Choosing to make the project both in 3D, and multiplayer lead to some quite interesting new ideas and problems to solve. This is one such problem.

Taking a scene from an RPG such as Chrono Trigger. There is some floor, some walls, some furniture, and the main character - Crono. The measurements aren’t hard to make as the game uses a 16x16 pixel grid, and we can quickly get a rough idea of what the 3d level will look like:

A screenshot of an iconic scene from Chrono Trigger. A medieval room is shown with the main wall covering most of the screen width. In the middle of the wall is a large window with open curtains letting sunlight into the room. To the right(east) is Crono’s bed, north of which is a bookcase. To the west of the window is a desk with a book and desk lamp. Crono is standing in front of the drawer facing at the camera. South of the desk is a large staircase with wooden rails leading east, downstairs.Download chrono_room_back.xcf

A basic 3d representation of the same room in TrenchBroom. The furniture and bed are absent, as well as Crono.

Let’s say we want a view of the room from the other side. I imagine that would look something like this:

The same room is shown in the same top-down SNES as the very first image with the camera looking at the room from the north instead of the south. Majority of the furniture as well as Crono are obscured by the wall with the window.

It looks alright, but we can’t see Crono, or any of the furniture because the front wall is covering it. This would work well if we were viewing the house from the outside, but not when we’re inside. So let’s remove the closest wall. Let’s also add the walls that would be visible from this angle:

The 3D level in TrenchBroom viewed from the north showing an additional 3 walls visible from this angle which were not visible in Chrono Trigger due to its fixed viewing angle. A SNES screenshot of the same room is shown. Looking more refined and believeable than the TrenchBroom image.

Notice how which walls are shown depends on which direction the camera is facing. When viewing from the front, only the back wall with the window is shown. When viewed from the back however, 3 additional walls need to be drawn, and the previous ‘back’ wall is hidden.

The challenge here comes from determining which walls are obscuring the level from view and should be hidden, and which walls do not obscure the level and should be shown. In other words, ‘how to hide walls which obscure the level’.

Raycasts

At first I used raycasts to detect which walls were covering the floor. For each vertex of the wall, I cast a ray away from the camera, and checked for intersection with the floor.

A view of the 3D level from the west is shown. In the top-right corner of the diagram(south) a camera is depicted looking at the scene from the south. Yellow arrows are drawn from four of the vertices of the south-most wall representing ray-casts. these rays are created from each wall vertex, and point in the direction the camera is facing. Three of the arrow heads are marked with a red cross, indicating an intersection of the ray with the floor.

This method was not very reliable, as it did not detect a wall as obsctructing the floor when the wall was higher than the floor. Additionally, when the wall vertex was on the same plane as the floor, the collision was not always detected.

A modified version of the level is shown from the west. A 4-tile high and 8-tile deep wall is shown. South of it, and touching edges with it is a 3-tile wide and 5-tile deep floor. 6 Raycasts are shown coming off the wall, this time in a south-down direction. The two furthest raycasts do not collide with the floor as the wall is much deeper than the floor, while the four closest do not intersect with the floor as they are on the same north-up plane.

Wall normals

The other solution I wrote with the help of the following few resources1:

What caught my attention was this animated gif in the reddit post which achieved exactly the effect I wanted:

The rooftop of a 3D skyscraper stretching down into the void. The rooftop has a blue floor, and contains 4 pillars at each corner. 4 walls link these corners together into a square. The camera is rotating slowly around one pillar. When the camera looks at a pillar or wall which is obscuring the blue floor of the sky scraper it fades to being transparent. As the wall or column stops obscuring the floor, the object fades back to full opacity.

From then on it was a (fairly) simple matter of implementation.

  1. Each wall is assigned a normal (a direction which points to where the walls’ ‘inside’ is.

The level seen from the north-west. Cyan arrows point to the center of the room, perpendicular to the walls largest surface. The same level viewed from the south, showing arrows on the west and north walls which were not visible in the previous image.

  1. Some maths is used to determine if the camera is looking at the back face of a wall. If the camera is looking at the back face of a wall, the wall is hidden.

A screenshot of the level in Godot with an orthogonal perspective viewed from the south-west. The west-most and two south-most walls are hidden. A west-facing wall which is abscuring the east part of the level is visible.

  1. In some cases, this still shows a wall that is covering the floor. I go over each wall that is still shown after the previous step, and perform a few dozen2 raycasts for each wall

The previous screenshot except the west-facing wall is now also hidden.


  1. The term ‘backface occlusion culling’ used in both of the posts refers to a fairly un-common interpretation. The widely-used interpretation is hiding faces when viewed from behind, whereas in these two posts it was used to refer to hiding an entire object when viewed from behind, instead of an individual face ↩︎

  2. octets, really. Each wall has eight vertices ;) ↩︎