use notan::math::{vec3, Mat4, Vec3};
use notan::prelude::*;
const LIGHT_VERTEX_SHADER: ShaderSource = notan::vertex_shader! {
r#"
#version 450
layout (location = 0) in vec3 a_pos;
layout (location = 1) in vec3 a_normal;
layout (location = 0) out vec3 v_pos;
layout (location = 1) out vec3 v_normal;
layout(set = 0, binding = 0) uniform Transform {
mat4 model;
mat4 view;
mat4 projection;
};
void main()
{
v_pos = vec3(model * vec4(a_pos, 1.0));
v_normal = a_normal;
gl_Position = projection * view * vec4(v_pos, 1.0);
}
"#
};
const LIGHT_FRAGMENT_SHADER: ShaderSource = notan::fragment_shader! {
r#"
#version 450
layout(location = 0) in vec3 v_normal;
layout(location = 1) in vec3 v_pos;
layout(location = 0) out vec4 color;
layout(set = 0, binding = 1) uniform Light {
vec3 lightPos;
vec3 lightColor;
vec3 objectColor;
};
void main()
{
// ambient
float ambientStrength = 0.1;
vec3 ambient = ambientStrength * lightColor;
// diffuse
vec3 norm = normalize(v_normal);
vec3 lightDir = normalize(lightPos - v_pos);
float diff = max(dot(norm, lightDir), 0.0);
vec3 diffuse = diff * lightColor;
vec3 result = (ambient + diffuse) * objectColor;
color = vec4(result, 1.0);
}
"#
};
#[uniform]
#[derive(Copy, Clone)]
struct Transform {
model: Mat4,
view: Mat4,
projection: Mat4,
}
impl Default for Transform {
fn default() -> Self {
Self {
model: Mat4::IDENTITY,
view: Mat4::IDENTITY,
projection: Mat4::perspective_rh_gl(20.0_f32.to_radians(), 800.0 / 600.0, 0.1, 100.0),
}
}
}
impl Transform {
fn with_view(time: f32) -> Transform {
let radius = 10.0;
let cam_x = time.sin() * radius;
let cam_z = time.cos() * radius;
Transform {
view: Mat4::look_at_rh(
vec3(cam_x, cam_z * 0.5, cam_z),
vec3(0.0, 0.0, 0.0),
vec3(0.0, 1.0, 0.0),
),
..Default::default()
}
}
}
#[uniform]
#[derive(Copy, Clone)]
struct Light {
light_pos: Vec3,
light_color: Vec3,
object_color: Vec3,
}
impl Default for Light {
fn default() -> Self {
Self {
light_pos: vec3(1.2, 1.0, 2.0),
light_color: vec3(1.0, 1.0, 1.0),
object_color: vec3(1.0, 0.5, 0.31),
}
}
}
impl Light {
fn with_time(time: f32) -> Self {
Self {
light_pos: vec3(1.2 + time.sin() * 2.0, 1.0 + time.cos() * 2.0, 2.0),
..Default::default()
}
}
}
#[derive(AppState)]
struct State {
pipeline: Pipeline,
vbo: Buffer,
transform_ubo: Buffer,
light_ubo: Buffer,
}
#[notan_main]
fn main() -> Result<(), String> {
notan::init_with(setup).draw(draw).build()
}
fn setup(gfx: &mut Graphics) -> State {
let vertex_info = VertexInfo::new()
.attr(0, VertexFormat::Float32x3) .attr(1, VertexFormat::Float32x3);
let depth_test = DepthStencil {
write: true,
compare: CompareMode::Less,
};
let pipeline = gfx
.create_pipeline()
.from(&LIGHT_VERTEX_SHADER, &LIGHT_FRAGMENT_SHADER)
.with_vertex_info(&vertex_info)
.with_depth_stencil(depth_test)
.build()
.unwrap();
#[rustfmt::skip]
let vertices = [
-0.5, -0.5, -0.5, 0.0, 0.0, -1.0,
0.5, -0.5, -0.5, 0.0, 0.0, -1.0,
0.5, 0.5, -0.5, 0.0, 0.0, -1.0,
0.5, 0.5, -0.5, 0.0, 0.0, -1.0,
-0.5, 0.5, -0.5, 0.0, 0.0, -1.0,
-0.5, -0.5, -0.5, 0.0, 0.0, -1.0,
-0.5, -0.5, 0.5, 0.0, 0.0, 1.0,
0.5, -0.5, 0.5, 0.0, 0.0, 1.0,
0.5, 0.5, 0.5, 0.0, 0.0, 1.0,
0.5, 0.5, 0.5, 0.0, 0.0, 1.0,
-0.5, 0.5, 0.5, 0.0, 0.0, 1.0,
-0.5, -0.5, 0.5, 0.0, 0.0, 1.0,
-0.5, 0.5, 0.5, -1.0, 0.0, 0.0,
-0.5, 0.5, -0.5, -1.0, 0.0, 0.0,
-0.5, -0.5, -0.5, -1.0, 0.0, 0.0,
-0.5, -0.5, -0.5, -1.0, 0.0, 0.0,
-0.5, -0.5, 0.5, -1.0, 0.0, 0.0,
-0.5, 0.5, 0.5, -1.0, 0.0, 0.0,
0.5, 0.5, 0.5, 1.0, 0.0, 0.0,
0.5, 0.5, -0.5, 1.0, 0.0, 0.0,
0.5, -0.5, -0.5, 1.0, 0.0, 0.0,
0.5, -0.5, -0.5, 1.0, 0.0, 0.0,
0.5, -0.5, 0.5, 1.0, 0.0, 0.0,
0.5, 0.5, 0.5, 1.0, 0.0, 0.0,
-0.5, -0.5, -0.5, 0.0, -1.0, 0.0,
0.5, -0.5, -0.5, 0.0, -1.0, 0.0,
0.5, -0.5, 0.5, 0.0, -1.0, 0.0,
0.5, -0.5, 0.5, 0.0, -1.0, 0.0,
-0.5, -0.5, 0.5, 0.0, -1.0, 0.0,
-0.5, -0.5, -0.5, 0.0, -1.0, 0.0,
-0.5, 0.5, -0.5, 0.0, 1.0, 0.0,
0.5, 0.5, -0.5, 0.0, 1.0, 0.0,
0.5, 0.5, 0.5, 0.0, 1.0, 0.0,
0.5, 0.5, 0.5, 0.0, 1.0, 0.0,
-0.5, 0.5, 0.5, 0.0, 1.0, 0.0,
-0.5, 0.5, -0.5, 0.0, 1.0, 0.0,
];
let vbo = gfx
.create_vertex_buffer()
.with_data(&vertices)
.with_info(&vertex_info)
.build()
.unwrap();
let transform = Transform::with_view(0.0);
let transform_ubo = gfx
.create_uniform_buffer(0, "Transform")
.with_data(&transform) .build()
.unwrap();
let light = Light::with_time(0.0);
let light_ubo = gfx
.create_uniform_buffer(1, "Light")
.with_data(&light) .build()
.unwrap();
State {
pipeline,
vbo,
transform_ubo,
light_ubo,
}
}
fn draw(app: &mut App, gfx: &mut Graphics, state: &mut State) {
let time = app.timer.elapsed_f32();
gfx.set_buffer_data(&state.transform_ubo, &Transform::with_view(time));
gfx.set_buffer_data(&state.light_ubo, &Light::with_time(time));
let mut renderer = gfx.create_renderer();
let clear = ClearOptions {
color: Some(Color::from_rgb(0.1, 0.2, 0.3)),
depth: Some(1.0),
stencil: None,
};
renderer.begin(Some(clear));
renderer.set_pipeline(&state.pipeline);
renderer.bind_buffers(&[&state.vbo, &state.transform_ubo, &state.light_ubo]);
renderer.draw(0, 36);
renderer.end();
gfx.render(&renderer);
}