mirror of
https://github.com/bevyengine/bevy
synced 2025-01-24 19:05:17 +00:00
f9cc91d5a1
Although we cached hashes of `MeshVertexBufferLayout`, we were paying the cost of `PartialEq` on `InnerMeshVertexBufferLayout` for every entity, every frame. This patch changes that logic to place `MeshVertexBufferLayout`s in `Arc`s so that they can be compared and hashed by pointer. This results in a 28% speedup in the `queue_material_meshes` phase of `many_cubes`, with frustum culling disabled. Additionally, this patch contains two minor changes: 1. This commit flattens the specialized mesh pipeline cache to one level of hash tables instead of two. This saves a hash lookup. 2. The example `many_cubes` has been given a `--no-frustum-culling` flag, to aid in benchmarking. See the Tracy profile: <img width="1064" alt="Screenshot 2024-02-29 144406" src="https://github.com/bevyengine/bevy/assets/157897/18632f1d-1fdd-4ac7-90ed-2d10306b2a1e"> ## Migration guide * Duplicate `MeshVertexBufferLayout`s are now combined into a single object, `MeshVertexBufferLayoutRef`, which contains an atomically-reference-counted pointer to the layout. Code that was using `MeshVertexBufferLayout` may need to be updated to use `MeshVertexBufferLayoutRef` instead.
219 lines
8.2 KiB
Rust
219 lines
8.2 KiB
Rust
use crate::{Material, MaterialPipeline, MaterialPipelineKey, MaterialPlugin};
|
|
use bevy_app::{Plugin, Startup, Update};
|
|
use bevy_asset::{load_internal_asset, Asset, Assets, Handle};
|
|
use bevy_color::{Color, LinearRgba};
|
|
use bevy_ecs::prelude::*;
|
|
use bevy_reflect::{std_traits::ReflectDefault, Reflect, TypePath};
|
|
use bevy_render::{
|
|
extract_resource::ExtractResource, mesh::MeshVertexBufferLayoutRef, prelude::*,
|
|
render_resource::*,
|
|
};
|
|
|
|
pub const WIREFRAME_SHADER_HANDLE: Handle<Shader> = Handle::weak_from_u128(192598014480025766);
|
|
|
|
/// A [`Plugin`] that draws wireframes.
|
|
///
|
|
/// Wireframes currently do not work when using webgl or webgpu.
|
|
/// Supported rendering backends:
|
|
/// - DX12
|
|
/// - Vulkan
|
|
/// - Metal
|
|
///
|
|
/// This is a native only feature.
|
|
#[derive(Debug, Default)]
|
|
pub struct WireframePlugin;
|
|
impl Plugin for WireframePlugin {
|
|
fn build(&self, app: &mut bevy_app::App) {
|
|
load_internal_asset!(
|
|
app,
|
|
WIREFRAME_SHADER_HANDLE,
|
|
"render/wireframe.wgsl",
|
|
Shader::from_wgsl
|
|
);
|
|
|
|
app.register_type::<Wireframe>()
|
|
.register_type::<NoWireframe>()
|
|
.register_type::<WireframeConfig>()
|
|
.register_type::<WireframeColor>()
|
|
.init_resource::<WireframeConfig>()
|
|
.add_plugins(MaterialPlugin::<WireframeMaterial>::default())
|
|
.add_systems(Startup, setup_global_wireframe_material)
|
|
.add_systems(
|
|
Update,
|
|
(
|
|
global_color_changed.run_if(resource_changed::<WireframeConfig>),
|
|
wireframe_color_changed,
|
|
// Run `apply_global_wireframe_material` after `apply_wireframe_material` so that the global
|
|
// wireframe setting is applied to a mesh on the same frame its wireframe marker component is removed.
|
|
(apply_wireframe_material, apply_global_wireframe_material).chain(),
|
|
),
|
|
);
|
|
}
|
|
}
|
|
|
|
/// Enables wireframe rendering for any entity it is attached to.
|
|
/// It will ignore the [`WireframeConfig`] global setting.
|
|
///
|
|
/// This requires the [`WireframePlugin`] to be enabled.
|
|
#[derive(Component, Debug, Clone, Default, Reflect, Eq, PartialEq)]
|
|
#[reflect(Component, Default)]
|
|
pub struct Wireframe;
|
|
|
|
/// Sets the color of the [`Wireframe`] of the entity it is attached to.
|
|
/// If this component is present but there's no [`Wireframe`] component,
|
|
/// it will still affect the color of the wireframe when [`WireframeConfig::global`] is set to true.
|
|
///
|
|
/// This overrides the [`WireframeConfig::default_color`].
|
|
#[derive(Component, Debug, Clone, Default, Reflect)]
|
|
#[reflect(Component, Default)]
|
|
pub struct WireframeColor {
|
|
pub color: Color,
|
|
}
|
|
|
|
/// Disables wireframe rendering for any entity it is attached to.
|
|
/// It will ignore the [`WireframeConfig`] global setting.
|
|
///
|
|
/// This requires the [`WireframePlugin`] to be enabled.
|
|
#[derive(Component, Debug, Clone, Default, Reflect, Eq, PartialEq)]
|
|
#[reflect(Component, Default)]
|
|
pub struct NoWireframe;
|
|
|
|
#[derive(Resource, Debug, Clone, Default, ExtractResource, Reflect)]
|
|
#[reflect(Resource)]
|
|
pub struct WireframeConfig {
|
|
/// Whether to show wireframes for all meshes.
|
|
/// Can be overridden for individual meshes by adding a [`Wireframe`] or [`NoWireframe`] component.
|
|
pub global: bool,
|
|
/// If [`Self::global`] is set, any [`Entity`] that does not have a [`Wireframe`] component attached to it will have
|
|
/// wireframes using this color. Otherwise, this will be the fallback color for any entity that has a [`Wireframe`],
|
|
/// but no [`WireframeColor`].
|
|
pub default_color: Color,
|
|
}
|
|
|
|
#[derive(Resource)]
|
|
struct GlobalWireframeMaterial {
|
|
// This handle will be reused when the global config is enabled
|
|
handle: Handle<WireframeMaterial>,
|
|
}
|
|
|
|
fn setup_global_wireframe_material(
|
|
mut commands: Commands,
|
|
mut materials: ResMut<Assets<WireframeMaterial>>,
|
|
config: Res<WireframeConfig>,
|
|
) {
|
|
// Create the handle used for the global material
|
|
commands.insert_resource(GlobalWireframeMaterial {
|
|
handle: materials.add(WireframeMaterial {
|
|
color: config.default_color.into(),
|
|
}),
|
|
});
|
|
}
|
|
|
|
/// Updates the wireframe material of all entities without a [`WireframeColor`] or without a [`Wireframe`] component
|
|
fn global_color_changed(
|
|
config: Res<WireframeConfig>,
|
|
mut materials: ResMut<Assets<WireframeMaterial>>,
|
|
global_material: Res<GlobalWireframeMaterial>,
|
|
) {
|
|
if let Some(global_material) = materials.get_mut(&global_material.handle) {
|
|
global_material.color = config.default_color.into();
|
|
}
|
|
}
|
|
|
|
/// Updates the wireframe material when the color in [`WireframeColor`] changes
|
|
#[allow(clippy::type_complexity)]
|
|
fn wireframe_color_changed(
|
|
mut materials: ResMut<Assets<WireframeMaterial>>,
|
|
mut colors_changed: Query<
|
|
(&mut Handle<WireframeMaterial>, &WireframeColor),
|
|
(With<Wireframe>, Changed<WireframeColor>),
|
|
>,
|
|
) {
|
|
for (mut handle, wireframe_color) in &mut colors_changed {
|
|
*handle = materials.add(WireframeMaterial {
|
|
color: wireframe_color.color.into(),
|
|
});
|
|
}
|
|
}
|
|
|
|
/// Applies or remove the wireframe material to any mesh with a [`Wireframe`] component, and removes it
|
|
/// for any mesh with a [`NoWireframe`] component.
|
|
fn apply_wireframe_material(
|
|
mut commands: Commands,
|
|
mut materials: ResMut<Assets<WireframeMaterial>>,
|
|
wireframes: Query<
|
|
(Entity, Option<&WireframeColor>),
|
|
(With<Wireframe>, Without<Handle<WireframeMaterial>>),
|
|
>,
|
|
no_wireframes: Query<Entity, (With<NoWireframe>, With<Handle<WireframeMaterial>>)>,
|
|
mut removed_wireframes: RemovedComponents<Wireframe>,
|
|
global_material: Res<GlobalWireframeMaterial>,
|
|
) {
|
|
for e in removed_wireframes.read().chain(no_wireframes.iter()) {
|
|
if let Some(mut commands) = commands.get_entity(e) {
|
|
commands.remove::<Handle<WireframeMaterial>>();
|
|
}
|
|
}
|
|
|
|
let mut wireframes_to_spawn = vec![];
|
|
for (e, wireframe_color) in &wireframes {
|
|
let material = if let Some(wireframe_color) = wireframe_color {
|
|
materials.add(WireframeMaterial {
|
|
color: wireframe_color.color.into(),
|
|
})
|
|
} else {
|
|
// If there's no color specified we can use the global material since it's already set to use the default_color
|
|
global_material.handle.clone()
|
|
};
|
|
wireframes_to_spawn.push((e, material));
|
|
}
|
|
commands.insert_or_spawn_batch(wireframes_to_spawn);
|
|
}
|
|
|
|
type WireframeFilter = (With<Handle<Mesh>>, Without<Wireframe>, Without<NoWireframe>);
|
|
|
|
/// Applies or removes a wireframe material on any mesh without a [`Wireframe`] or [`NoWireframe`] component.
|
|
fn apply_global_wireframe_material(
|
|
mut commands: Commands,
|
|
config: Res<WireframeConfig>,
|
|
meshes_without_material: Query<Entity, (WireframeFilter, Without<Handle<WireframeMaterial>>)>,
|
|
meshes_with_global_material: Query<Entity, (WireframeFilter, With<Handle<WireframeMaterial>>)>,
|
|
global_material: Res<GlobalWireframeMaterial>,
|
|
) {
|
|
if config.global {
|
|
let mut material_to_spawn = vec![];
|
|
for e in &meshes_without_material {
|
|
// We only add the material handle but not the Wireframe component
|
|
// This makes it easy to detect which mesh is using the global material and which ones are user specified
|
|
material_to_spawn.push((e, global_material.handle.clone()));
|
|
}
|
|
commands.insert_or_spawn_batch(material_to_spawn);
|
|
} else {
|
|
for e in &meshes_with_global_material {
|
|
commands.entity(e).remove::<Handle<WireframeMaterial>>();
|
|
}
|
|
}
|
|
}
|
|
|
|
#[derive(Default, AsBindGroup, TypePath, Debug, Clone, Asset)]
|
|
pub struct WireframeMaterial {
|
|
#[uniform(0)]
|
|
pub color: LinearRgba,
|
|
}
|
|
|
|
impl Material for WireframeMaterial {
|
|
fn fragment_shader() -> ShaderRef {
|
|
WIREFRAME_SHADER_HANDLE.into()
|
|
}
|
|
|
|
fn specialize(
|
|
_pipeline: &MaterialPipeline<Self>,
|
|
descriptor: &mut RenderPipelineDescriptor,
|
|
_layout: &MeshVertexBufferLayoutRef,
|
|
_key: MaterialPipelineKey<Self>,
|
|
) -> Result<(), SpecializedMeshPipelineError> {
|
|
descriptor.primitive.polygon_mode = PolygonMode::Line;
|
|
descriptor.depth_stencil.as_mut().unwrap().bias.slope_scale = 1.0;
|
|
Ok(())
|
|
}
|
|
}
|