The next little game in the series where I make a fancy rogue game.
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.
 
 
 
 
 
 
Zed A. Shaw c11384df59 Forgot the little matrix wrapper. 1 month ago
assets Now have a basic prototype lighting system. 1 month ago
scratchpad Now have more test for the base functionality but need to push render tests and find a way to test the GUI. I've also brought in sol2 for lua integration but not sure what to do with it. 1 month ago
scripts First coverage reports. Tests don't get near enough coverage but running the program does. 1 month ago
tests Map is now cleaned out of anything not directly related to the map, and there's a new WorldBuilder class that will turn into a more sophisticated random world generator. 1 month ago
wraps Now have more test for the base functionality but need to push render tests and find a way to test the GUI. I've also brought in sol2 for lua integration but not sure what to do with it. 1 month ago
.gdbinit FINALLLYYYY!!! Can't believe it took so long to solve this but now I can gdb and catch aborts. 2 months ago
.gitignore Ignore more irrelevant things. 3 months ago
.tarpit.json Ignore more irrelevant things. 3 months ago
.vimrc_proj Mostly working prototype that uses FTXUI to render to SFML and then plays a sound when you hit a wall. 3 months ago
LICENSE Initial commit 3 months ago
Makefile A bit of cleanup and refinement before refactoring. 1 month ago
README.md Forgot the little matrix wrapper. 1 month ago
ansi_parser.cpp Now have a basic prototype lighting system. 1 month ago
ansi_parser.hpp Rewrote the ansi parser to exactly callback on color setting, so now just need to clean this all up and fix a few little bugs. 2 months ago
ansi_parser.rl Now have a basic prototype lighting system. 1 month ago
collider.cpp Light works with multiple sources and strengths, walls are faked out but I think I may keep that to make it easier to play. 1 month ago
collider.hpp Light works with multiple sources and strengths, walls are faked out but I think I may keep that to make it easier to play. 1 month ago
color.hpp Color is now in one nice location. 2 months ago
combat.cpp Almost working save sytem but the data I store is totally wrong. I need to also save the entity IDs being used and map them to the components. 2 months ago
combat.hpp First coverage reports. Tests don't get near enough coverage but running the program does. 1 month ago
components.hpp Refactor the light calculations to be part of map instead of spread out all over. Still need to bring back lighting on walls and also pathing for enemies is currently busted. 1 month ago
config.cpp Have to force the terminal detected by FTXUI to use full color no matter what it detects in the actual terminal. 2 months ago
config.hpp Have to force the terminal detected by FTXUI to use full color no matter what it detects in the actual terminal. 2 months ago
dbc.cpp Initial commit that has most of what I need. 3 months ago
dbc.hpp Initial commit that has most of what I need. 3 months ago
dinkyecs.hpp Event system now accepts any data and the GUI receives simpler events with data for them. 2 months ago
events.hpp Event system now accepts any data and the GUI receives simpler events with data for them. 2 months ago
flecs.wrap Brought in FLECS to play with, tomorrow we learn it. 3 months ago
fsm.hpp Conver to using \ for member variables in classes. In structs just use the name. 3 months ago
gui.cpp Lighting is now in its own class using the new Pathing class. This should allow me to make it more consistent and possibly make Pathing more efficient. 1 month ago
gui.hpp Lighting is now in its own class using the new Pathing class. This should allow me to make it more consistent and possibly make Pathing more efficient. 1 month ago
lights.cpp Lighting is now in its own class using the new Pathing class. This should allow me to make it more consistent and possibly make Pathing more efficient. 1 month ago
lights.hpp Lighting is now in its own class using the new Pathing class. This should allow me to make it more consistent and possibly make Pathing more efficient. 1 month ago
main.cpp Map is now cleaned out of anything not directly related to the map, and there's a new WorldBuilder class that will turn into a more sophisticated random world generator. 1 month ago
map.cpp Map is now cleaned out of anything not directly related to the map, and there's a new WorldBuilder class that will turn into a more sophisticated random world generator. 1 month ago
map.hpp Map is now cleaned out of anything not directly related to the map, and there's a new WorldBuilder class that will turn into a more sophisticated random world generator. 1 month ago
matrix.hpp Forgot the little matrix wrapper. 1 month ago
meson.build Map is now cleaned out of anything not directly related to the map, and there's a new WorldBuilder class that will turn into a more sophisticated random world generator. 1 month ago
panel.cpp Now have more test for the base functionality but need to push render tests and find a way to test the GUI. I've also brought in sol2 for lua integration but not sure what to do with it. 1 month ago
panel.hpp Now have more test for the base functionality but need to push render tests and find a way to test the GUI. I've also brought in sol2 for lua integration but not sure what to do with it. 1 month ago
pathing.cpp Lighting is now in its own class using the new Pathing class. This should allow me to make it more consistent and possibly make Pathing more efficient. 1 month ago
pathing.hpp Lighting is now in its own class using the new Pathing class. This should allow me to make it more consistent and possibly make Pathing more efficient. 1 month ago
point.hpp Lighting is now in its own class using the new Pathing class. This should allow me to make it more consistent and possibly make Pathing more efficient. 1 month ago
rand.cpp Just wrote my own entity system to figure it out. 3 months ago
rand.hpp Just wrote my own entity system to figure it out. 3 months ago
render.cpp A bit of some clean up, API unifying, and some performance tweaks. 2 months ago
render.hpp A bit of some clean up, API unifying, and some performance tweaks. 2 months ago
save.cpp Lighting is now in its own class using the new Pathing class. This should allow me to make it more consistent and possibly make Pathing more efficient. 1 month ago
save.hpp More notes on the next things to do. 2 months ago
sound.cpp Rendering code stripped out of the GUI code. 2 months ago
sound.hpp Rendering code stripped out of the GUI code. 2 months ago
status.txt Created an initial README. 1 month ago
systems.cpp Lighting is now in its own class using the new Pathing class. This should allow me to make it more consistent and possibly make Pathing more efficient. 1 month ago
systems.hpp Lighting is now in its own class using the new Pathing class. This should allow me to make it more consistent and possibly make Pathing more efficient. 1 month ago
tser.hpp The json output is a problem but I don't even use it. 2 months ago
tser.wrap Stripped tser.hpp down to the essentials so I can study it. No base64 encoding, less than comparison (wtf is that for), and I may even remove the 'json' output. 2 months ago
worldbuilder.cpp Map is now cleaned out of anything not directly related to the map, and there's a new WorldBuilder class that will turn into a more sophisticated random world generator. 1 month ago
worldbuilder.hpp Map is now cleaned out of anything not directly related to the map, and there's a new WorldBuilder class that will turn into a more sophisticated random world generator. 1 month ago

README.md

An Homage to Rogue

When I was a kid I played a game that used only text characters in a terminal called [Rogue](https://en.wikipedia.org/wiki/Rogue_(video_game). This game ran in my little MSDOS computer using only the ASCII characters the computer supported. No graphics and I think no sound but I can't remember. Despite this very low quality graphical experience the game delivered an amazing game experience. Even today the term "roguelike" denotes certain qualities that come from the original Rogue:

  • Dying is permanent but each time you restart you'll retain some previous skills or equipment so the next run is easier.
  • However, it's only easier to the point where you previously died, as each level through the dungeon gets more and more difficult.
  • A randomly generated world (dungeon) that features a new experience every time.
  • A focus on mechanics rather than graphics, but creative use of the limited graphical characters to make the game playable.
  • This lack of graphics weirdly makes the game more imaginative because you have to fill in what's going on with your own imagination, similar to a game of Dungeons and Dragons.

This is why I chose Rogue as the basis for my first actual game in C++.

A "Classical" Game Development Education

In the world of art there's a concept of a "classical education," which means you learn to paint and draw using old methods meant to teach a realistic style. These methods are hundreds--maybe even thousands--of years old and work very well...assuming you have the patience. Learning to paint and draw like this is almost an art history lesson since not many people do art using these techniques anymore.

I'm planning to do my study of game development in the same way by making games in historic order. Rather than run off and start using Unreal Engine immediately I'm going to make an homage to each kind of game I played when I was younger. Each game I make will hopefully teach me both the history of Game Development and also a technique that's either very useful or long forgotten.

My little Rogue game is the first entry in this series.

It's Not 1980 Anymore

My goal is to maintain the constraint of "Text Only, but not Terminal Only." What I mean by this is I'll use only text glyphs for the graphics in the game but I won't restrict myself to the textual grid, layout, or traditional Terminal constraints. I'm rendering these characters using modern graphics capabilities that allow me to create a more fun experience.

For example, I code the whole UI in a Terminal UI (TUI) library named FTXUI but I render the ANSI terminal output of FTXUI into an SFML window using an ANSI Code Parser that renders to a Render Pipeline. This lets me render full Text UIs at one scale while rendering the map in a different scale and allow zooming.

I also use a full true color encoding so I can implement things like lighting effects and complex color changes. I even have sounds, simple camera shake effects when you take damage, and other things found in top-down games.

In addition to that, I'm not limited to one character per cell. Since I'm only treating characters as sprites I'm actually able to layer them in the same cell, so I can show weapons being held, damage effects, and other changes using character on the player or other entities.

Why Not Just Use Pixel Art?

I mean...I am...it just looks like text. An Homage doesn't mean you copy something exactly. It means you create something that's your own, but references or takes things from another artist's work. It's a respectful copy that doesn't really steal but instead pays respects to those who came before while still creating something new for the future.

In my game I'm using the constraint of "Text Only, not Terminal Only" to give me an artistic constraint and to keep the game an homage to all the Rogues that came before.

But also, I mean c'mon folks, it's 2025. We do have sound now.

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 shit. 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

Pre-requisites:

  • Meson -- which need Python
  • C++ Compiler -- Tested with Clang and G++
  • GNU make -- For the convenience Makefile

Windows instructions

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

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

# first compile takes a while
make

# will play a sound and open windows
make test

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

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?
  • This points out a problem that I'm getting the keys using FTXUI but should either get them from SFML or connect FTXUI to SFML's keyboard input events instead.
  • No actually this first run delay seems to be related to the security feature that blocks keyboard access on iTerm, so probably fixing that would speed it up.