Cascadia 0.1
Learning Rust to develop my dream game!
Table of Contents
This first devlog concludes the test phase of Cascadia, my survival * I know there isn’t any survival mechanics yet, but those will come soon! sandbox game!
Download Pre-Alpha 0.1 and give it a try
data:image/s3,"s3://crabby-images/e2294/e2294b9df3d408a6a0a5d55d2687055e7d9f457f" alt=""
Features
- Placing and breaking blocks
- 10 blocks
- Bedrock
- Stone
- Dirt
- Grass
- Logs
- Leaves
- Planks
- Bricks
- Sand
- Glass
- Randomly generated 256×64×256 world
- Cliffs
- Trees
- Sand traps
- Caves
- Physics
- Lighting
Cascadia’s test phase of development spanned from September 21st to November 3rd, with a 3 week break between October 1st and October 23rd due to college. (If you’re bad at math, that’s 21 days!)
It saw a few architectural changes before settling on a bespoke game engine written in Rust, using winit for window creation and input handling and wgpu for graphics.
So let’s take a trip down memory lane, shall we?
Pre-wgpu
Not knowing where to start, Kett recommended that I try raylib, a C library for game programming. I found a binding for Rust and made this simple test scene using raylib’s built-in draw functions.
data:image/s3,"s3://crabby-images/6a33a/6a33a94d712e27c992b69a496b4067364cdb8c3e" alt=""
I then figured out how to texture those cubes!
data:image/s3,"s3://crabby-images/dec9f/dec9f81852f42b3146eba862379aa11f57be9e02" alt=""
But I was having a lot of trouble trying to figure out what the Rust bindings for raylib were. Raylib has plenty of examples on their website, but they were only for C. So I gave up and switched to Bevy, a data-driven game engine built in Rust.
data:image/s3,"s3://crabby-images/47927/47927f6cedc899db606e5a6811c101af61fe2817" alt=""
Bevy uses an Entity Component System (ECS) pattern where entities are blank slates that you attach components to and process with systems. It’s a pretty novel idea to me, and it seems to be quite popular in Rust and gaining traction outside of it too.
But Bevy’s ham-fisted approach to treating everything as an ECS—including UI—just rubbed me the wrong way. I needed to find something better.
Seeing that Bevy used wgpu for its graphics, a low-level cross-platform graphics API, I found a wgpu tutorial for creating a render pipeline from scratch. For the next 3 days I poured hours reading, learning, and writing low-level graphics.
I really hoped this would be worth the effort.
wgpu
data:image/s3,"s3://crabby-images/9ee85/9ee85c2229950ff69708c5158a06853e38ff6c09" alt=""
I was quite exhilarated with what I had made, my very own render pipeline! From scratch! Without using a game engine!
But the future still felt uncertain to me as I didn’t know how much steam I would have to continue working at such a low level, nor if I would be able to find enough resources and tutorials online to know what to do next.
And count my lucky stars, because I found a different tutorial that cleaned up and expanded upon the previous wgpu tutorial! (Which left me with a monolithic file with nearly a thousand lines of code)
With a much more readable source code, I felt invigorated to continue toiling away and implemented raycasting, allowing me to place and break blocks for the first time!
I was still rendering individually instanced cubes which wasn’t going to work in the long-term, so I got to work on mesh generation. Here is the very first mesh I generated.
data:image/s3,"s3://crabby-images/8f4b2/8f4b217edc256d261c1699f31ed2173274891c12" alt=""
I then rendered square faces proper…
data:image/s3,"s3://crabby-images/79bfc/79bfc3a927ebef40ddd6a6a8ab32440e4a4b41a8" alt=""
…culled hidden faces…
data:image/s3,"s3://crabby-images/06d17/06d1760868952281c5ac41abe80c0c215610fe35" alt=""
…and added the rest of the faces! This is a view from inside the mesh.
data:image/s3,"s3://crabby-images/e6bb0/e6bb0b9dbf173dfde16b91784fb73320ac407620" alt=""
Vertex data
Soon after I integrated texture array support into the render pipeline and rendered grass!
I wanted to use texture arrays instead of a texture sheet because it would be easier to set up. Instead of calculating the correct UVs for a given texture, I just pass a texture index to the shader.
With this index-first approach, I could even make animated textures super easy to implement in the future.
data:image/s3,"s3://crabby-images/4c262/4c262172660d9d48982b782dc14d85f0442231fc" alt=""
And with my newfound knowledge of the render pipeline and the vertex buffer data, I added lighting data to the vertices to give blocks some contrast lighting.
data:image/s3,"s3://crabby-images/201ea/201eacc2dc7bb0c339fd5b1ab7755a13d462b425" alt=""
Adding shadows was pretty easy too using the same vertex light data queried from a light array I added to the chunk.
data:image/s3,"s3://crabby-images/e186b/e186b04735da7f1f1257d8a92aed3da1fb808bb2" alt=""
Chunks
The next step was to have more than one chunk. This was going to have a few challenges, namely getting lighting to work properly across chunks, and to cull the faces of blocks between chunks, but for now I put those problems off for later.
data:image/s3,"s3://crabby-images/50d8b/50d8b4ec0b56468a8fb4ffa0da31e9b87841b8ef" alt=""
With the ability to place different blocks now, I made the first dirt house in Cascadia.
data:image/s3,"s3://crabby-images/b5f20/b5f20ea1cf25a4333db433e20cd79a246574cc46" alt=""
I then added some new blocks that needed their own rendering code. Glass and leaves are both transparent, but leaves fully render to neighboring leaves for a neat layering effect. Logs were the first block to use multiple textures.
data:image/s3,"s3://crabby-images/1a23a/1a23a7177a6a3ade1466d01b2cc0e3b0086774ff" alt=""
data:image/s3,"s3://crabby-images/21e75/21e75a79b52555d8a13087eabad9110aec399a6a" alt=""
Here’s a gratuity shot of 64,000 chunks being rendered at once :3
data:image/s3,"s3://crabby-images/942a1/942a188ec86c71571523871dd06c838e859241d3" alt=""
Although I don’t have footage of it, this test version marked the inclusion of the first iteration of physics! (Which unbeknownst to me would haunt me for a week as I pulled my hair out trying to fix numerous collision bugs)
data:image/s3,"s3://crabby-images/15e3d/15e3df667fb1a460ac9a48551f5f064ec47dda2c" alt=""
Terrain generation
Here’s the first version with naturally generated trees! Yeah they’re pretty silly looking.
data:image/s3,"s3://crabby-images/9011a/9011aad7e5d8ec5476a8c983731155c831d540c0" alt=""
Lighting has returned along with some new terrain generation!
data:image/s3,"s3://crabby-images/9af2a/9af2a47ce50702a6a26ad8a36fdbccaa10894996" alt=""
I sprited a new font for Cascadia, and using a distance fog * The fog is coming the fog is coming the fog is coming tutorial meant for glsl shaders, I was able to translate it into wgsl, the shading language that wgpu uses.
data:image/s3,"s3://crabby-images/61060/61060b880e502e96779bca78da05673e70ad45dc" alt=""
Another terrain generation change to include more varied terrain such as cliffs.
data:image/s3,"s3://crabby-images/eb505/eb505131679308d434ec57a8bce40f193d2a407d" alt=""
And finally, I vanquished the last collision bug. There were few edge cases where you could phase through the edge of a block, and this video was me testing those.
Postmortem
Whew, that was a lot of progress in a short span of time! And in that time I learned a lot, from programming in Rust (gotta love that borrow checker) to writing low-level graphics with wgpu.
I’m genuinely thrilled with what I have made so far, and I’m excited to continue working on Cascadia to make it my dream game!
data:image/s3,"s3://crabby-images/19976/19976d0c630a8a81de263fa779066b751ee1ee97" alt="signature that says "Chai""