CelesteWFC_Showcase.mp4
Video can also be found in
media/
folder, calledCelesteWFC_Showcase.mp4
.
I love Celeste. Everything about the game is great: the pixel art, the music, the gameplay, the story. One of the best parts of the game are the levels. They are designed so well and I can't believe that they're tile based because everything looks so natural.
Celeste is the perfect candidate for exploring wave function collapse, due to its reliance on a grid system. I always thought WFC sounded cool as heck and wanted to try implementing it on my own. What better way to do so than to use one of the best platformers of all time [citation needed]?
- Learn in detail how wave function collapse works and how to implement it
- Write well-structured, performant, and idomatic C# code
- Work more with 2D Unity, including tilemaps and player movement
These links helped me decide on this project and will guide me through the process:
- Celeste Tilesets, Step-by-Step (article)
- Why I use Wave Function Collapse to create levels for my game (video)
- Superpositions, Sudoku, the Wave Function Collapse algorithm. (video)
- EPC2018 - Oskar Stalberg - Wave Function Collapse in Bad North (video)
- Paul Merrell, "Example-Based Model Synthesis" (2007)
- mxgnm / WaveFunctionCollapse (repo)
- Recreating Celeste's Movement | Mix and Jam (video)
There are three overarching parts of this project:
- C# implementation of WFC, most likely customized for Celeste's purposes
- Celeste-specific
- Includes at least Chapter 1's "Forsaken City" tileset
- Preset that constrains the grid to be exactly that of Celeste's
- Grid editor/viewer GUI
- Ability to specify number of grid columns and rows
- Ability to click a tile to constrain it to a limited number of states (e.g. forcing a path to be carved out)
- Ability to zoom in/out and pan around the grid/output
I have some ideas on additional features if I have more time:
- Ability to export/download finished output as a single image
- Generate more levels using different tilesets
- Performance improvements beyond the naive implementation
- Allowing the user to give an example input which the algorithm uses instead to build its ruleset, generating levels from that instead
- Porting the project to browser via WebGL
- Loading the generated output as an actual custom map in Celeste via Everest, a community mod loader
This project is specifically about wave function collapse so that's the algorithmic technique I'm focusing on. In particular, I'll be reading Paul Merell's specification from his original 2007 i3D paper (linked above) as well as Maxim Gumin's work, which introduced the name "wave function collapse" for the algorithm and popularized it (the original repo is also linked above).
I've already watched the videos I linked above. I think they provided me a more intuitive understanding of the repo and I'll probably revisit them if I become confused by the more technical jargon of the papers. Also, the video themselves reference the original papers and provide me a good overview of how/where to start.
The article on Celeste's tileset was useful in understanding the rules behind how Celeste laid out its levels. This will be used for developing the ruleset.
This is how I imagine the application will look like:
- Milestone 1 (11/13)
- Project and Unity setup
- Working basic implementation with a simpler tileset (basic pipe-looking structures)
- No GUI or customization, should just run upon playing the game
- By 11/20
- Implement the grid editor
- Ability to pan and zoom around
- Adjust grid width and height
- Generate button
- Milestone 2 (11/25)
- Import Celeste's tileset and develop customized tileset
- Add a player controller and mimic Celeste's movement as best as I can and make the generated output actually playable
- Ability to select a tile and edit its constraints in the editor
- Final (12/2)
- Final polish, fix bugs, making the UI look good, etc.
- Any of the stretch goals
I've successfully reached my milestone 1 goals and then some. In particular, a basic Unity 6 project was set up with TextMeshPro imported. This allowed me to work on the main focus of this milestone (and overarching project), the actual algorithm. It took a few iterations and rewrites but I think I've settled on a design that is understandable and modular.
I didn't specify it in my goals above but I wanted an easy way to swap in and out tile palettes and their rules. This would make the generator more useful (can switch tilesets at runtime) and made debugging much easier (adding, removing, disabling certain tiles). I was able to do that and probably saved myself a lot of future work.
As I learned more about WFC, it turned out that there are many small decisions I had to make about the implementation. For example, how to actually check which states in a neighbor cell are valid. I ended up on a socket-style system that I could configure via a ScriptableObject before runtime. Again, I'm proud of how modular it is.
I made a lot of dumb mistakes and spent a lot of time debugging issues that turned out to not be related to my WFC implementation at all. It's okay though, everything works now and it's in the past.
Using a simple tile palette made out of interconnecting pipes, I'm already able to generate interesting-looking patterns. I took the opportunity to also implement some milestone 2 features early, including the ability to specify grid width/height and the ability to "step" through iterations of the algorithm (instead of it all being solved at once). All of these helped with debugging. Take a look:
For my second milestone I mostly worked on the grid editor and the tile constraining functionality. There's an actual UI now, and you're able to select a tile from the grid and pick from one of its possible states to collapse to.
milestone2.mp4
Video can also be found in
media/
folder, calledmilestone2.mp4
.
I was not able to finish all the goals I set. Namely, importing the Celeste tileset as well as the player controller. The grid editor took more time than expected to implement, and honestly I think I tried doing too much at once. Timeline wise I should still be fine, because my generator is 100% done and my plans for the final week were bugfixing and polish, so I'm not too far behind.
I've also begun looking for Celeste's tilesets already. Out of all places I found this premade Celeste tileset for Pizza Tower, which has everything I need. As for the player controller, I'll basically be copying the video I listed in my references above.
For the final results, please see the showcase video above. It demonstrates all the features and many different executions of the algorithm.
On the surface the final submission doesn't really differ from milestone 2, but in the backend things were heavily modified. In particular, I rewrote stuff so the algorithm can handle multiple tilesets, introduced a "fold"/"unfold" flag, and fixed many bugs, including a big one: the probability of a tile being chosen isn't affected by how many orientations it has anymore.
I also added player movement and a whole new scene where you can play the generated level as fake Madeline.
It took me longer than expected to import Celeste's sprites. Every bit of progress I made I realized that some assumption I made was wrong or hardcoded. So in the process of adding the game sprites I touched every part of the codebase. But now everything is better than ever!
Overall, I achieved everything I wanted. I learned how to implement one possible version of WFC, I learned more about C# and its many upsides and downsides, and I got very familiar with Unity tilemaps, especially runtime generation.
I'm a bit sad that I wasn't able to get to any of the stretch goals. I mean, yeah, they're stretch goals, but it would've been really cool to do a live demo of me playing the level I generated within the actual game. When I have more time I would love to revisit any of those goals and further expand the project. I see so much potential here and I'm sure I'll slowly hack away at stuff for the foreseeable future.
One interesting thing is that over the course of the project, my mindset towards the end product shifted. Originally I put more emphasis on being able to play the generated level, but over time I grew to see CelesteWFC as a tool more than a game. As I wrote code I kept thinking to myself, "this could be more modular and generalized." I ended up with a system that didn't really have anything to do with Celeste at all! In fact, I did player movement in the last few days; it feels more like a demo than an integral part of the project.
List of stuff I can rewrite to speed up algorithm propagation that I ran out of time for:
- Storing the states more efficiently in
Cell
so that we don't have to create and destroy the dictionary every time we callCollapse()
, or some other way of tracking the "weight" of a state- Same with
PickLowestEntropyCell()
- Same with
- Get rid of 2D for loop to check if whole grid is collapsed in
IsCollapsed()