//! Demonstrates how to enable per-object motion blur. This rendering feature can be configured per //! camera using the [`MotionBlur`] component.z use bevy::{ core_pipeline::motion_blur::{MotionBlur, MotionBlurBundle}, prelude::*, }; fn main() { let mut app = App::new(); // MSAA and Motion Blur together are not compatible on WebGL #[cfg(all(feature = "webgl2", target_arch = "wasm32", not(feature = "webgpu")))] app.insert_resource(Msaa::Off); app.add_plugins(DefaultPlugins) .add_systems(Startup, (setup_camera, setup_scene, setup_ui)) .add_systems(Update, (keyboard_inputs, move_cars, move_camera).chain()) .run(); } fn setup_camera(mut commands: Commands) { commands.spawn(( Camera3dBundle::default(), // Add the MotionBlurBundle to a camera to enable motion blur. // Motion blur requires the depth and motion vector prepass, which this bundle adds. // Configure the amount and quality of motion blur per-camera using this component. MotionBlurBundle { motion_blur: MotionBlur { shutter_angle: 1.0, samples: 2, #[cfg(all(feature = "webgl2", target_arch = "wasm32", not(feature = "webgpu")))] _webgl2_padding: Default::default(), }, ..default() }, )); } // Everything past this point is used to build the example, but isn't required to use motion blur. #[derive(Resource)] enum CameraMode { Track, Chase, } #[derive(Component)] struct Moves(f32); #[derive(Component)] struct CameraTracked; #[derive(Component)] struct Rotates; fn setup_scene( asset_server: Res, mut images: ResMut>, mut commands: Commands, mut meshes: ResMut>, mut materials: ResMut>, ) { commands.insert_resource(AmbientLight { color: Color::WHITE, brightness: 300.0, }); commands.insert_resource(CameraMode::Chase); commands.spawn(DirectionalLightBundle { directional_light: DirectionalLight { illuminance: 3_000.0, shadows_enabled: true, ..default() }, transform: Transform::default().looking_to(Vec3::new(-1.0, -0.7, -1.0), Vec3::X), ..default() }); // Sky commands.spawn(PbrBundle { mesh: meshes.add(Sphere::default()), material: materials.add(StandardMaterial { unlit: true, base_color: Color::linear_rgb(0.1, 0.6, 1.0), ..default() }), transform: Transform::default().with_scale(Vec3::splat(-4000.0)), ..default() }); // Ground let mut plane: Mesh = Plane3d::default().into(); let uv_size = 4000.0; let uvs = vec![[uv_size, 0.0], [0.0, 0.0], [0.0, uv_size], [uv_size; 2]]; plane.insert_attribute(Mesh::ATTRIBUTE_UV_0, uvs); commands.spawn(PbrBundle { mesh: meshes.add(plane), material: materials.add(StandardMaterial { base_color: Color::WHITE, perceptual_roughness: 1.0, base_color_texture: Some(images.add(uv_debug_texture())), ..default() }), transform: Transform::from_xyz(0.0, -0.65, 0.0).with_scale(Vec3::splat(80.)), ..default() }); spawn_cars(&asset_server, &mut meshes, &mut materials, &mut commands); spawn_trees(&mut meshes, &mut materials, &mut commands); spawn_barriers(&mut meshes, &mut materials, &mut commands); } fn spawn_cars( asset_server: &AssetServer, meshes: &mut Assets, materials: &mut Assets, commands: &mut Commands, ) { const N_CARS: usize = 20; let box_mesh = meshes.add(Cuboid::new(0.3, 0.15, 0.55)); let cylinder = meshes.add(Cylinder::default()); let logo = asset_server.load("branding/icon.png"); let wheel_matl = materials.add(StandardMaterial { base_color: Color::WHITE, base_color_texture: Some(logo.clone()), ..default() }); let mut matl = |color| { materials.add(StandardMaterial { base_color: color, ..default() }) }; let colors = [ matl(Color::linear_rgb(1.0, 0.0, 0.0)), matl(Color::linear_rgb(1.0, 1.0, 0.0)), matl(Color::BLACK), matl(Color::linear_rgb(0.0, 0.0, 1.0)), matl(Color::linear_rgb(0.0, 1.0, 0.0)), matl(Color::linear_rgb(1.0, 0.0, 1.0)), matl(Color::linear_rgb(0.5, 0.5, 0.0)), matl(Color::linear_rgb(1.0, 0.5, 0.0)), ]; for i in 0..N_CARS { let color = colors[i % colors.len()].clone(); let mut entity = commands .spawn(( PbrBundle { mesh: box_mesh.clone(), material: color.clone(), transform: Transform::from_scale(Vec3::splat(0.5)), ..default() }, Moves(i as f32 * 2.0), )) .insert_if(CameraTracked, || i == 0); entity.with_children(|parent| { parent.spawn(PbrBundle { mesh: box_mesh.clone(), material: color, transform: Transform::from_xyz(0.0, 0.08, 0.03) .with_scale(Vec3::new(1.0, 1.0, 0.5)), ..default() }); let mut spawn_wheel = |x: f32, z: f32| { parent.spawn(( PbrBundle { mesh: cylinder.clone(), material: wheel_matl.clone(), transform: Transform::from_xyz(0.14 * x, -0.045, 0.15 * z) .with_scale(Vec3::new(0.15, 0.04, 0.15)) .with_rotation(Quat::from_rotation_z(std::f32::consts::FRAC_PI_2)), ..default() }, Rotates, )); }; spawn_wheel(1.0, 1.0); spawn_wheel(1.0, -1.0); spawn_wheel(-1.0, 1.0); spawn_wheel(-1.0, -1.0); }); } } fn spawn_barriers( meshes: &mut Assets, materials: &mut Assets, commands: &mut Commands, ) { const N_CONES: usize = 100; let capsule = meshes.add(Capsule3d::default()); let matl = materials.add(StandardMaterial { base_color: Color::srgb_u8(255, 87, 51), reflectance: 1.0, ..default() }); let mut spawn_with_offset = |offset: f32| { for i in 0..N_CONES { let pos = race_track_pos( offset, (i as f32) / (N_CONES as f32) * std::f32::consts::PI * 2.0, ); commands.spawn(PbrBundle { mesh: capsule.clone(), material: matl.clone(), transform: Transform::from_xyz(pos.x, -0.65, pos.y).with_scale(Vec3::splat(0.07)), ..default() }); } }; spawn_with_offset(0.04); spawn_with_offset(-0.04); } fn spawn_trees( meshes: &mut Assets, materials: &mut Assets, commands: &mut Commands, ) { const N_TREES: usize = 30; let capsule = meshes.add(Capsule3d::default()); let sphere = meshes.add(Sphere::default()); let leaves = materials.add(Color::linear_rgb(0.0, 1.0, 0.0)); let trunk = materials.add(Color::linear_rgb(0.4, 0.2, 0.2)); let mut spawn_with_offset = |offset: f32| { for i in 0..N_TREES { let pos = race_track_pos( offset, (i as f32) / (N_TREES as f32) * std::f32::consts::PI * 2.0, ); let [x, z] = pos.into(); commands.spawn(PbrBundle { mesh: sphere.clone(), material: leaves.clone(), transform: Transform::from_xyz(x, -0.3, z).with_scale(Vec3::splat(0.3)), ..default() }); commands.spawn(PbrBundle { mesh: capsule.clone(), material: trunk.clone(), transform: Transform::from_xyz(x, -0.5, z).with_scale(Vec3::new(0.05, 0.3, 0.05)), ..default() }); } }; spawn_with_offset(0.07); spawn_with_offset(-0.07); } fn setup_ui(mut commands: Commands) { let style = TextStyle::default(); commands.spawn( TextBundle::from_sections(vec![ TextSection::new(String::new(), style.clone()), TextSection::new(String::new(), style.clone()), TextSection::new("1/2: -/+ shutter angle (blur amount)\n", style.clone()), TextSection::new("3/4: -/+ sample count (blur quality)\n", style.clone()), TextSection::new("Spacebar: cycle camera\n", style.clone()), ]) .with_style(Style { position_type: PositionType::Absolute, top: Val::Px(12.0), left: Val::Px(12.0), ..default() }), ); } fn keyboard_inputs( mut motion_blur: Query<&mut MotionBlur>, presses: Res>, mut text: Query<&mut Text>, mut camera: ResMut, ) { let mut motion_blur = motion_blur.single_mut(); if presses.just_pressed(KeyCode::Digit1) { motion_blur.shutter_angle -= 0.25; } else if presses.just_pressed(KeyCode::Digit2) { motion_blur.shutter_angle += 0.25; } else if presses.just_pressed(KeyCode::Digit3) { motion_blur.samples = motion_blur.samples.saturating_sub(1); } else if presses.just_pressed(KeyCode::Digit4) { motion_blur.samples += 1; } else if presses.just_pressed(KeyCode::Space) { *camera = match *camera { CameraMode::Track => CameraMode::Chase, CameraMode::Chase => CameraMode::Track, }; } motion_blur.shutter_angle = motion_blur.shutter_angle.clamp(0.0, 1.0); motion_blur.samples = motion_blur.samples.clamp(0, 64); let mut text = text.single_mut(); text.sections[0].value = format!("Shutter angle: {:.2}\n", motion_blur.shutter_angle); text.sections[1].value = format!("Samples: {:.5}\n", motion_blur.samples); } /// Parametric function for a looping race track. `offset` will return the point offset /// perpendicular to the track at the given point. fn race_track_pos(offset: f32, t: f32) -> Vec2 { let x_tweak = 2.0; let y_tweak = 3.0; let scale = 8.0; let x0 = (x_tweak * t).sin(); let y0 = (y_tweak * t).cos(); let dx = x_tweak * (x_tweak * t).cos(); let dy = y_tweak * -(y_tweak * t).sin(); let x = x0 + offset * dy / (dx.powi(2) + dy.powi(2)).sqrt(); let y = y0 - offset * dx / (dx.powi(2) + dy.powi(2)).sqrt(); Vec2::new(x, y) * scale } fn move_cars( time: Res