Skip to main content

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

Canned Heat Game "Engine"...

A topic by oofoe created 93 days ago Views: 431 Replies: 17
Viewing posts 1 to 11
Submitted (1 edit) (+2)

Greetings!

It is with great pride and a certain amount of trepidation (or maybe it's the other way around?) that I would like to announce the first public release of the Canned Heat Game "Engine"! It uses the power of Racket Scheme and @soegaard's Urlang to produce self-contained browser based games that can be enjoyed standalone, or served from itch.io.

Why is "engine" in quotes? Because it's not really a library or framework, but more of a mindset. It implements a simple (very simple) Entity Store and provides some bracketing functionality. This may not sound like so much, but you can use it to build almost any style of computer game, from text-based to 2D or 3D. There are four (more-or-less) complete example 2D games which you are welcome to use as starting points. Or delete most of what's in them and go from there. I don't care...

Aside from the README file, here is the presentation that I just gave at the Toronto Lisp User's Group about it. The examples are also commented (and most of the comments may even be up to date!). More documentation and examples will be added as I have time.

Support? Ask in this topic and I'll try to answer if I can.

Thank you and good luck!

Submitted(+3)

Looks great! Here is the ECS I wrote and will be using: https://github.com/sam-d/klecs

However I am still looking for a way to make my games "web ready". Hoping for hoot to get proper R7RS (or dare I dream, R6RS) support. Best of luck for this jam!

Host

What does "proper R7RS" mean here? eval? R7RS-small was the initial target spec for Hoot, which it supports sans eval (which it will get soon).

Submitted(+1)

I believe I have confused hoot with prescheme (where R7RS support is still ongoing). No offense intended. Perhaps a goal for my next jam: a compatibility layer or rewrite to R7RS of my ECS library and then porting my games to hoot to run in the browser! So much happening in Scheme-land recently to be excited about.

Host

No offense taken! Was just curious if we were missing something I wasn't aware of.

Submitted (1 edit) (+1)

Just because I needed it to start my own entry, there's now a %skel.rkt file in the repo. It's a bare-bones version of Canned Heat, ready for you to plug in all the clever bits!

Submitted (1 edit) (+2)

Hint #2 -- Self documenting controls...

If you add a few extra fields (components) to your key specifications, you can use it to document the controls for the user:

     (store (object [tag 'key] [key "F1"]
                    [do (λ (e) (:= SV.helping (not SV.helping)))]
                    [label "help"] [help "Show/dismiss control help screen."]))
     (store (object [tag 'key] [key "ArrowLeft"] [do key-left]
                    [label "left"] [help "Press to go left."]))

Then add a function to display it when you render:

   (define (draw-help cx)
     "( cx --) Show controls help."
     (let* ([qw (/ SV.screen.width 4)]
            [qh (/ SV.screen.height 4)]
            [x (+ qw 32)] [y (+ qh 32)])
               
       (:= cx 'globalAlpha 0.75)
       (cx.fillRect qw qh (* 2 qw) (* 2 qh))
       (:= cx 'globalAlpha 1.0)
       (:= cx 'fillStyle 'white)
       (:= cx 'font "Bold 16px sans-serif")
       (cx.fillText "CONTROL HELP" x y)
       (+= y 32)
       (:= cx 'font "12px sans-serif")
       (for ([k in-array (qq 'key)])
         (cx.fillText k.key x y)
         (cx.fillText k.label (+ x 80) y)
         (cx.fillText k.help (+ x 128) y)
         (+= y 16))
       ))

Now, when I press [F1], I see this:


One of the great things about this is that if you're using the overlay system, the help will update to whatever keys you currently have assigned.

Just thought of this when wiring up the keys for my own entry and thought I'd share!

Submitted(+1)

This is a great idea, might steal it! Self-documentation is the best.

Submitted(+1)

Hint #3 -- Powerful queries and collision detection...

So, if you've looked at my presentation or read through some of the examples, you know that the "grab/s" functions can query the Store for entities with a particular tag or key value setting. However, you can also feed grab and friends a query function, which can be arbitrarily powerful...

Here is a system that implements 2D collision detection. Yes, there's an example of this in %invade.rkt, but this one is better.

   (define (sys-hit)
     "( --) Collision detection."
     
     (let* ([hero (q1 'hero)]
            [items (grabs (λ (e) (and e.hit
                                      (< hero.x (+ e.x e.w))
                                      (> (+ hero.x hero.w) e.x)
                                      (< hero.y (+ e.y e.h))
                                      (> (+ hero.y hero.h) e.y))))]
            )
       (when (not (empty? items))
         (for ([item in-array items])
           (item.hit item hero)))
       ))
It gets the "hero" entity and runs a function query to check it's position against everything that has a collision handler ("hit"). When it finds them, it runs the handler with the item and the hero as arguments (this is a Canned Heat convention). 

Notice that I check to see if the entity has a collision handler before I attempt the geometry testing -- this make things faster because it only does the expensive comparisons on things that warrant it.

Submitted

Isn't it great how to game jam gives us more idea on how to improve our tooling! I find it interesting that you can query the component values, this is typically left out of ECS.

Submitted (1 edit)

Interesting. It seemed very natural to provide an abstraction for running across the Store and selecting the appropriate things. What do "normal" ECS imp's do? Or is it expected that the user will provide that part?

Submitted (1 edit) (+1)

Hint #4 -- Audio!

Sound is to games what the frame is to a painting. Even a simple effort makes a huge improvement in the overall feel of the game. And thanks to sites like Open Game Art, or Incompetech, there are a wealth of audio resources available. And, of course, every game author who is not me, is also an acid-house-trance DJ on the side. So make sure you use these resources to bring your games to life!

Oh? What's that? You say Canned Heat doesn't have any audio handling capabilities? Hmmm. I guess I didn't get around to putting those in before the jam started. Let's see if we can fix that. First we need to load some sound:

   (define (mk-sound name path)
     "( name path -- entity) Create audio entity."
     (let ([a (new Audio)])
       (:= a.src path)
       (store (object [tag 'audio] [name name] [au a]))))

This function makes an "audio" entity in the Store. It's pretty simple. It creates a web Audio object, sets the source to whatever file you might have for it and drops it in the Store, ready for use. Most modern browsers can load most any audio file format you're likely to use (although FireFox seems to have some problems with %.wav format for some reason). If you don't have a particular requirement, I would recommend %.ogg for sound effects and %.mp3 for longer pieces, like the orchestral score you  composed in Ableton last night. Here's how to use mk-sound when initializing stuff:

(mk-sound 'ponk "a/Interface Element 3.mp3")
(mk-sound 'spring "a/spring.ogg")
(mk-sound 'pop  "a/Pop.mp3")

I generally try to load it before my image resources. It probably doesn't matter that much, but...

And now, maestro, you need to play those sounds where they'll do the most good. Maybe in a key handler when the player presses the jump key, or perhaps in the collision detection System when they run into something. To do that, you'll need a way to play them:

   (define (play name)
     "( name --) Play named audio entity."
     (let ([a (grab (λ (e) (and (= e.tag 'audio)
                                (= e.name name))))])
       (when a
         (:= a.au.currentTime 0)
         (a.au.play))))

With this function, you can just say "(play 'spring)" and the heady sound of coiled metal will grace your ears!

Oh! One quick note -- sometimes browsers will be reluctant to play sounds without user interaction. This is a defense against all those awful auto-playing adverts that were popular a while back. Haven't ever seen that? Thank your browser for protecting you. However, it does mean that you may not hear any sound after all the trouble you took to retrofit your game. Fortunately, the solution is (fairly) simple. You can tell the browser to allow auto-playing media on the game page, or... You can just have the user do something on the page, which usually unblocks the audio. This can be as simple as hitting a key or clicking a button. When you publish your web game on itch.io, if you set it for "click to play", then that initial click will do that for you.

Make mine music!

Submitted (1 edit) (+1)

Hint #5 -- Taking advantage of E-S to avoid special cases.

It's very tempting at times to just draw something in the render as a special case. And that's not necessarily bad! Especially if you're trying to get something working. However, if you can take advantage of Canned Heat's entity store, your code will be a lot more flexible and maintainable. Not to mention probably shorter!

Here's an example... I wanted to put a "GAME OVER" message up, so I dropped it in at the end of the main draw function as a special case just so I could see it:

       (when (= 'dead hero.state)
         (:= cx 'fillStyle 'orange)
              (:= cx 'font "Bold 100px sans-serif")
              (:= cx 'textAlign 'center)
              (cx.fillText "GAME OVER"
                           (+ (/ SV.screen.width 2)  (- (* 5 (Math.random)) 3))
                           (+ (/ SV.screen.height 2) (- (* 5 (Math.random)) 3)))
              (:= cx 'textAlign 'start))
       )

The drawing is conditional on the hero being dead, which is simple enough. However that's one place where stuff like this can get you into trouble, when you have too many special cases to keep track of. Then, I set special font attributes and text alignments. While I clean up the alignment, I didn't bother resetting anything else. And, notice that I'm computing during the draw with the jitter factor being applied to the location of the cx.filllText. That should be really happening in a System.

So, to clean this up, I used mk-text instead. You can specify all the properties you need and let the draw loop handle it for you:

   (define (sys-lose?)
     "( --) Check to see if Kat has fallen and can't get up."
     (let ([hero (q1 'hero)])
       (when (and (not (= 'dead hero.state))
                  (< (+ SV.screen.height (/ SV.screen.height 4)) hero.y))
         (:= hero.state 'dead)
         (mk-text '(("GAME OVER"))
                  (/ SV.screen.width 2) (/ SV.screen.height 2) 'orange
                  '((font "Bold 100px sans-serif")
                    (jitter 3) (textAlign center)))
         )))

Now I set create the GAME OVER message in the System where I determine that the hero has died. The jitter property is set so that the sys-jitter System will animate it properly. But best of all, instead of dead pixels drawn irretrievably into a frame buffer, I have a live object for the GAME OVER text that I could do other things with if I wanted. The other advantage, of course, is that I have a general way to create text from the data in the Store instead of having to write code to do it.

Submitted(+1)

Hint #6 -- Cheap(ish) particle systems...

Everybody loves particles, right? They really liven things up. 

Canned Heat has the basics for particle handling built right in -- it uses (sys-tick), which compares the entities "age" (incremented every tick) with its "span" (if it has one). If age has exceeded the span, the entity is marked for destruction, which means, next redraw, it will be gone for good. 

A super simple case for particles is to make a trail, like the one Tinkerbell (or Nyancat, if you like) sports. To do this, create a particle object every time the entity is moved. Here, I have a simple System that will emit a particle whenever any entity that has a "trail" component is set to true:

   (define (sys-trail)
     "( --) Track trajectory of objects with trail component."
     (for ([e in-array (qq 'trail #t)])
       (let ([x (+ e.x (/ e.w 2))]
             [y (+ e.y (/ e.h 2))])
         (when (and (< 0 x) (< x SV.screen.width)
                    (< 0 y) (< y SV.screen.height))
           (store (object [tag 'trail] [scene #t] [x x] [y y] [age 0] [span 90])))))
     )
Don't be put off by that goofy math in the middle -- I'm just centreing the particle in the footprint of the trailing entity. The last bit checks to make sure the location is within the displayable area of the screen before creating it. I've had thousands of entities created with Canned Heat, but i don't want to let them get away from me... The parameters of the particle entity will snuff it out after ninety frames.

Oh, right... How would you display the particles? Try this in your render section:

     (for ([e in-array (qq 'scene #t)])
       (cond ...
             [(= e.tag 'trail)               (:= cx 'fillStyle 'red)               (cx.fillRect e.x e.y 4 4)]              ))
I just draw a red square for every trail particle. Easy as that.

You can supply deltas and use (sys-move) to animate the particles so they don't just hang around. With some trivial extensions, there are tons of possibilities -- add a colour ramp that interpolates as the entity ages and a rising dy with some jitter? You've got fire! Collisions can be pretty fast, since you just test for the point (x, y) instead of having to take the particle size into account. 

Canned Heat was partially inspired by the data structures used to manage particle systems, so it's no surprise that it handles them well!

Submitted

Just to let you know that I greatly enjoyed reading these all throughout the jam.

Submitted

Very happy you liked it! I'm going to drop one more entry here tonight, but later I'll try to summarize all these hints into the repo documentation.

Submitted (2 edits)

Hint #7 -- Changing the World

Remember I talked briefly about "live" objects (or entities...) in the Store instead of "dead" ones that get drawn implicitly? One of the benefits of the former is that you can easily change them dynamically. This has important benefits for development beyond just unifying way you store things.

When the objects are live, you can directly access them in the Store, modify their properties and see instant updates. To get started, pop open the web console in your browser while your game is running. The Entity Store (JavaScript variable STORE) is a global variable you can access:


You can use all the web debugging tools at your command (and there are many powerful ones!) to access and interrogate what's happening in your game. You can also use the Canned Heat query functions:

Remember that Urlang compiles directly to JavaScript, right? And because you're able to access the entities, you can change their properties, right in your console window. That's how I fine tuned a lot of my text placement for the title and credit screens in Kipple Kat. I would put in rough coordinates for things, then I would adjust them on the fly to get them where they really needed to be:

As I change it in the console, the text moves up and down (or down and up, in this case). Once I have it where I want it, I note the latest number and update the source. Not completely smooth, but way better than trying to update a number, recompile, renavigate to wherever it is in the game, deciding it's wrong, then updating a number... The point is, live objects are your friend!

Because the E-S encourages a generic approach, I can do the same thing with almost any displayed object -- they all work the same way because they're displayed or operated on by the same mechanisms. 

Have you ever wanted a level editor, or some way to dynamically modify your game world? With Canned Heat, you get it almost for free. To add the layout editor (because Kipple Kat doesn't really have traditional levels...) was about 150 or so lines of code. Most of that was new key handlers to drive a cursor -- you guessed it! -- Entity around and select what it rolled over. I didn't have much time to be fancy, but I was still able to: move a cursor, add new entities at the cursor, move them around with the cursor, delete selected objects, delete all objects (clean slate!) and dump the current state of the layout to the web console. Rather rough, but effective -- In about twenty minutes, I was able to do most of the layouts seen in the game. And that time includes fixing a bug a found along the way!

When I first made Canned Heat available, I provided four simple example games that I used to validate the process. My jam entry (demonstrating the editor among other more advanced techniques) is now one of those examples. And, while I'm very happy with it and how it worked out, it's only scratching the surface.

So, if you're writing games in Lisp for the web, I hope you'll have a chance to check out Canned Heat. Thanks for reading!

Submitted

Quick archival note... These articles have been collected and checked in to the Canned Heat documentation at https://hg.sr.ht/~oofoe/candheat/browse/dox?rev=tip.