A full view of the app, with a user interface on the right side of the pixelated map

Project Description

This project was an exciting use of the new concepts I learned in my Data Structures class.

Development

I initially struggled to find a simple way to generate the various continent shapes, but I figured out a method with randomly connecting nodes that were placed around the screen. Once I had a rough version of the project, I got really excited about how I could optimize it and expand the customizability. It was at this point that I added an export feature that allowed me to document the progress I made.

A black and white map with pure polygons

One of the first exported images from the app.

The first method I used to connect the nodes was by just randomly picking a node and jumping around until I met back up with my path. The main issue with this was the polygon perimeters crossing itself which led to less complete continents and a lot of awkward blank space inside of them. To solve this, I created a graph (though I didn't realize at the time) to predetermine which connections were allowed. During this process I removed any overlapping connections so the continents started to look a bit fuller.

A red and black colored graph

A visualization of the graph that appears before continents are drawn on top.

During this time, I was also expirementing with different visual effects I could use to blend the various colors together and make it more realistic. I tried gaussian blur, which required a separate version of pygame, as well as stamping. The stamping process I created takes a bunch of random locations and draws a shape (dot, square, or triangle) based on the center color. This distorts the edges and makes everything looks a lot more natural.

A colored, pixelated continent map

A series of continents refined using polygons.

At this point in the project, it took a significant amount of time to generate the nodes and continent paths. Using my new knowledge of graphs from my Data Structures course, I replaced the awkward object array implementation of the graph with a matrix. I then spent a long time refactoring my classes to support partitioning. While I did see a large improvement from my first version, I never was able to improve the efficiency enough for there to be semi-instant feedback with new nodes or continents. It still takes a few seconds, but that is defintely better than almost 30 seconds of waiting!

These improvements also helped a lot with improving the refinement time, especially with the ocean depths. I decided to try layering the darkness of the water, and from my first attempt, I was blown away with how much it made the maps pop, so I continued to improve it. The main issue came from determining how far the closest land is for each dot drawn in the water. Checking every point is too hard, and checking only some meant that the layers didn't always line up with the land. With the partitioning, I was able to check every point within a range. This does create some rectangular patterns far away from land, it seems to be fine more most cases.

I also made a nice UI with some tools from the pygame_widgets library. I had to change some of the internal calculations to support the customizability, but in the end I think it will definitely be useful for my various fantasy projects. I even made differently colored themes to choose from.

A normally colored pixelated continent map A firey pixelated continent map A parchment colored pixelated continent map

Run It Yourself

If you would like to run this script yourself, clone the repository. Then activate the virtual environment with venv/Scripts/activate and run the program with python map-generator.py.