Skip to content

Latest commit

 

History

History
221 lines (138 loc) · 14.3 KB

README.md

File metadata and controls

221 lines (138 loc) · 14.3 KB

OCaml Boulder Dash

Boulder Dash was my favourite computer game in the 8-bit era, first released on the Atari 400/800 in 1984. Though I never owned an 8-bit machine myself, I had friends that I annoyed enough to let me play it on theirs.

This is a homage to its 40th anniversary in form of a fairly faithful and feature-complete clone of the original game and its first successor, implemented in just a few lines of OCaml.

Version 2 of it was an excuse for me to mess around with the OCaml bindings to popular graphics and media APIs, and hence, it comes with 3 possible backends:

  1. the homely bare OCaml Graphics library,
  2. the TSDL binding to the SDL API,
  3. the Raylib binding to the Raylib game engine.

The above is in order of increasingly better user experience, for the price of a potentially harder build experience. Some unfortunate limitations apply to the Graphics library option in particular.

In theory, all versions should run on Windows, Mac, and Linux, though I was too lazy to test all combinations, and had trouble installing some of the dependencies on some of the systems.

Features

  • Faithful original physics, graphics, animations, sound, and music
  • Authentic scrolling mechanics combined with dynamic resizing
  • All 40 levels and 5 difficulties of Boulder Dash 1 & 2
  • Pause-and-go mode for relaxed playing

If you have previously looked at this, version 2 added the following niceties:

  • Support for SDL and Raylib graphics engines
  • Original sound effects and music
  • Original level color schemes
  • Full screen mode
  • Dynamic graphics scaling adjustment
  • Gamepad/joystick support
  • Precise keyboard controls and control via arrow keys
  • Fix behaviour of expired magic wall
  • All Boulder Dash 2 levels

Further changes:

  • 2.0.1: Internal renaming of Engine to Api
  • 2.0.2: A couple of fixes to SDL backend; less sensitive gamepad
  • 2.0.3: Snappier keyboard controls; hide mouse in fullscreen; avoid uncaught Graphics exceptions; build .app on Mac
  • 2.0.4: Further key control refinements; suppress console window on Windows; directory reorg
  • 2.0.5: fix SPACE/5; add help overlay (Raylib only); avoid "true" fullscreen on Windows to keep desktop intact; add icon to Windows exe

Building

There are 3 possible backends, resulting in 3 ways to build the program. Provided you have the necessary dependencies installed, either of the following should work:

dune build src/api-graphics/main_graphics.exe
dune build src/api-tsdl/main_tsdl.exe
dune build src/api-raylib/main_raylib.exe

If you are old-school and lazy like me, you can also simply say one of

make graphics
make tsdl
make raylib

and as a bonus, get a link to the executable in the current directory.

Prerequisites:

  • either the Graphics library (opam install graphics),

    Note: When the graphics didn't work out of the box (e.g., the window cannot be opened on Mac), I found the recipes here here helpful.

  • or the TSDL library (opam install tsdl),

    Note: Installing this library worked fine for me on MacOS, but I had some trouble installing it on Windows, although it should work. Perhaps your system is cleaner.

  • or the Raylib library (opam install raylib).

If you have Nix installed, you can build and run this version with nix run github:rossberg/boulder-dash?dir=platform/nix

Controls

If you are nerd enough to own a gamepad or joystick, the following controls should work (though only with the TSDL or Raylib build):

  • , , , on d-pad - move in the respective direction
  • direction on either analogue stick - likewise
  • A//fire with the above - grab/reach out without moving
  • B/ - do not move for one turn (useful in pause mode)
  • X/ - pause/unpause game
  • Y/ - continue (if died or exited) / kill Rockford (if not)

Alternatively, here you have your old-fashioned keyboard controls:

  • W, A, S, D - move in the respective direction (poor man's joystick)
  • Z, Q, S, D - move in the respective direction if you're French
  • 8, 4, 2, 6 - move in the respective direction on number pad
  • , , , - move in the respective direction on arrow keys (TSDL or Raylib lib only)
  • SHIFT with the above - grab/reach out without moving (poor man's fire button)
  • SPACE, 5 - do not move for one turn (useful in pause mode)
  • SPACE - continue (after Rockford has died or exited)
  • K - kill Rockford (by setting timer to 0)
  • P - pause/unpause game

Meta controls:

  • H - toggle help overlay (Raylib lib only)
  • TAB - skip current level (cheater!)
  • BACKSPACE - go back one level (really?)
  • +, - - increase/decrease difficulty
  • / - advance directly to Boulder Dash 2 levels
  • F - toggle full-screen mode (TSDL or Raylib lib only)
  • [, ] - decrease/increase graphics scaling factor (TSDL or Raylib lib only)
  • ESC - quit game

There are also a couple of cheat codes, but I won't tell.

When paused, you can still move with W, A, S, D and friends, or wait a turn using SPACE. That will effectively unpause the game for a single tick and then immediately pause again, using up a single time slice. If you are old and slow like me, or fed up up with the keybord lag (see below), that allows you to play it like a puzzler rather than a stressful action game.

Beyond that, Boulder Dash is mostly a self-explanatory game. Figuring out the mechanics of some elements is part of the experience. We didn't have a manual in the olden days either.

Oh, and if you don't like the size of the sprites, you can scale the whole image using the [ and ] keys. However, that doesn't work in the Graphics library build, where the only alternative is passing --scale <int> on the command line. The default is 4 (times the original 8-bit sprite sizes, which were designed for the fancy 320x200 displays of the times).

Implementation

I tried to keep the code simple. Unfortunately, it turned out fairly imperative, given the stateful nature of the game and of the graphics library.

The modules are:

  • Cave - the representation of a game level
  • Levels - the original levels data with a decoder
  • Physics - the core game logic for transitioning a cave one tick
  • Game - the main game loop (functorised over backend library)
  • Render - the graphics backend (functorised over backend library)
  • Sound - handling of sound effects (functorised over backend library)
  • Control - handling of keyboard and joystick input (functorised over backend library)
  • Main - common main module (functorised over backend library)
  • Api - unified signature for backend library API
  • api-*/Api_{graphics,tsdl,raylib} - wrapper for respective backend library
  • api-*/Main_{graphics,tsdl,raylib} - main entry point for respective build
  • api-graphics/Bmp - a simple decoder for .bmp files
  • Help - text for help overlay

Not too much surprising to say there, please look at the code for details. For what it's worth, the game logic was straightforward to hack down and worked almost on first try. By far the trickiest part was getting the scrolling logic correct for all edge cases — it mimics the original's famous follow-the-player behaviour, but with arbitrary window resizing thrown into the mix, and I wanted the two to interact smoothly.

Limitations

Each of the 3 backends has its pros and cons. Needless to say that they all sport very imperative interfaces.

Graphics Library

On one hand, I was surprised and thrilled how easy it was to use the Graphics library, and how far I was able to push it with full screen animation and scrolling. I love that it has no external dependencies, making it much less of a headache to build than the alternatives.

On the other, it has a few shortcomings, being the excuse for some limitations to this build of this little game:

  • Controls: Keyboard only and a bit sluggish, since the library only offers buffered keyboard input, no direct key events. For the best experience, set your system's key repeat rate to max and key repeat delay to zero. Also, cursor keys cannot be recognised, so you're stuck with WASD.

    Hint: A particularly dangerous consequence of the key repeat delay is that standing under a rock and then trying to run downwards by holding the S key usually turns out fairly deadly, because the rock will have moved twice already before the key repeat kicks in. For such a maneuver to succeed, you have to press the key individually (and fast) for each step. Sorry about that!

  • Start-up time: When initialising, the program creates a mere 240 tiny images for sprites. Saying that the Graphics library's make_image function is dog-slow on Windows is a polite understatement, as it is literally taking multiple seconds for that, at least on my machine (even worse for larger scaling factors). I'm not sure what's going on there. Also, it raises an exception when the user dares resizes the window during this process, so we have to work around that.

  • Colour schemes: Colour changes between levels as in the original are not implemented, since that would exacerbate the aforementioned image creation hold-up. Did I mention it's reeaallly slow on Windows?

  • Flicker: Window resizing causes terrible flicker, especially because the window background color cannot be changed to something reasonable, like nice, cosy, neutral black. And the library's coordinate system with the origin in the lower left corner of the window is completely at odds with how window resizing works, which causes this coordinate system to be volatile and even racy.

  • No sound: The Graphics library's sound function is a wee bit too lame to produce proper effects, especially since it appears to be blocking.

  • Windows: On Windows, some of the API is defunct, e.g., setting the window size or text size has no effect. Some draw commands are randomly ignored on occasion. And when the user closes the Window, we get unexpected Graphic_failure exceptions.

  • Mac: On MacOs, the library requires X11, so you'll have to install XQuartz.

Finally, because the Graphics library is much more unhurried in general, rendering has to be optimised to only update tiles that either changed or are animated. In situations where a lot is happening or the entire screen changes due to scrolling, the game tends to become less fluid.

TSDL Library

Most of what's needed for this little retro game could easily be done with TSDL, although it is rather low-level, so lacks certain convenience, e.g., for manipulating images. The only real limitation revolved around its sound subsystem:

  • No sound mixing: Although SDL3 can freely mix multiple audio streams, TSDL currently only supports SDL2, which does not have that capability. Since I did not feel like implementing that myself, I hacked around it by using all available audio devices as voices. Depending on your hardware, that could mean that some sound effects are swallowed in noisy situations.

  • Only .wav: In addition, SDL only supports plain wave files, so I can't deploy the sound assets as smaller MP3s without introducing additional dependencies.

  • Key bindings: SDL2 has no way to produce key values relative to modifiers, that's another SDL3 feature. So I wasn't able to customise keyboard layout for users daring to not use an English keyboard.

  • Unsafe: When screwing up and using the API incorrectly, I witnessed crashes in some cases, though I can't reproduce the details. Sometimes even when I didn't do anything wrong, like plugging in a joystick without using any related API.

Raylib Library

Raylib is meant to be easy to use, so has more high-level functionality and is generally feairly pleasent. I only ran into a couple of quirks:

  • Fullscreen bugs: Borderless_windowed mode does not work correctly on MacOS. For true Fullscreen on the other hand, Raylib decides to change the resolution, which messes up Windows desktops. And on Mac, the fullscreen resolution it picks isn't even a natural one for the screen. Worse, when a Mac user clicks on the OS's green Fullscreen button, a segfault ensues.

  • Key bindings: Like TSDL, Raylib cannot hand me processed key codes.

  • Unsafe: Raylib's imperative interface did not prevent me from trying stupid things like creating a texture before opening a window, which it doesn't like much and will respond to with an irritated crash.

Acknowledgements and Resources

License

The source code in this repository is distributed under the Creative Commons Attribution/Non-commercial CC BY-NC 4.0 license.