Dungeon Wizard: Action roguelike deckbuilder in Zig

I spent about 6 months making this game in Zig (with Raylib), and recently released it for download on itch:

You play as a Wizard who fights their way through a series of dungeon rooms, adding new spells to your deck along the way.

dungeon-wizard-gif-1

dungeon-wizard-gif-3

dungeon-wizard-gif-2


This project tested both my game design and programming skills - it is by far my most complex project to date and I’m really proud of it!

I had to build (or hack together) many systems along the way, including:

  • Game code hot reload
  • Asset loading and lookup (with hot reload of assets)
  • Frame-by-frame rewind for debugging the game state
  • Custom icons and formatting encoded and decoded into unicode
  • Polyphonic sound playback system built on top of Raylib’s sound API
  • Entities and their complex interactions - hit/hurtboxes, physics, animation, creature AI, spells, abilities…

You can check out the code on github:

Some reflections on my programming style and Zig-specific things:

  • I use BoundedBuffer almost everywhere I need an array, including any strings I need to store in the game state.

  • The whole game state can be copied by value (well, almost) and that’s how I implemented debug rewind, just copy the state each frame into a circular buffer.

  • Lots of unions. Anytime I wanted something ‘polymorphic’ like different spells, AI behaviors, projectiles, entity renderers, etc, it’s just a union.

  • comptime is very useful. In most cases I just use it to save typing, like generating an enum and a union from an array of types.

  • The UI code is basically all ad-hoc garbage and was painful to iterate on. Hot code reload made this more bearable.

  • It’s a common observation, but the standard library is very educational and easy to read - I didn’t shy away from using it like I do with C++ because I can just look and see exactly how everything works.

Overall I really enjoy Zig. It’s been over a year since I began using it for personal projects. For me, it hits a nice sweet spot between simplicity and expressiveness.

Feel free to ask questions if you’re curious about anything.

52 Likes

Sick! I’ve tried it on M1 for a couple runs, it played pretty well, didn’t get past 4th room, though, a bit difficult to move and aim with touchpad. Btw, are you building a universal bundle for MacOS (aarch64), or just x86_64? M1 was getting somewhat hot as if it was using Rosetta.

3 Likes

Hey, thanks for playing! I definitely recommend playing with a mouse if you can. It might also help to make liberal use of ‘pause’ feature (spacebar), since you can aim your spells while paused.

The Mac version is built for aarch64 only.
I never noticed it heating up my M1 machine, but checking the CPU usage just now it does seem high… and looking at the code I just realized that I turned off sleeping in the main loop last time I worked on the frame timing code! So that doesn’t seem good…hmmm
<1 hour later>
Okay I released an update. CPU usage should be greatly reduced now - Let me know if it helped, and thanks for the feedback!

4 Likes

No Linux? Is it possible to build for Linux?

1 Like

Great question…I myself haven’t used Linux as a desktop OS for some time, so I haven’t tried.

I’m informed the Windows version can be played on Linux with wine, so maybe give that a try?

I tried building for Linux and had to fix a case sensitive issue with a @import("data.zig"); instead of @import("Data.zig");
Now I am stuck on the following error in src/raylib.zig

src/raylib.zig:126:56: error: parameter of type ‘[1]cimport.struct___va_list_tag_1’ not allowed in function with calling convention ‘x86_64_sysv’
fn raylibTraceLog(msg_type: c_int, text: [*c]const u8, args: stdio.va_list) callconv(.c) void {
^~~~~~~~~~~~~~~~~~~
src/raylib.zig:126:56: note: arrays are not allowed as a parameter type

I don’t know if this is a translate-c issue or just something I could override by typing. I tried using an anyopaque pointer but couldn’t get it to work.

1 Like

This branch worked for me linux-fixes. I just ended up disabling the custom logging format, because I don’t know how to get the varargs stuff to work.

4 Likes

Wonderful! I went ahead and pushed a couple of commits which achieve the same thing as your fork, so I suppose main should build for Linux now (can’t actually test it myself).

3 Likes

Yes it works now!

3 Likes

On X11, I was able to compile and run after adding .linux_display_backend = .X11, to raylib_dep in the build script (and install the necessary dependencies). Really nice!

2 Likes

This is AWESOME!!! Immaculate vibes.

The pixel art is great, love the animation between levels, and the tutorial is great too.

I got linux to work after the suggestion from @lufe (I am on debian 12).

const raylib_dep = b.dependency("raylib", .{
        .target = target,
        .optimize = optimize,
        .shared = true,
        .config = raylib_config,
        .linux_display_backend = .X11,
    });

Is there a reason you chose to use git submodule instead of the zig build system to fetch raylib for you?

3 Likes

Very cool - the merchant looks terrifying!

1 Like

Is there a reason you chose to use git submodule instead of the zig build system to fetch raylib for you?

Mainly ignorance I think :sweat_smile:… It is nice to have all the code dependencies right there in the repo instead of fetched and cached somewhere which is harder to find and edit if needed.

At several points in development I hacked around issues with raylib’s build.rs after upgrading my zig version, not sure I could have done that so easily otherwise.

2 Likes

Wow, great to see this beautiful work! Raylib + Zig is a match made in heaven. Thanks for sharing!

1 Like

Is there a zig build option to make it work with Wayland?

install
└─ install raylib
   └─ zig build-lib raylib Debug native failure
error: unable to find dynamic system library 'GLX' using strategy 'paths_first'
error: unable to find dynamic system library 'X11' using strategy 'paths_first'
error: unable to find dynamic system library 'Xcursor' using strategy 'paths_first'
error: unable to find dynamic system library 'Xext' using strategy 'paths_first'
error: unable to find dynamic system library 'Xfixes' using strategy 'paths_first'
error: unable to find dynamic system library 'Xi' using strategy 'paths_first'
error: unable to find dynamic system library 'Xinerama' using strategy 'paths_first'
error: unable to find dynamic system library 'Xrandr' using strategy 'paths_first'
error: unable to find dynamic system library 'Xrender' using strategy 'paths_first'
error: unable to find dynamic system library 'EGL' using strategy 'paths_first'
error: unable to find dynamic system library 'xkbcommon' using strategy 'paths_first'

EDIT: I got it to work with this patch:

--- a/build.zig
+++ b/build.zig
@@ -64,6 +64,7 @@ pub fn buildDynamic(b: *std.Build, target: std.Build.ResolvedTarget, optimize: s
         .optimize = optimize,
         .shared = true,
         .config = raylib_config,
+        .linux_display_backend = .Wayland,
     });
     const raylib = raylib_dep.artifact("raylib");

Combined with this invocation:

nix-shell -p wayland-scanner xorg.libxcb wayland libdrm libGL libxkbcommon
6 Likes

Planning to stream in about 30m and do a “Let’s Play” of this :slight_smile:

14 Likes

Oh cool! I’ll check it out

Heck yeah! NGL my first thought was:

So that’s what it takes to make Andrew stream again – someone making a cool original game in Zig (with Wayland support) :sweat_smile:

Everyone, watch the VOD while you can:

12 Likes

Hey @NunoDasNeves

I got the version from itch and spent a good half hour playing, weirdly captivating! I’m sorta new to zig (sorta…completely). Creating a release build from source with zig build --release=fast gives me a .exe to play (I’m on windows) but that version for me runs REALLY slow, at unplayable FPS. Unsure if I’m missing some parameters to building (zig build for a debug build did the same). Any tips you can give me here? I’m looking forward to playing around with the source.

1 Like

I recommend running zig build -h and looking in the “Project-Specific Options” section. It seems like there is a -Ddo-release option that you can set to “build all targets for release”. So maybe try running zig build -Ddo-release instead of zig build --release=fast. I also looked into the build.zig a bit and it seems like the normal release parameter should be passed through most parts as well, but the do-release parameter has some more usages than just setting build target optimize mode.

2 Likes