Render Pass
You can define your own render passes that extends the renderer, currently there are render passes only for scenes, so no custom post-effects (this is planned to be improved in Fyrox 0.28). Render pass has full access to graphics framework (which is a thin wrapper around OpenGL) so it can utilize full power of it to implement various graphical effects.
Creating a render pass
Render pass is a complex thing, that requires relatively deep knowledge in computer graphics. It is intended to be used by experienced graphics programmers. Here's the simplest render pass that renders unit quad without any textures.
#![allow(unused)] fn main() { struct MyRenderPass { enabled: bool, shader: Box<dyn GpuProgram>, target_scene: Handle<Scene>, quad: Box<dyn GeometryBuffer>, uniform_location: usize, uniform_buffer: Box<dyn Buffer>, } impl MyRenderPass { pub fn new( renderer: &mut Renderer, target_scene: Handle<Scene>, ) -> Result<Self, FrameworkError> { let vs = r" layout(location = 0) in vec3 vertexPosition; layout(std140) uniform Uniforms { mat4 worldViewProjectionMatrix; } void main() { gl_Position = worldViewProjectionMatrix * vertexPosition; } "; let fs = r" out vec4 FragColor; void main() { FragColor = vec4(1.0, 0.0, 0.0, 1.0); } "; let shader = renderer.server.create_program("MyShader", vs, fs)?; let uniform_buffer = renderer .server .create_buffer(256, BufferKind::Uniform, BufferUsage::DynamicDraw)?; uniform_buffer.write_data_of_type( &StaticUniformBuffer::<256>::new() .with(&Matrix4::identity()) .finish(), )?; Ok(Self { enabled: true, uniform_location: shader.uniform_block_index(&"Uniforms".into())?, uniform_buffer, target_scene, quad: <dyn GeometryBuffer as GeometryBufferExt>::from_surface_data( &SurfaceData::make_quad(&Matrix4::identity()), BufferUsage::StaticDraw, renderer.server.as_ref(), )?, shader, }) } } impl SceneRenderPass for MyRenderPass { fn on_ldr_render( &mut self, ctx: SceneRenderPassContext, ) -> Result<RenderPassStatistics, FrameworkError> { let mut stats = RenderPassStatistics::default(); // Make sure to render only to target scene. if self.enabled && ctx.scene_handle == self.target_scene { let resources = ResourceBindGroup { bindings: &[ResourceBinding::Buffer { buffer: self.uniform_buffer.as_ref(), binding: BufferLocation::Auto { shader_location: self.uniform_location, }, data_usage: BufferDataUsage::default(), }], }; stats += ctx.framebuffer.draw( self.quad.as_ref(), ctx.viewport, self.shader.as_ref(), &DrawParameters::default(), &[resources], ElementRange::Full, )?; } Ok(stats) } fn source_type_id(&self) -> TypeId { ().type_id() } } }
The code snippet shows how to create a shader, find its uniforms, and finally how to actually render something in target frame buffer.
Registering a render pass
Every render pass must be registered in the renderer, otherwise it won't be used. You can register a render pass using
add_render_pass
method of the Renderer
:
#![allow(unused)] fn main() { fn usage_example(renderer: &mut Renderer, render_pass: MyRenderPass) { let shared_pass = Rc::new(RefCell::new(render_pass)); // You can share the pass across multiple places to be able to control it. renderer.add_render_pass(shared_pass); } }
Please notice that we've wrapped render pass in Rc<RefCell<..>>
, this means that you can share it across multiple places
and modify its data from the code of your game.