Skip to main content

Indie game storeFree gamesFun gamesHorror games
Game developmentAssetsComics
SalesBundles
Jobs
TagsGame Engines

Built a custom engine for my game, looking to see if anyone else is interested in using/testing it.

A topic by ScarletTwizby created Dec 05, 2023 Views: 237 Replies: 3
Viewing posts 1 to 2

I've been working on a game for some time now, and while it's been slow going lately due to getting a day job and having to replace my computer earlier in the year, I've finally gotten around to releasing the engine for the game as an open source project on git hub here: https://github.com/tstwizby/ScarletCore/tree/master

While at the moment the documentation is somewhat opaque (and my coding style is objectively cursed) I would be interested in working with people to develop it further as its own project, separate from the game (which I'm partly developing as a proof-of-concept for the engine).

As for some details: The engine is written in C, using SDL for IO, and is intended to be extremely lightweight and portable. In the short term I'm only actively working on a Windows version, but a Linux version is trivial to implement and a Mac version requires very little tweaking (though due to Apple's current security practices I have no ability to actually distribute a Mac version of the engine at this time). It is intended to be used mainly for Visual Novel/Adventure style games, though it has some flexibility, and development at the moment is closely tied to development of my game (with features being added to the engine as they are required for the game).

Games running on the engine (currently named ScarletCore) consist of a number of 'locations', each of which has a background image and a collection of 'scripts' associated with it describing how the game behaves. There are a number of variables tracking data in the game, including both default variables like 'time of day' or 'weather', as well as user-defined variables, which can  be used to control script behavior. Backgrounds can be dynamically generated depending on the time of day, weather, etc. Scripts are compiled both for obfuscation and to slightly improve performance, since one of the goals of the engine is to have it be as lightweight as I can make it. Right now a lot of the details are not set in stone, for obvious reasons, and I would appreciate any feedback people are willing to give me. While I'm unlikely to work on tools for the engine much in the near future, I would be happy to talk with anyone interested in learning more directly.

(+1)

First off: amazing to see this. I don’t make visual novels myself, but I’m excited to see anything new and low-end.

You’re aware that your code is kind of screwed up, so I won’t mention that.

Please, please, reconsider your scripting. In my last project I used an XML parser to read levels, and over time it morphed into some completely hideous programming language. I know for a fact that it would’ve tainted people’s modding experience, so badly even, it could’ve made a cult following. Next time I’m just using Lua or something else. This is also, for the record, coming from someone who’s already dabbled in compiler/interpreter design for years, and is “working” on a compiler of his own – quotes because everything is on indefinite hiatus.

If you’re worried about even Lua being too bloated, there exist other ones, but Lua is most well-known and it’s very easy to integrate. You can keep the important state like the weather and time in the main engine, and have Lua access it through metatables and the __index & __newindex metamethods.

A Lisp would be even more lightweight, but also more exotic. Still better than some XML-based crap, and uh.. your crap as well (meant endearingly :) ).

Heh, no worries, I totally understand where you're coming from and for the record consider you objectively correct. The following paragraphs are a description of a personal challenge I set myself for this project and not what I consider 'good practice' in general. In short, this is not intended as an 'argument in favor' of doing things the way I did them, but instead just an 'explanation of why' I did.

Looking back at my initial post, I didn't make it as clear as I should have that a lot of my poor design decisions are intentional as part of my goal for this project. Since I'm not making this professionally, I can get away with doing things the "wrong" way for experience/learning purposes (to paraphrase a quote I can't quite remember off the top of my head, you build things from the ground up when you're learning so that you know how things work, then use off-the-shelf stuff in practice to save time). I do want this to be as usable for people as I can make it within these parameters, but part of my goal is to have built as much of it as possible myself (with SDL being a compromise for portability).

I did consider using Lua, before deciding that writing my own language would be simple enough that I should do it in accordance with the last paragraph, but decided against it for two mostly philosophical rather than practical reasons. The first is that I already knew I was going to need SDL to realistically make the engine portable, and so I already had to both learn how to use/debug SDL, learn how to program a game engine, and integrate my engine with SDL to complete the project, and adding Lua to that would mean that I would have to also learn how to use/debug Lua, learn how to interface Lua with SDL, and integrate my engine with Lua. Making my own scripting language on the other hand I would just need to... make my own scripting language, it would necessarily fit into my engine 'for free' and since it wouldn't have its own subsystem running in the background as it were it wouldn't be able to conflict with SDL in any way. Also, using Lua inherently limits the design space of the engine. Again, I didn't spend enough time learning it to know how significant the impact would actually be, but I definitely both don't need all the power that Lua offers and might want extremely specific things that Lua cannot (easily) provide. Off the top of my head, some things I was concerned about were a means for customizing text display via randomly varying font (which isn't something the game currently uses but that I have ideas for down the line) and how easily it would interface with SDL to update config variables. Again, not something I know Lua *can't* do easily, but something that I do know is a non-issue with my own engine. Concerns I'd have about switching now (admittedly colored by what I've already accomplished) are how it would handle the dynamic background system and time-based scripts.


TLDR, it all comes down to opportunity cost of time spent learning how to do things. If I were already familiar with Lua I quite possibly would have made a different decision, but in the end I wanted this to be a project for learning 'how to make a game engine' rather than 'how to integrate Lua into a game engine', and I will save that project for another day.


Also, just for the record, you may be overestimating the complexity of the scripting portion of the engine based on your suggestions. It's not meant to be a fully-featured programming language in-and-of itself, but just a simple human-read/writeable way to encode game-specific logic independent of the engine as a whole. Since I don't think there are any in the project as is currently available, here are a few example scripts, that (assuming the player is in a location called "roomA" with exits left to "roomB" and right to "roomC") plays a crashing sound, and blocks the player from moving left until they've moved right to investigate:

crash:{playSE(crash, 100, 100) wait(50)}_Huh? What was that noise?@I think it came from the right.{setlocscript(roomA, left, nowait, 0) setlocscript(roomB, loc, checksound, 0)}

nowait:Hold on, I should see what happened to the right.

checksound:Oh, looks like a lamp just fell over.{setlocscript(roomC,loc,none,0) setlocdest(roomA, left, roomC, 1)}


'@' means 'wait for player input, then clear the textbox and continue', '_' represents an ellipses, and the part left of the ':' is the name of each of the three scripts. playSE plays a sound effect, modifying the speed and volume of playback, and setlocscript/setlocdest change what happens when you try to go in a direction at a given location to either running the given script or moving to the given location.