HeavylI Engine

HeavylI Engine is a game engine (with graphics, ECS, and scripting support) based on the HeavylI graphics library.

Usage:

This crate should be used with the heavyli (currently version 0.0.6) crate to get the best results from the engine. This engine includes ECS support (check heavyli_engine::ecs), native script support (check heavyli_engine::ecs::native_script), Lua scripting support (check heavyli_engine::lua_script), and basic sprite handling using Renderer2D and Sprite2D (check heavyli::render).

Code Example:

First, checkout the resources folder in the heavyli repository in order to load the images needed for this example.

In this example we'll create a little mario game (no physics here though).

To start, add this Lua script example at res/test.lua (see the resources folder in the repo): ```lua function start() math.randomseed(os.time())

-- Load new textures and save their IDs:
mario_texture = add_texture("res/mario-stand.png")
block_texture = add_texture("res/basic-block.png")

-- Add new sprites:
renderer:add_sprite(0, 0.0, 0.0, 0.5, 0.5, mario_texture)
renderer:add_sprite(2, 1.0, 1.0, 0.5, 0.5, block_texture)
renderer:add_sprite(3, 0.5, 1.0, 0.5, 0.5, block_texture)
renderer:add_sprite(4, 1.0, 0.5, 0.5, 0.5, block_texture)
renderer:add_sprite(5, 0.5, 0.5, 0.5, 0.5, block_texture)

end

counter = 6 posx = 0 posy = 0 speed = 1 mariotexture = 0 blocktexture = 0

function update() speed = delta_time

-- User input:
if key_pressed("up") then
    pos_y = pos_y + speed
elseif key_pressed("down") then
    pos_y = pos_y - speed
end

if key_pressed("left") then
    pos_x = pos_x + speed
elseif key_pressed("right") then
    pos_x = pos_x - speed
end

renderer:set_sprite_position(0, pos_x, pos_y)
renderer:set_camera_position(0, pos_x, pos_y)

-- Randbom block generation:
if key_pressed("a") then
    renderer:add_sprite(counter, counter % 12 * 0.5, math.random() % 30, 0.5, 0.5, block_texture)

    counter = counter + 1
    print(counter)
end

end

```

Also, you should have these two shader files: shader/basic_fragment.glsl: ```c

version 330 core

out vec4 FragColor;

in vec3 ourColor; in vec2 texCoord;

uniform sampler2D texture1;

void main() { vec4 col = texture(texture1, texCoord) * vec4(ourColor, 1.0f);

if (0 == col.r && 0 == col.g && 0 == col.b)
{
    discard;
}

FragColor = col;

}

```

shader/basic_vertex.glsl: ```c

version 330 core

layout (location = 0) in vec3 aPos; layout (location = 1) in vec3 aColor; layout (location = 2) in vec2 aTexCoord;

out vec3 ourColor; out vec2 texCoord;

uniform mat4 translation;

void main() { gl_Position = translation * vec4(aPos, 1.0); ourColor = aColor; texCoord = aTexCoord; }

```

With this script you'll have a little mario game running.

Now, for the rust code: ```rust // Dependencies: extern crate glfw; extern crate heavyli; extern crate heavyliengine; extern crate nalgebraglm as glm;

// Modules: use crate::heavyli::{openglmodules::initglfw, rendering::window::Window};

use crate::heavyliengine::{ ecs::scene::{Scene, SceneCore, SceneState, Update}, render::{ camera::Camera, utils::{configurewindow, windowendframe, windowstartframe}, }, };

// Screen Size: const SCRWIDTH: u32 = 800; const SCRHEIGHT: u32 = 600;

// Main Code: fn main() { // Initialize OpenGL with GLFW and create a new Window: let mut glfw = initglfw(); let mut window = Window::new(&glfw, "Sandbox", SCRWIDTH, SCR_HEIGHT);

// Configure the new window:
configure_window(&mut window);

// Create a new scene:
let mut scene = Scene::new(&mut glfw, &mut window, Scene1::new());

// Initial scene state:
scene.start();

// Run scene until it closes:
while SceneState::End != scene.get_state() {
    scene.update(120.0);
}

}

fn setwindowtitle(window: &mut Window, deltatime: f32) { let mut title = "Sandbox | FPS: ".tostring();

title.push_str(
    (1.0 / if 0.0 != delta_time && delta_time > 0.000001 {
        delta_time
    } else {
        f32::MIN_POSITIVE
    })
    .to_string()
    .as_str(),
);

window.set_title(&title);

}

// Scene Loop Implementation: pub struct Scene1 { delta_count: f32, }

impl Scene1 { fn new() -> Self { Self { delta_count: 0.0 } } }

impl Update for Scene1 { fn start(&mut self, core: &mut SceneCore) { // Add camera to the scene: core.registry.lock().unwrap().add_component( 0, Camera::new(glm::vec3(0.0, 0.0, -5.0), glm::vec2(0.0, 90.0)), );

    // Create a new script handler:
    let script_id = core.lua_script_manager.create_script_handler();

    // Load the test script:
    if let Err(err) = core
        .lua_script_manager
        .script_load(script_id, "res/test.lua")
    {
        println!("Error: {}", err);
    }
}

fn update(&mut self, core: &mut SceneCore) {
    window_start_frame(core.window);

    // Get camera view for world-location calculations:
    let cam_view = core
        .registry
        .lock()
        .unwrap()
        .get_component::<Camera>(0)
        .unwrap()
        .borrow_mut()
        .lock()
        .unwrap()
        .get_view();

    // Render all sprites:
    core.renderer
        .render(glm::vec2(SCR_WIDTH as f32, SCR_HEIGHT as f32), &cam_view);

    // Change the FPS count in title when 1 min passed:
    self.delta_count += core.delta_time;

    if self.delta_count >= 1.0 {
        set_window_title(core.window, core.delta_time);

        self.delta_count = 0.0;
    }

    // End scene when window is closed:
    if !core.window.is_open() {
        core.state = SceneState::End;
    }

    // IMPORTANT: remove all sprites' data at the end of the program:
    if SceneState::End == core.state {
        core.renderer.delete_all_sprites();
    }

    window_end_frame(core.window, core.glfw);
}

}

```

Features: