177 lines
4.4 KiB
Odin
177 lines
4.4 KiB
Odin
package main
|
|
|
|
import rl "vendor:raylib"
|
|
import "vendor:raylib/rlgl"
|
|
import "core:fmt"
|
|
import "core:math"
|
|
import "core:math/ease"
|
|
import "core:math/rand"
|
|
import "core:strings"
|
|
import "core:strconv"
|
|
|
|
// Virtual game field dimensions
|
|
GameField := vec2{400, 200}
|
|
|
|
Game :: struct {
|
|
using state: GameState,
|
|
health: u8,
|
|
player: Player,
|
|
camera: rl.Camera3D,
|
|
camera_offset: vec3,
|
|
score: u32,
|
|
background: rl.Texture,
|
|
plane: rl.Model,
|
|
bullets: [dynamic]Bullet,
|
|
snake_max_health: int,
|
|
snake_health: int,
|
|
music: rl.Music,
|
|
}
|
|
|
|
|
|
|
|
game_init :: proc(prev: ^GameState = nil) -> ^GameState {
|
|
state := new(Game)
|
|
state.previous = prev
|
|
state.variant = state
|
|
state.draw = game_draw
|
|
state.update = game_update
|
|
state.free = game_free
|
|
rlgl.DisableBackfaceCulling()
|
|
|
|
img := rl.GenImageChecked(1024, 1024, 128, 128, rl.Color{60, 255, 255, 255}, rl.Color{30, 220, 220, 255})
|
|
state.background = rl.LoadTextureFromImage(img)
|
|
|
|
state.plane = rl.LoadModel(".\\assets\\models\\background.obj")
|
|
rl.UnloadImage(img)
|
|
|
|
game_setup(state)
|
|
return state
|
|
}
|
|
|
|
game_setup :: proc(game: ^Game) {
|
|
clear(&game.bullets)
|
|
clear(&Segments)
|
|
clear(&trails)
|
|
clear(&explosions)
|
|
explosions_init()
|
|
trail_init()
|
|
game.player = player_spawn({GameField.x / 2 + 70, 20, 0})
|
|
// rl.StopMusicStream(game.music)
|
|
|
|
game.health = 100
|
|
game.snake_max_health = 0
|
|
|
|
snake_spawn({10, 10, 0}, math.PI, 70)
|
|
|
|
|
|
for segment in Segments {
|
|
game.snake_max_health += int(segment.health)
|
|
}
|
|
|
|
game.camera = rl.Camera3D{
|
|
target = game.player.pos,
|
|
position = game.player.pos + vec3backward * 50,
|
|
fovy = 60,
|
|
//offset = WSize/2,
|
|
projection = rl.CameraProjection.PERSPECTIVE,
|
|
up = vec3up
|
|
}
|
|
game.camera.target.x = clamp(game.camera.target.x, -GameField.x/2, GameField.x/2)
|
|
game.camera.target.y = clamp(game.camera.target.y, 0, GameField.y)
|
|
game.camera.position = game.camera.target + vec3backward * 50
|
|
}
|
|
|
|
|
|
game_gen_level :: proc(game: ^Game) {
|
|
|
|
}
|
|
|
|
game_update :: proc(state: ^GameState, delta: f32) {
|
|
game := transmute(^Game)state
|
|
using game
|
|
// if rl.IsMusicStreamPlaying(game.music) {
|
|
rl.UpdateMusicStream(game.music)
|
|
// }
|
|
explosions_process(delta)
|
|
|
|
if rl.IsKeyPressed(rl.KeyboardKey.ESCAPE) {
|
|
pause := pause_init(game)
|
|
stack_push(pause)
|
|
}
|
|
|
|
|
|
player_update(&player, game, delta)
|
|
#reverse for &bullet, i in bullets {
|
|
bullet_process(&bullet, game, delta)
|
|
if !bullet.alive {
|
|
unordered_remove(&bullets, i)
|
|
}
|
|
}
|
|
snake_process(game, delta)
|
|
target_offset := player.vel / 5
|
|
camera_offset = rl.Vector3MoveTowards(camera_offset, target_offset, rl.Vector3Length(target_offset - camera_offset) * 10 * delta)
|
|
camera.target = player.pos + camera_offset
|
|
camera.target.x = clamp(camera.target.x, -GameField.x/2, GameField.x/2)
|
|
camera.target.y = clamp(camera.target.y, 0, GameField.y)
|
|
camera.position = camera.target + vec3backward * 50
|
|
trail_update(delta)
|
|
}
|
|
|
|
game_draw :: proc(state: ^GameState) {
|
|
rlgl.SetLineWidth(4)
|
|
game := transmute(^Game)state
|
|
using game
|
|
|
|
rl.BeginMode3D(camera)
|
|
rl.DrawModel(game.plane, {0, 0, 500}, 1000, rl.WHITE)
|
|
|
|
yy : i32 = 0
|
|
|
|
snake_draw(game)
|
|
player_draw(&player)
|
|
|
|
for &bullet, i in bullets {
|
|
bullet_draw(&bullet, game)
|
|
if i != 0 {
|
|
// rl.DrawLine3D(bullets[i].pos, bullets[i-1].pos, rl.WHITE)
|
|
}
|
|
}
|
|
trail_draw()
|
|
explosions_draw()
|
|
rl.EndMode3D()
|
|
|
|
|
|
if stack_top() == game {
|
|
hb_text : cstring = "Jörmungandr"
|
|
height := 30 * (WSize.y / 480)
|
|
hb_health : f32 = f32(snake_health) / f32(snake_max_health)
|
|
if snake_health == 0 {
|
|
hb_health = f32(Head.health) / f32(Head.max_health)
|
|
hb_text = "Jörmungandr's head"
|
|
}
|
|
// rl.DrawRectangleV({0, WSize.y - height - 10}, {WSize.x, height + 10}, rl.WHITE)
|
|
// hb_width := hb_health * WSize.x
|
|
// rl.DrawRectangleV({WSize.x / 2 - hb_width / 2, WSize.y - height - 7}, {hb_width, height + 4}, rl.RED)
|
|
// draw_text_centered(Res.Fonts.Title, hb_text, {WSize.x / 2, WSize.y - height / 2}, height)
|
|
draw_hbar(hb_text, hb_health, {0, WSize.y - height}, {WSize.x, height}, rl.RED)
|
|
draw_hbar("Þórr", f32(health) / 100.0, {0, 0}, {WSize.x, height}, rl.GREEN)
|
|
}
|
|
}
|
|
|
|
draw_hbar :: proc(text: cstring, value: f32, pos, size: vec2, color: rl.Color) {
|
|
rl.DrawRectangleV(pos, size, rl.WHITE)
|
|
hb_width := value * size.x
|
|
rl.DrawRectangleV(pos + {size.x / 2 - hb_width / 2, 2}, {hb_width, size.y - 4}, color)
|
|
draw_text_centered(Res.Fonts.Title, text, pos + size / 2, size.y / 1.3)
|
|
}
|
|
|
|
|
|
|
|
game_free :: proc(state: ^GameState) {
|
|
game := transmute(^Game)state
|
|
rl.UnloadTexture(game.background)
|
|
free(state)
|
|
}
|
|
|
|
|