# Objective Cleanup naming and docs, add missing migration guide after #15591 All text root nodes now use `Text` (UI) / `Text2d`. All text readers/writers use `Text<Type>Reader`/`Text<Type>Writer` convention. --- ## Migration Guide Doubles as #15591 migration guide. Text bundles (`TextBundle` and `Text2dBundle`) were removed in favor of `Text` and `Text2d`. Shared configuration fields were replaced with `TextLayout`, `TextFont` and `TextColor` components. Just `TextBundle`'s additional field turned into `TextNodeFlags` component, while `Text2dBundle`'s additional fields turned into `TextBounds` and `Anchor` components. Text sections were removed in favor of hierarchy-based approach. For root text entities with `Text` or `Text2d` components, child entities with `TextSpan` will act as additional text sections. To still access text spans by index, use the new `TextUiReader`, `Text2dReader` and `TextUiWriter`, `Text2dWriter` system parameters.
		
			
				
	
	
		
			237 lines
		
	
	
		
			7.4 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
			
		
		
	
	
			237 lines
		
	
	
		
			7.4 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
//! A simple 3D scene showing how alpha blending can break and how order independent transparency (OIT) can fix it.
 | 
						|
//!
 | 
						|
//! See [`OrderIndependentTransparencyPlugin`] for the trade-offs of using OIT.
 | 
						|
//!
 | 
						|
//! [`OrderIndependentTransparencyPlugin`]: bevy::render::pipeline::OrderIndependentTransparencyPlugin
 | 
						|
use bevy::{
 | 
						|
    color::palettes::css::{BLUE, GREEN, RED},
 | 
						|
    core_pipeline::oit::OrderIndependentTransparencySettings,
 | 
						|
    prelude::*,
 | 
						|
    render::view::RenderLayers,
 | 
						|
};
 | 
						|
 | 
						|
fn main() {
 | 
						|
    std::env::set_var("RUST_BACKTRACE", "1");
 | 
						|
    App::new()
 | 
						|
        .add_plugins(DefaultPlugins)
 | 
						|
        .add_systems(Startup, setup)
 | 
						|
        .add_systems(Update, (toggle_oit, cycle_scenes))
 | 
						|
        .run();
 | 
						|
}
 | 
						|
 | 
						|
/// set up a simple 3D scene
 | 
						|
fn setup(
 | 
						|
    mut commands: Commands,
 | 
						|
    mut meshes: ResMut<Assets<Mesh>>,
 | 
						|
    mut materials: ResMut<Assets<StandardMaterial>>,
 | 
						|
) {
 | 
						|
    // camera
 | 
						|
    commands
 | 
						|
        .spawn((
 | 
						|
            Camera3d::default(),
 | 
						|
            Transform::from_xyz(0.0, 0.0, 10.0).looking_at(Vec3::ZERO, Vec3::Y),
 | 
						|
            // Add this component to this camera to render transparent meshes using OIT
 | 
						|
            OrderIndependentTransparencySettings::default(),
 | 
						|
            RenderLayers::layer(1),
 | 
						|
        ))
 | 
						|
        .insert(
 | 
						|
            // Msaa currently doesn't work with OIT
 | 
						|
            Msaa::Off,
 | 
						|
        );
 | 
						|
 | 
						|
    // light
 | 
						|
    commands.spawn((
 | 
						|
        PointLight {
 | 
						|
            shadows_enabled: false,
 | 
						|
            ..default()
 | 
						|
        },
 | 
						|
        Transform::from_xyz(4.0, 8.0, 4.0),
 | 
						|
        RenderLayers::layer(1),
 | 
						|
    ));
 | 
						|
 | 
						|
    // spawn help text
 | 
						|
    commands
 | 
						|
        .spawn((Text::default(), RenderLayers::layer(1)))
 | 
						|
        .with_children(|p| {
 | 
						|
            p.spawn(TextSpan::new("Press T to toggle OIT\n"));
 | 
						|
            p.spawn(TextSpan::new("OIT Enabled"));
 | 
						|
            p.spawn(TextSpan::new("\nPress C to cycle test scenes"));
 | 
						|
        });
 | 
						|
 | 
						|
    // spawn default scene
 | 
						|
    spawn_spheres(&mut commands, &mut meshes, &mut materials);
 | 
						|
}
 | 
						|
 | 
						|
fn toggle_oit(
 | 
						|
    mut commands: Commands,
 | 
						|
    text: Single<Entity, With<Text>>,
 | 
						|
    keyboard_input: Res<ButtonInput<KeyCode>>,
 | 
						|
    q: Single<(Entity, Has<OrderIndependentTransparencySettings>), With<Camera3d>>,
 | 
						|
    mut text_writer: TextUiWriter,
 | 
						|
) {
 | 
						|
    if keyboard_input.just_pressed(KeyCode::KeyT) {
 | 
						|
        let (e, has_oit) = *q;
 | 
						|
        *text_writer.text(*text, 2) = if has_oit {
 | 
						|
            // Removing the component will completely disable OIT for this camera
 | 
						|
            commands
 | 
						|
                .entity(e)
 | 
						|
                .remove::<OrderIndependentTransparencySettings>();
 | 
						|
            "OIT disabled".to_string()
 | 
						|
        } else {
 | 
						|
            // Adding the component to the camera will render any transparent meshes
 | 
						|
            // with OIT instead of alpha blending
 | 
						|
            commands
 | 
						|
                .entity(e)
 | 
						|
                .insert(OrderIndependentTransparencySettings::default());
 | 
						|
            "OIT enabled".to_string()
 | 
						|
        };
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
fn cycle_scenes(
 | 
						|
    mut commands: Commands,
 | 
						|
    keyboard_input: Res<ButtonInput<KeyCode>>,
 | 
						|
    mut meshes: ResMut<Assets<Mesh>>,
 | 
						|
    mut materials: ResMut<Assets<StandardMaterial>>,
 | 
						|
    q: Query<Entity, With<Mesh3d>>,
 | 
						|
    mut scene_id: Local<usize>,
 | 
						|
) {
 | 
						|
    if keyboard_input.just_pressed(KeyCode::KeyC) {
 | 
						|
        // depsawn current scene
 | 
						|
        for e in &q {
 | 
						|
            commands.entity(e).despawn_recursive();
 | 
						|
        }
 | 
						|
        // increment scene_id
 | 
						|
        *scene_id = (*scene_id + 1) % 2;
 | 
						|
        // spawn next scene
 | 
						|
        match *scene_id {
 | 
						|
            0 => spawn_spheres(&mut commands, &mut meshes, &mut materials),
 | 
						|
            1 => spawn_occlusion_test(&mut commands, &mut meshes, &mut materials),
 | 
						|
            _ => unreachable!(),
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
/// Spawns 3 overlapping spheres
 | 
						|
/// Technically, when using `alpha_to_coverage` with MSAA this particular example wouldn't break,
 | 
						|
/// but it breaks when disabling MSAA and is enough to show the difference between OIT enabled vs disabled.
 | 
						|
fn spawn_spheres(
 | 
						|
    commands: &mut Commands,
 | 
						|
    meshes: &mut Assets<Mesh>,
 | 
						|
    materials: &mut Assets<StandardMaterial>,
 | 
						|
) {
 | 
						|
    let pos_a = Vec3::new(-1.0, 0.75, 0.0);
 | 
						|
    let pos_b = Vec3::new(0.0, -0.75, 0.0);
 | 
						|
    let pos_c = Vec3::new(1.0, 0.75, 0.0);
 | 
						|
 | 
						|
    let offset = Vec3::new(0.0, 0.0, 0.0);
 | 
						|
 | 
						|
    let sphere_handle = meshes.add(Sphere::new(2.0).mesh());
 | 
						|
 | 
						|
    let alpha = 0.25;
 | 
						|
 | 
						|
    let render_layers = RenderLayers::layer(1);
 | 
						|
 | 
						|
    commands.spawn((
 | 
						|
        Mesh3d(sphere_handle.clone()),
 | 
						|
        MeshMaterial3d(materials.add(StandardMaterial {
 | 
						|
            base_color: RED.with_alpha(alpha).into(),
 | 
						|
            alpha_mode: AlphaMode::Blend,
 | 
						|
            ..default()
 | 
						|
        })),
 | 
						|
        Transform::from_translation(pos_a + offset),
 | 
						|
        render_layers.clone(),
 | 
						|
    ));
 | 
						|
    commands.spawn((
 | 
						|
        Mesh3d(sphere_handle.clone()),
 | 
						|
        MeshMaterial3d(materials.add(StandardMaterial {
 | 
						|
            base_color: GREEN.with_alpha(alpha).into(),
 | 
						|
            alpha_mode: AlphaMode::Blend,
 | 
						|
            ..default()
 | 
						|
        })),
 | 
						|
        Transform::from_translation(pos_b + offset),
 | 
						|
        render_layers.clone(),
 | 
						|
    ));
 | 
						|
    commands.spawn((
 | 
						|
        Mesh3d(sphere_handle.clone()),
 | 
						|
        MeshMaterial3d(materials.add(StandardMaterial {
 | 
						|
            base_color: BLUE.with_alpha(alpha).into(),
 | 
						|
            alpha_mode: AlphaMode::Blend,
 | 
						|
            ..default()
 | 
						|
        })),
 | 
						|
        Transform::from_translation(pos_c + offset),
 | 
						|
        render_layers.clone(),
 | 
						|
    ));
 | 
						|
}
 | 
						|
 | 
						|
/// Spawn a combination of opaque cubes and transparent spheres.
 | 
						|
/// This is useful to make sure transparent meshes drawn with OIT
 | 
						|
/// are properly occluded by opaque meshes.
 | 
						|
fn spawn_occlusion_test(
 | 
						|
    commands: &mut Commands,
 | 
						|
    meshes: &mut Assets<Mesh>,
 | 
						|
    materials: &mut Assets<StandardMaterial>,
 | 
						|
) {
 | 
						|
    let sphere_handle = meshes.add(Sphere::new(1.0).mesh());
 | 
						|
    let cube_handle = meshes.add(Cuboid::from_size(Vec3::ONE).mesh());
 | 
						|
    let cube_material = materials.add(Color::srgb(0.8, 0.7, 0.6));
 | 
						|
 | 
						|
    let render_layers = RenderLayers::layer(1);
 | 
						|
 | 
						|
    // front
 | 
						|
    let x = -2.5;
 | 
						|
    commands.spawn((
 | 
						|
        Mesh3d(cube_handle.clone()),
 | 
						|
        MeshMaterial3d(cube_material.clone()),
 | 
						|
        Transform::from_xyz(x, 0.0, 2.0),
 | 
						|
        render_layers.clone(),
 | 
						|
    ));
 | 
						|
    commands.spawn((
 | 
						|
        Mesh3d(sphere_handle.clone()),
 | 
						|
        MeshMaterial3d(materials.add(StandardMaterial {
 | 
						|
            base_color: RED.with_alpha(0.5).into(),
 | 
						|
            alpha_mode: AlphaMode::Blend,
 | 
						|
            ..default()
 | 
						|
        })),
 | 
						|
        Transform::from_xyz(x, 0., 0.),
 | 
						|
        render_layers.clone(),
 | 
						|
    ));
 | 
						|
 | 
						|
    // intersection
 | 
						|
    commands.spawn((
 | 
						|
        Mesh3d(cube_handle.clone()),
 | 
						|
        MeshMaterial3d(cube_material.clone()),
 | 
						|
        Transform::from_xyz(x, 0.0, 1.0),
 | 
						|
        render_layers.clone(),
 | 
						|
    ));
 | 
						|
    commands.spawn((
 | 
						|
        Mesh3d(sphere_handle.clone()),
 | 
						|
        MeshMaterial3d(materials.add(StandardMaterial {
 | 
						|
            base_color: RED.with_alpha(0.5).into(),
 | 
						|
            alpha_mode: AlphaMode::Blend,
 | 
						|
            ..default()
 | 
						|
        })),
 | 
						|
        Transform::from_xyz(0., 0., 0.),
 | 
						|
        render_layers.clone(),
 | 
						|
    ));
 | 
						|
 | 
						|
    // back
 | 
						|
    let x = 2.5;
 | 
						|
    commands.spawn((
 | 
						|
        Mesh3d(cube_handle.clone()),
 | 
						|
        MeshMaterial3d(cube_material.clone()),
 | 
						|
        Transform::from_xyz(x, 0.0, -2.0),
 | 
						|
        render_layers.clone(),
 | 
						|
    ));
 | 
						|
    commands.spawn((
 | 
						|
        Mesh3d(sphere_handle.clone()),
 | 
						|
        MeshMaterial3d(materials.add(StandardMaterial {
 | 
						|
            base_color: RED.with_alpha(0.5).into(),
 | 
						|
            alpha_mode: AlphaMode::Blend,
 | 
						|
            ..default()
 | 
						|
        })),
 | 
						|
        Transform::from_xyz(x, 0., 0.),
 | 
						|
        render_layers.clone(),
 | 
						|
    ));
 | 
						|
}
 |