rein 0.1.1

rein 3D Rendering Library
Documentation
// Instanced mesh shader with per-instance transforms and colors

struct CameraUniform {
    view_proj: mat4x4<f32>,
    eye: vec4<f32>,
};

@group(0) @binding(0)
var<uniform> camera: CameraUniform;

struct VertexInput {
    // Per-vertex attributes
    @location(0) position: vec3<f32>,
    @location(1) normal: vec3<f32>,
    @location(2) color: vec3<f32>,
    // Per-instance attributes
    @location(4) instance_col0: vec4<f32>,
    @location(5) instance_col1: vec4<f32>,
    @location(6) instance_col2: vec4<f32>,
    @location(7) instance_col3: vec4<f32>,
    @location(8) instance_color: vec4<f32>,
};

struct VertexOutput {
    @builtin(position) clip_position: vec4<f32>,
    @location(0) world_position: vec3<f32>,
    @location(1) world_normal: vec3<f32>,
    @location(2) color: vec4<f32>,
};

@vertex
fn vs_main(input: VertexInput) -> VertexOutput {
    var output: VertexOutput;

    // Reconstruct the instance transform matrix
    let instance_matrix = mat4x4<f32>(
        input.instance_col0,
        input.instance_col1,
        input.instance_col2,
        input.instance_col3
    );

    let world_pos = instance_matrix * vec4<f32>(input.position, 1.0);
    output.clip_position = camera.view_proj * world_pos;
    output.world_position = world_pos.xyz;

    // Calculate normal matrix (inverse transpose of upper-left 3x3)
    let normal_matrix = mat3x3<f32>(
        instance_matrix[0].xyz,
        instance_matrix[1].xyz,
        instance_matrix[2].xyz
    );
    output.world_normal = normalize(normal_matrix * input.normal);

    // Modulate vertex color with instance color
    output.color = vec4<f32>(input.color, 1.0) * input.instance_color;

    return output;
}

@fragment
fn fs_main(input: VertexOutput) -> @location(0) vec4<f32> {
    // Simple directional light
    let light_dir = normalize(vec3<f32>(0.3, 1.0, 0.5));
    let ambient = 0.3;

    let n = normalize(input.world_normal);
    let diffuse = max(dot(n, light_dir), 0.0);

    // Specular (Blinn-Phong)
    let view_dir = normalize(camera.eye.xyz - input.world_position);
    let half_dir = normalize(light_dir + view_dir);
    let specular = pow(max(dot(n, half_dir), 0.0), 32.0) * 0.3;

    let lighting = ambient + diffuse * 0.7 + specular;
    let final_color = input.color.rgb * lighting;

    return vec4<f32>(final_color, input.color.a);
}