A retro style homage to 80s dungeon crawlers hand crafted in C++.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
This repo is archived. You can view files and clone it, but cannot push or open issues/pull-requests.
 
 
 
 
 
 
Zed A. Shaw 6d73c87c4e Debug UI now has little icons for each enemy and can spawn any enemy. Also I forgot the files. 7 months ago
amt The tracy directory now has an experiment in getting Tracy to work. It's _not_ as easy as it is touted to be. 9 months ago
assets Gave up on trying to get the GOAP algorithm to correctly apply the cost structure to competing choices, and instead I take the resulting action list and simply find the next best one based on cost. 7 months ago
magick Some quick hacks that do the seamless texture splits. 9 months ago
scratchpad Bring over the dbg.h to see if it's usefule. 7 months ago
scripts After some prototyping I have what I think I want for the map. Just a simple piece of paper you take out that has the ASCII map on it. 7 months ago
shaders The lighting now uses the original grid based lighting calculations rather than a global single source from the player. 9 months ago
tests This does a 'fit_sort' whenever the state is changed. fit_sort effectively sorts the actions by distance+cost so that the cost is actually present unlike the original algorithm. 7 months ago
tools New debug_ui that shows perf data, other debug info, and allows spawning enemies. 7 months ago
tracy The tracy directory now has an experiment in getting Tracy to work. It's _not_ as easy as it is touted to be. 9 months ago
wraps Did a full code coverage review and improved many of the tests and a bunch of code. I'll do one more final walk through all the code before getting back to work on the new combat system. 7 months ago
.gdbinit My gdb debug thing. 9 months ago
.gitignore A bit more cleanup to avoid duplicate testing and to separate the GOAP algorithm code from the little AI Manager thing. 8 months ago
.vimrc_proj First build that actually works. SDL_main errors before but didn't figure out the cause. 10 months ago
Makefile Gave up on trying to get the GOAP algorithm to correctly apply the cost structure to competing choices, and instead I take the resulting action list and simply find the next best one based on cost. 7 months ago
README.md AI now follows the A* algorithm more closely by using a separate priority queue from the open_set. 7 months ago
ai.cpp Actually I can just use the ai::distance_to_goal function on the fit_sort to sort by cost and distance. 7 months ago
ai.hpp This does a 'fit_sort' whenever the state is changed. fit_sort effectively sorts the actions by distance+cost so that the cost is actually present unlike the original algorithm. 7 months ago
ai_debug.cpp Gave up on trying to get the GOAP algorithm to correctly apply the cost structure to competing choices, and instead I take the resulting action list and simply find the next best one based on cost. 7 months ago
ai_debug.hpp Gave up on trying to get the GOAP algorithm to correctly apply the cost structure to competing choices, and instead I take the resulting action list and simply find the next best one based on cost. 7 months ago
animation.cpp Fixed some of the easing functions but still not sure with RAT_GIANT doesn't move. 7 months ago
animation.hpp Basic Ritual crafting UI is prototyped, so next step is to create some items and refine the UI with a possible FSM to keep it organized. 7 months ago
autowalker.cpp Converted almost everything to use wstring so that it works better with SFML and the unicode/utf8 usage in the system. 7 months ago
autowalker.hpp Converted almost everything to use wstring so that it works better with SFML and the unicode/utf8 usage in the system. 7 months ago
boss_fight_ui.cpp Converted almost everything to use wstring so that it works better with SFML and the unicode/utf8 usage in the system. 7 months ago
boss_fight_ui.hpp Basic ability to create a 'stage' for a boss fight, which is a thing in front the boss animates behind. 8 months ago
camera.cpp More refactoring of the gui. Now most things are out of the FSM and MainUI is responsible for the rayvew and its overlay. 8 months ago
camera.hpp More refactoring of the gui. Now most things are out of the FSM and MainUI is responsible for the rayvew and its overlay. 8 months ago
color.hpp I now have a semi-functional GUI system that uses the ECS style to build gui elements rather than inheritance. 8 months ago
combat.cpp Started working on this 'arena tester' tool that would let me load an enemy and test them, but then realized I could just make it so I can spawn enemies in the game. I'm keeping the arena around as it will be useful later as a scriptable testing tool, but for now just spawn and test. 7 months ago
combat_ui.cpp New debug_ui that shows perf data, other debug info, and allows spawning enemies. 7 months ago
combat_ui.hpp New debug_ui that shows perf data, other debug info, and allows spawning enemies. 7 months ago
components.cpp Created a nice utility library for doing animations, and used it in the ritual crafting UI. 7 months ago
components.hpp Map is way better and components::Tile is _vastly_ improved by switching to a wchar_t on display and letting nlohmann::json auto convert it for me. 7 months ago
config.cpp Did a full code coverage review and improved many of the tests and a bunch of code. I'll do one more final walk through all the code before getting back to work on the new combat system. 7 months ago
config.hpp Did a full code coverage review and improved many of the tests and a bunch of code. I'll do one more final walk through all the code before getting back to work on the new combat system. 7 months ago
constants.hpp After some prototyping I have what I think I want for the map. Just a simple piece of paper you take out that has the ASCII map on it. 7 months ago
dbc.cpp AI engine is working and I have a little BattleEngine going but the AI is working better than it should in systems.cpp. Need to find out why then make the BattleEngine avoid running entities that have END in action lists. 7 months ago
dbc.hpp Brought line numbers back for the dbc.cpp logging stuff. May not work in clang because of the bug they have (had?). 7 months ago
dbg.h Fixed that crash and cleaned up more variables for some study next. I might also try out my debug macros. 9 months ago
debug_ui.cpp Debug UI now has little icons for each enemy and can spawn any enemy. Also I forgot the files. 7 months ago
debug_ui.hpp Debug UI now has little icons for each enemy and can spawn any enemy. Also I forgot the files. 7 months ago
devices.cpp Game now builds and is using the new dynamic component loading but enemies do not spawn in and device events are really working. Also inventory is a giant bag of fail and needs a rewrite. 9 months ago
dinkyecs.hpp Enemy AI is now prototyped and can find the player and attack them. 7 months ago
easings.hpp Fixed some of the easing functions but still not sure with RAT_GIANT doesn't move. 7 months ago
events.hpp New debug_ui that shows perf data, other debug info, and allows spawning enemies. 7 months ago
fsm.hpp Trying out an FSM for controlling the main loop. 9 months ago
goap.cpp This does a 'fit_sort' whenever the state is changed. fit_sort effectively sorts the actions by distance+cost so that the cost is actually present unlike the original algorithm. 7 months ago
goap.hpp Mostly fixed up but I have to figure out why cost on actions isn't changing the priority. 7 months ago
guecs.cpp Started working on this 'arena tester' tool that would let me load an enemy and test them, but then realized I could just make it so I can spawn enemies in the game. I'm keeping the arena around as it will be useful later as a scriptable testing tool, but for now just spawn and test. 7 months ago
guecs.hpp New debug_ui that shows perf data, other debug info, and allows spawning enemies. 7 months ago
gui_fsm.cpp New debug_ui that shows perf data, other debug info, and allows spawning enemies. 7 months ago
gui_fsm.hpp New debug_ui that shows perf data, other debug info, and allows spawning enemies. 7 months ago
inventory.cpp AI engine is working and I have a little BattleEngine going but the AI is working better than it should in systems.cpp. Need to find out why then make the BattleEngine avoid running entities that have END in action lists. 7 months ago
inventory.hpp Inventory and lighting improved, now to get ready for going down a level and that's most of the game loop working. 8 months ago
json_mods.hpp Can now mark json/components with std::optional and then they can be null/false to disable them. 8 months ago
lel.cpp A simple A* pathing function that works on maps, but I'll be changing it to do the GOAP pathing. 8 months ago
lel.hpp I now have a semi-functional GUI system that uses the ECS style to build gui elements rather than inheritance. 8 months ago
lel_parser.cpp Fixed up dbc.cpp so now just use it everywhere. I next need to find a way to pass that to format automatically. 7 months ago
lel_parser.rl Fixed up dbc.cpp so now just use it everywhere. I next need to find a way to pass that to format automatically. 7 months ago
levelmanager.cpp Started working on this 'arena tester' tool that would let me load an enemy and test them, but then realized I could just make it so I can spawn enemies in the game. I'm keeping the arena around as it will be useful later as a scriptable testing tool, but for now just spawn and test. 7 months ago
levelmanager.hpp Started working on this 'arena tester' tool that would let me load an enemy and test them, but then realized I could just make it so I can spawn enemies in the game. I'm keeping the arena around as it will be useful later as a scriptable testing tool, but for now just spawn and test. 7 months ago
lights.cpp Wasn't even using MAX so changed to just a simple BOOST variable that can become a config. 8 months ago
lights.hpp Wasn't even using MAX so changed to just a simple BOOST variable that can become a config. 8 months ago
main.cpp After some prototyping I have what I think I want for the map. Just a simple piece of paper you take out that has the ASCII map on it. 7 months ago
main_ui.cpp New debug_ui that shows perf data, other debug info, and allows spawning enemies. 7 months ago
main_ui.hpp New debug_ui that shows perf data, other debug info, and allows spawning enemies. 7 months ago
map.cpp Refactor the Map::neighbors so that it's part of pathing where it should be. 8 months ago
map.hpp Better structure on the autowalker, but still gets stuck in some combat situations. Next is after we kill everything we head to the exit. 8 months ago
map_view.cpp Converted almost everything to use wstring so that it works better with SFML and the unicode/utf8 usage in the system. 7 months ago
map_view.hpp Now have a full map and a mini map, but I think the mini map will stop rendering sometimes. 7 months ago
matrix.cpp Autowalker is now using the GOAP AI system and works way better. Still quite a lot of jank in the code but that'll get removed over time. Next thing is being able to detect when its near an item/enemy and properly react. 8 months ago
matrix.hpp Brought over a bunch of code from the roguelike and now will use it to generate a random map. 9 months ago
meson.build New debug_ui that shows perf data, other debug info, and allows spawning enemies. 7 months ago
meson.options Add the tracy wrap and meson options. 9 months ago
mini_map.cpp Now have a full map and a mini map, but I think the mini map will stop rendering sometimes. 7 months ago
mini_map.hpp Converted almost everything to use wstring so that it works better with SFML and the unicode/utf8 usage in the system. 7 months ago
overlay_ui.cpp New debug_ui that shows perf data, other debug info, and allows spawning enemies. 7 months ago
overlay_ui.hpp New debug_ui that shows perf data, other debug info, and allows spawning enemies. 7 months ago
pathing.cpp Figuring out something weird about the Pathing::random_walk code. 8 months ago
pathing.hpp Figuring out something weird about the Pathing::random_walk code. 8 months ago
point.hpp BROKEN: Big refactoring happening, so it compiles but game does not run and the tests fail. 9 months ago
rand.cpp Brought over a bunch of code from the roguelike and now will use it to generate a random map. 9 months ago
rand.hpp Now have a simple stats test. 7 months ago
raycaster.cpp After some prototyping I have what I think I want for the map. Just a simple piece of paper you take out that has the ASCII map on it. 7 months ago
raycaster.hpp After some prototyping I have what I think I want for the map. Just a simple piece of paper you take out that has the ASCII map on it. 7 months ago
ritual_ui.cpp Have a few basic monochrome test items and the first little leather pouches on the 'tool belt'. 7 months ago
ritual_ui.hpp Basic Ritual crafting UI is prototyped, so next step is to create some items and refine the UI with a possible FSM to keep it organized. 7 months ago
rituals.cpp AI now follows the A* algorithm more closely by using a separate priority queue from the open_set. 7 months ago
rituals.hpp AI engine is working and I have a little BattleEngine going but the AI is working better than it should in systems.cpp. Need to find out why then make the BattleEngine avoid running entities that have END in action lists. 7 months ago
save.cpp Did a full code coverage review and improved many of the tests and a bunch of code. I'll do one more final walk through all the code before getting back to work on the new combat system. 7 months ago
save.hpp Did a full code coverage review and improved many of the tests and a bunch of code. I'll do one more final walk through all the code before getting back to work on the new combat system. 7 months ago
shiterator.hpp Brought over a bunch of code from the roguelike and now will use it to generate a random map. 9 months ago
sound.cpp Boss fight looking better, but I need to get this bounce animation in the main game fights. 8 months ago
sound.hpp Boss fight looking better, but I need to get this bounce animation in the main game fights. 8 months ago
spatialmap.cpp Performance check showed that I was checking every sprite even if they're way far away so now just do ones near-ish. 8 months ago
spatialmap.hpp Performance check showed that I was checking every sprite even if they're way far away so now just do ones near-ish. 8 months ago
stats.cpp Fixed up dbc.cpp so now just use it everywhere. I next need to find a way to pass that to format automatically. 7 months ago
stats.hpp Now have a simple stats test. 7 months ago
status_ui.cpp Converted almost everything to use wstring so that it works better with SFML and the unicode/utf8 usage in the system. 7 months ago
status_ui.hpp Converted almost everything to use wstring so that it works better with SFML and the unicode/utf8 usage in the system. 7 months ago
systems.cpp AI engine is working and I have a little BattleEngine going but the AI is working better than it should in systems.cpp. Need to find out why then make the BattleEngine avoid running entities that have END in action lists. 7 months ago
systems.hpp Map is way better and components::Tile is _vastly_ improved by switching to a wchar_t on display and letting nlohmann::json auto convert it for me. 7 months ago
textures.cpp Started working on this 'arena tester' tool that would let me load an enemy and test them, but then realized I could just make it so I can spawn enemies in the game. I'm keeping the arena around as it will be useful later as a scriptable testing tool, but for now just spawn and test. 7 months ago
textures.hpp Level traversal works better now, compass is accurate, and direction is maintained when you traverse. 8 months ago
tilemap.cpp Map is way better and components::Tile is _vastly_ improved by switching to a wchar_t on display and letting nlohmann::json auto convert it for me. 7 months ago
tilemap.hpp After some prototyping I have what I think I want for the map. Just a simple piece of paper you take out that has the ASCII map on it. 7 months ago
worldbuilder.cpp Started working on this 'arena tester' tool that would let me load an enemy and test them, but then realized I could just make it so I can spawn enemies in the game. I'm keeping the arena around as it will be useful later as a scriptable testing tool, but for now just spawn and test. 7 months ago
worldbuilder.hpp Inventory system basically works now but is in a alpha hack stage. Time to refactor. 8 months ago

README.md

The Artisanal Handcrafted Retro-Future "3D" Dungeon Crawler

Welcome to my latest obsession, and turn based dungeon crawler in the style of old school raycasted games like Wizardry, Might and Magic, Ultima, and similar games. The game uses SFML 3.x as it's "cross platform layer" but other than that everything is hand coded by me. It's fully artisinal, created manually, with nothing but a terminal and vim. No LSPs, AI, or anything.

This code is truly a work of art. Like an espresso at that Speakeasy Coffee bar in Brooklyn nobody talks about. You know the one? You don't? Oh sorry, I thought you were cool.

STATUS

Currently it's only officially tested on Windows, but I'm not really using anything OS specific (I think).

Where's the LICENSE?

You don't need a LICENSE that gives everything away to thieving corporations just to publish your works online. Nobody makes artists, musicians, painters, photographers, or sculptors get a license before posting online, so why do programmers need one? You worried you'll get sued? Ok, so just put a disclaimer but why do you also have to give your hard work away for anyone to steal and profit from just so they don't sue you?

You don't, and no matter what the OSI says, nobody can sue you if they steal your code and cause a plane to crash. They would get sued for stealing your code and putting it in a plane, not you. Requiring only programmers to release their code with a license to avoid lawsuits creates a Chilling Effect on programmer free speech and that violates the First Amendment.

So this code isn't licensed, it's copyright by default. I'm publishing it using my free speech rights to express myself and that means you can look at it the same as if I posted a painting or an essay on my blog. I obviously can't sue you for just looking at it and playing the game because I published it so you can, but that doesn't mean you own it. You can't resell it, fork it, nothing.

Just grab the code and play it. That's it. Tell people about it. Fair use says you can even record videos reviewing it and talking about it.

See? That's how Free Speech works. You don't need a LICENSE.

Build Instructions

On all platforms you'll need these components:

  • Meson -- which needs Python.
  • C++ Compiler -- Tested with Clang and GCC 14.2.0. You can use my Windows C++ Setup Guide which features an automated installer for Windows.
  • GNU make -- For the convenience Makefile. On Windows you should have this if you used my setup scripts. Otherwise winget install ezwinports.make will set you up.
  • Ninja -- Meson uses this to do builds on most systems.
  • git -- Which should be on almost every platform, and is installed by default with my Windows setup scripts.

Windows Instructions

I primarily develop in Windows using the above setup, so this should work the best. Open Windows Terminal and run these commands one at a time. Don't copy-past bomb this:

git clone https://git.learnjsthehardway.com/learn-code-the-hard-way/raycaster.git

cd raycaster

# ignore the errors the first time
./scripts/reset_build.ps1

# first compile takes a while
make

# this copies the binary so you can run it
make run

After that the game should be running. It'll be in different states depending on how far I've pushed it, but you should at least have a few enemies, some loot, and rooms light in it. Go find them.

Linux and OSX

Linux and OSX have the same requirements as Windows and almost the same install steps. The only difference is that once you get your developer tools installed then you only need Meson. Linux and OSX should have everything else you need or there's a package for it.

Once you have that installed you can run these commands:

git clone https://git.learnjsthehardway.com/learn-code-the-hard-way/raycaster.git

cd raycaster

# ignore the errors the first time
./scripts/reset_build.sh

# first compile takes a while
make

./builddir/zedcaster

You don't need make run because Linux and OSX are sane operating systems that don't lock every damn thing a process touches.

Other Platforms

No testing done on other platforms but let me know if you get it to build somewhere fun and I'll mention it.

Development Guide

You can look in the notes.txt file for my informal TODO list of things to fix and make. I'm not really accepting contributions from others, but if you want to follow along then that's what I'm doing.

If you're just starting out in C++ or programming then the project is designed to be readable by someone who knows very little. Every file is small and should be easy to read. I don't use any insane tricks or weird C++ idioms. I also try to avoid too many external libraries so I'll use plain old std::vector and std::unordered_map rather than external libraries that might be faster. This is done on purpose so people (myself included) can learn about the basics of C++ and the STL.

I also don't do a lot of performance tuning or obsession over THE CACHE. Clean, simple, readable code is more important than squeezing 4% performance out of the code. I do however attempt to design things so that it doesn't do useless work because the fastest thing you can do in a computer is nothing. If I can architect away a performance issue and not make the code too complex then I'll do that instead.

That means if you have a suggestion for a micro-benchmark improvement that will dramatically boost performance, but the code is convoluted and hard to understand, then it won't work. If your suggestion is interesting and provides a massive boost then let me know and I'll check it out. But, I would also like statistics that show it's better, not just your word.

Known Bugs

It's early so probably a bunch of bugs.

Linux Build Notes

Libraries Needed:

  • libxi-dev
  • libfreetype-dev

It uses c++ so you may need to install a libg++ or libc++ for your system. Usually this is all you need:

apt install build-essential

OSX Build Notes

  • Quite a bad experience. Need to install Python, cmake, meson, and ninja all which are in homebrew but if you don't use homebrew then this is a problem.
  • You need to run the .command script in Application/your python that updates the SSL certs.
  • You have to give iTerm access to your keystrokes...because wtf it already has them?