Add example game using raylib

This commit is contained in:
Bruce Hill 2024-09-08 19:49:47 -04:00
parent f86cc6549f
commit a7ae25ec08
7 changed files with 205 additions and 0 deletions

6
examples/game/README.md Normal file
View File

@ -0,0 +1,6 @@
# Example Game
This is a simple example game that uses [raylib](https://www.raylib.com/) to
demonstrate a project that spans multiple files and showcases some game
programming concepts used in idiomatic Tomo code. It also showcases how to
interact with an external C library.

9
examples/game/box.tm Normal file
View File

@ -0,0 +1,9 @@
# Defines a struct representing boxes on the terrain
use vectors
use ./world.tm
use ./color.tm
struct Box(pos:Vec2, size=Vec2(50, 50), color=Color.GRAY, blocking=yes):
func draw(b:&Box):
b.color:draw_rectangle(b.pos, b.size)

22
examples/game/color.tm Normal file
View File

@ -0,0 +1,22 @@
# Defines a struct used to represent colors using 64-bit floats (0.0 - 1.0),
# which can be used to draw colored rectangles in raylib
use <raylib.h>
use vectors
struct Color(r,g,b:Num32,a=1.0f32):
RED := Color(1,0,0)
GRAY := Color(.2f32,.2f32,.2f32)
LIGHT_GRAY := Color(.7f32,.7f32,.7f32)
func draw_rectangle(c:Color, pos:Vec2, size:Vec2):
inline C {
DrawRectangle(
(int)($pos.$x), (int)($pos.$y), (int)($size.$x), (int)($size.$y),
((Color){
(int8_t)(uint8_t)(255.*$c.$r),
(int8_t)(uint8_t)(255.*$c.$g),
(int8_t)(uint8_t)(255.*$c.$b),
(int8_t)(uint8_t)(255.*$c.$a),
})
);
}

42
examples/game/game.tm Normal file
View File

@ -0,0 +1,42 @@
# This game demo uses Raylib to present a simple
use libraylib.so
use <raylib.h>
use <raymath.h>
use file
use ./world.tm
func main():
extern InitWindow:func(w:Int32, h:Int32, title:CString)->Void
InitWindow(1600, 900, "raylib [core] example - 2d camera")
map := when read("map.txt") is Success(m): m
else: exit(code=1, "Could not find the game map!")
World.CURRENT:load_map(map)
extern SetTargetFPS:func(fps:Int32)
SetTargetFPS(60)
extern WindowShouldClose:func()->Bool
while not WindowShouldClose():
extern GetFrameTime:func()->Num32
dt := GetFrameTime()
World.CURRENT:update(Num(dt))
extern BeginDrawing:func()
BeginDrawing()
inline C {
ClearBackground((Color){0xCC, 0xCC, 0xCC, 0xFF});
}
World.CURRENT:draw()
extern EndDrawing:func()
EndDrawing()
extern CloseWindow:func()
CloseWindow()

17
examples/game/map.txt Normal file
View File

@ -0,0 +1,17 @@
[][][][][][][][][][][][][][][][][][][][][][][][][][][][][][][]
[] [] []
[] [] []
[] [] []
[] [] []
[] [] []
[] [] [] []
[] @@ [] []
[] [] []
[] [] []
[] [] []
[] [] []
[] [] []
[] [] []
[] [] []
[] [] []
[][][][][][][][][][][][][][][][][][][][][][][][][][][][][][][]

31
examples/game/player.tm Normal file
View File

@ -0,0 +1,31 @@
# Defines a struct representing the player, which is controlled by WASD keys
use libraylib.so
use <raylib.h>
use <raymath.h>
use vectors
use ./world.tm
struct Player(pos,prev_pos:Vec2):
WALK_SPEED := 500.
ACCEL := 0.3
FRICTION := 0.99
SIZE := Vec2(50, 50)
func update(p:&Player):
target_x := inline C (
(Num_t)((IsKeyDown(KEY_A) ? -1 : 0) + (IsKeyDown(KEY_D) ? 1 : 0))
) : Num
target_y := inline C (
(Num_t)((IsKeyDown(KEY_W) ? -1 : 0) + (IsKeyDown(KEY_S) ? 1 : 0))
) : Num
target_vel := Vec2(target_x, target_y):norm() * WALK_SPEED
vel := (p.pos - p.prev_pos)/World.DT
vel *= FRICTION
vel = vel:mix(target_vel, ACCEL)
p.prev_pos, p.pos = p.pos, p.pos + World.DT*vel
func draw(p:&Player):
Color.RED:draw_rectangle(p.pos, Player.SIZE)

78
examples/game/world.tm Normal file
View File

@ -0,0 +1,78 @@
use vectors
use ./player.tm
use ./color.tm
use ./box.tm
# Return a displacement relative to `a` that will push it out of `b`
func solve_overlap(a_pos:Vec2, a_size:Vec2, b_pos:Vec2, b_size:Vec2)->Vec2:
a_left := a_pos.x
a_right := a_pos.x + a_size.x
a_top := a_pos.y
a_bottom := a_pos.y + a_size.y
b_left := b_pos.x
b_right := b_pos.x + b_size.x
b_top := b_pos.y
b_bottom := b_pos.y + b_size.y
# Calculate the overlap in each dimension
overlap_x := (a_right _min_ b_right) - (a_left _max_ b_left)
overlap_y := (a_bottom _min_ b_bottom) - (a_top _max_ b_top)
# If either axis is not overlapping, then there is no collision:
if overlap_x <= 0 or overlap_y <= 0:
return Vec2(0, 0)
if overlap_x < overlap_y:
if a_right > b_left and a_right < b_right:
return Vec2(-(overlap_x), 0)
else if a_left < b_right and a_left > b_left:
return Vec2(overlap_x, 0)
else:
if a_top < b_bottom and a_top > b_top:
return Vec2(0, overlap_y)
else if a_bottom > b_top and a_bottom < b_bottom:
return Vec2(0, -overlap_y)
return Vec2(0, 0)
struct World(player:@Player, boxes:[@Box], dt_accum=0.0):
DT := 1./60.
CURRENT := @World(@Player(Vec2(0,0), Vec2(0,0)), [:@Box])
STIFFNESS := 0.3
func update(w:&World, dt:Num):
w.dt_accum += dt
while w.dt_accum > 0:
w:update_once()
w.dt_accum -= World.DT
func update_once(w:&World):
w.player:update()
# Resolve player overlapping with any boxes:
for i in 3:
for b in w.boxes:
w.player.pos += STIFFNESS * solve_overlap(w.player.pos, Player.SIZE, b.pos, b.size)
func draw(w:&World):
for b in w.boxes:
b:draw()
w.player:draw()
func load_map(w:&World, map:Text):
map = map:replace_all({$/[]/: "X", $/@{1..}/: "@", $/ /: " "})
w.boxes = [:@Box]
box_size := Vec2(50., 50.)
for y,line in map:lines():
for x,cell in line:split():
if cell == "X":
pos := Vec2((Num(x)-1) * box_size.x, (Num(y)-1) * box_size.y)
box := @Box(pos, size=box_size, color=Color.GRAY)
(&w.boxes):insert(box)
else if cell == "@":
pos := Vec2((Num(x)-1) * box_size.x, (Num(y)-1) * box_size.y)
w.player = @Player(pos,pos)