viewport-lib 0.2.1

3D viewport rendering library
Documentation
// OIT composite shader — blends weighted-blended accum/reveal into the HDR buffer.
//
// Reads the accum (Rgba16Float) and reveal (R8Unorm) textures produced by the OIT
// geometry pass and reconstructs the final composite color using the McGuire & Bavoil
// formula, writing premultiplied alpha RGBA into the HDR color target.
//
// Group 0:
//   binding 0 — accum_tex:  texture_2d<f32>   (Rgba16Float accumulation)
//   binding 1 — reveal_tex: texture_2d<f32>   (R8Unorm reveal / transmittance)
//   binding 2 — samp:       sampler           (linear, clamp-to-edge)

@group(0) @binding(0) var accum_tex:  texture_2d<f32>;
@group(0) @binding(1) var reveal_tex: texture_2d<f32>;
@group(0) @binding(2) var samp:       sampler;

struct VertexOut {
    @builtin(position) clip_pos: vec4<f32>,
    @location(0) uv: vec2<f32>,
};

// Fullscreen triangle — no vertex buffer required.
@vertex
fn vs_main(@builtin(vertex_index) vertex_index: u32) -> VertexOut {
    let uv = vec2<f32>(
        f32((vertex_index << 1u) & 2u),
        f32(vertex_index & 2u),
    );
    var out: VertexOut;
    out.clip_pos = vec4<f32>(uv * 2.0 - 1.0, 0.0, 1.0);
    out.uv = vec2<f32>(uv.x, 1.0 - uv.y);
    return out;
}

@fragment
fn fs_main(in: VertexOut) -> @location(0) vec4<f32> {
    let accum = textureSample(accum_tex, samp, in.uv);
    let r     = textureSample(reveal_tex, samp, in.uv).r;

    // r is accumulated per-fragment alpha; if close to 0 no transparent pixels wrote here.
    // Discard to avoid darkening regions with no transparent geometry.
    if r < 0.001 {
        discard;
    }

    let avg_color = accum.rgb / max(accum.a, 1e-5);
    // Output premultiplied alpha so the HDR blend state (One / OneMinusSrcAlpha) composites correctly.
    return vec4<f32>(avg_color * (1.0 - r), 1.0 - r);
}