use crate::{Material, MaterialPipeline, MaterialPipelineKey, MaterialPlugin}; use bevy_app::{Plugin, Startup, Update}; use bevy_asset::{load_internal_asset, Asset, Assets, Handle}; use bevy_ecs::prelude::*; use bevy_reflect::{std_traits::ReflectDefault, Reflect, TypePath}; use bevy_render::{ color::Color, extract_resource::ExtractResource, mesh::{Mesh, MeshVertexBufferLayout}, prelude::*, render_resource::*, }; pub const WIREFRAME_SHADER_HANDLE: Handle = 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::() .register_type::() .register_type::() .register_type::() .init_resource::() .add_plugins(MaterialPlugin::::default()) .add_systems(Startup, setup_global_wireframe_material) .add_systems( Update, ( global_color_changed.run_if(resource_changed::), wireframe_color_changed, apply_wireframe_material, apply_global_wireframe_material.run_if(resource_changed::), ), ); } } /// 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, } fn setup_global_wireframe_material( mut commands: Commands, mut materials: ResMut>, config: Res, ) { // Create the handle used for the global material commands.insert_resource(GlobalWireframeMaterial { handle: materials.add(WireframeMaterial { color: config.default_color, }), }); } /// Updates the wireframe material of all entities without a [`WireframeColor`] or without a [`Wireframe`] component fn global_color_changed( config: Res, mut materials: ResMut>, global_material: Res, ) { if let Some(global_material) = materials.get_mut(&global_material.handle) { global_material.color = config.default_color; } } /// Updates the wireframe material when the color in [`WireframeColor`] changes #[allow(clippy::type_complexity)] fn wireframe_color_changed( mut materials: ResMut>, mut colors_changed: Query< (&mut Handle, &WireframeColor), (With, Changed), >, ) { for (mut handle, wireframe_color) in &mut colors_changed { *handle = materials.add(WireframeMaterial { color: wireframe_color.color, }); } } /// Applies or remove the wireframe material to any mesh with a [`Wireframe`] component. fn apply_wireframe_material( mut commands: Commands, mut materials: ResMut>, wireframes: Query< (Entity, Option<&WireframeColor>), (With, Without>), >, mut removed_wireframes: RemovedComponents, global_material: Res, ) { for e in removed_wireframes.read() { if let Some(mut commands) = commands.get_entity(e) { commands.remove::>(); } } 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, }) } 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>, Without, Without); /// Applies or removes a wireframe material on any mesh without a [`Wireframe`] component. fn apply_global_wireframe_material( mut commands: Commands, config: Res, meshes_without_material: Query>)>, meshes_with_global_material: Query>)>, global_material: Res, ) { 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::>(); } } } #[derive(Default, AsBindGroup, TypePath, Debug, Clone, Asset)] pub struct WireframeMaterial { #[uniform(0)] pub color: Color, } impl Material for WireframeMaterial { fn fragment_shader() -> ShaderRef { WIREFRAME_SHADER_HANDLE.into() } fn specialize( _pipeline: &MaterialPipeline, descriptor: &mut RenderPipelineDescriptor, _layout: &MeshVertexBufferLayout, _key: MaterialPipelineKey, ) -> Result<(), SpecializedMeshPipelineError> { descriptor.primitive.polygon_mode = PolygonMode::Line; descriptor.depth_stencil.as_mut().unwrap().bias.slope_scale = 1.0; Ok(()) } }