Compare commits

...

23 Commits

Author SHA1 Message Date
da9cf82129 Merge pull request 'Added HDR Support (pipeline specialization)' (#4) from 3-add-hdr-support into master
All checks were successful
Build / Build (push) Successful in 2m14s
Reviewed-on: #4
2025-03-23 11:40:17 -04:00
85251459b5 Added HDR Support (pipeline specialization)
All checks were successful
Build / Build (push) Successful in 2m19s
2025-03-22 20:43:55 -04:00
6f019ff2d0 Merge pull request 'Add build action' (#2) from 1-add-build-action into master
All checks were successful
Build / Build (push) Successful in 2m47s
Reviewed-on: #2
2025-03-19 05:58:51 -04:00
914ad234a1
Added build action
All checks were successful
Build / Build (push) Successful in 41m14s
2025-03-18 20:07:54 -04:00
07d84f6018 color support 2025-02-26 00:46:53 -05:00
035259fba7 Added Depth-based Outlines 2025-02-25 19:48:54 -05:00
8c65dc75b3 Removed build workflow 2024-12-11 02:09:58 -05:00
337b852d9f Updated screenshot links 2024-12-11 02:07:20 -05:00
21c5461abd Upgraded to Bevy 0.15 2024-12-11 02:03:02 -05:00
1242f8352a
Fri Nov 22 02:42:27 PM EST 2024 2024-11-22 14:42:27 -05:00
05f6f76408
Thu Nov 21 04:22:13 PM EST 2024 2024-11-21 16:22:13 -05:00
48382cc39b
Thu Nov 21 12:23:45 PM EST 2024 (Introduced Defferred Rendering/Shading Threshold) 2024-11-21 12:23:45 -05:00
68c0010bd4
Remove unnecessary bevy features 2024-08-04 13:37:36 -04:00
46230d6664
Update to 0.14 + Add 0BSD option 2024-07-22 13:48:57 -04:00
294e8bb07a
Adaptive Outline 2024-06-13 20:40:46 -04:00
ac37b21edd
Merge branch 'master' of github.com:exvacuum/bevy_outline_post_process 2024-06-05 10:09:12 -04:00
ed2665afe2
Added OutlineRenderLabel documentation 2024-06-05 10:09:05 -04:00
fb9276ccf2
Update README.md 2024-06-05 08:35:47 -04:00
3d89c34567
Fixed docs redirect 2024-06-04 14:36:26 -04:00
8f15162b48
Renamed + Added Docs 2024-06-04 14:32:59 -04:00
19449be002
Merge branch 'master' of github.com:exvacuum/bevy_outline_post_process 2024-06-04 13:05:05 -04:00
d98878f313
Removed toolchain config 2024-06-04 13:01:27 -04:00
0e6f21a192
Update README.md 2024-06-02 18:19:51 -04:00
11 changed files with 1058 additions and 1684 deletions

View File

@ -1,34 +1,2 @@
# Add the contents of this file to `config.toml` to enable "fast build" configuration. Please read the notes below.
# NOTE: For maximum performance, build using a nightly compiler
# If you are using rust stable, remove the "-Zshare-generics=y" below.
[target.x86_64-unknown-linux-gnu]
linker = "clang"
rustflags = ["-Clink-arg=-fuse-ld=lld", "-Zshare-generics=y"]
# NOTE: you must install [Mach-O LLD Port](https://lld.llvm.org/MachO/index.html) on mac. you can easily do this by installing llvm which includes lld with the "brew" package manager:
# `brew install llvm`
[target.x86_64-apple-darwin]
rustflags = [
"-C",
"link-arg=-fuse-ld=/usr/local/opt/llvm/bin/ld64.lld",
"-Zshare-generics=y",
]
[target.aarch64-apple-darwin]
rustflags = [
"-C",
"link-arg=-fuse-ld=/opt/homebrew/opt/llvm/bin/ld64.lld",
"-Zshare-generics=y",
]
[target.x86_64-pc-windows-msvc]
linker = "rust-lld.exe"
rustflags = ["-Zshare-generics=n"]
# Optional: Uncommenting the following improves compile times, but reduces the amount of debug info to 'line number tables only'
# In most cases the gains are negligible, but if you are on macos and have slow compile times you should see significant gains.
#[profile.dev]
#debug = 1
[profile.dev.package."*"]
opt-level = 2

View File

@ -0,0 +1,25 @@
name: Build
on: [push]
jobs:
Build:
env:
RUNNER_TOOL_CACHE: /toolcache
container: rust:alpine
steps:
- name: Install node
run: apk add nodejs gcc libc-dev pkgconf libx11-dev alsa-lib-dev eudev-dev tar
- name: Check out repository code
uses: actions/checkout@v4
- name: Restore cache
uses: actions/cache@v4
with:
path: |
~/.cargo/bin/
~/.cargo/registry/index/
~/.cargo/registry/cache/
~/.cargo/git/db/
target/
key: ${{ runner.os }}-cargo-${{ hashFiles('**/Cargo.lock') }}
- name: Build
run: cargo build --release

2230
Cargo.lock generated

File diff suppressed because it is too large Load Diff

View File

@ -1,7 +1,12 @@
[package]
name = "grex_outline_post_process"
version = "0.1.2"
name = "bevy_outline_post_process"
version = "0.5.0"
edition = "2021"
description = "An adaptive outline post-processing effect for the Bevy game engine."
license = "0BSD OR MIT OR Apache-2.0"
repository = "https://git.soaos.dev/bevy_outline_post_process"
[dependencies.bevy]
version = "0.13"
version = "0.15"
default-features = false
features = ["bevy_render", "bevy_asset", "bevy_core_pipeline", "png"]

5
LICENSE-0BSD Normal file
View File

@ -0,0 +1,5 @@
Copyright (C) 2024 by Silas Bartha silas@exvacuum.dev
Permission to use, copy, modify, and/or distribute this software for any purpose with or without fee is hereby granted.
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.

View File

@ -1,32 +1,39 @@
# grex_outline_post_process
# bevy_outline_post_process
[![Crates](https://img.shields.io/crates/v/bevy_outline_post_process)](https://crates.io/crates/bevy_outline_post_process)
![License](https://img.shields.io/badge/license-0BSD%2FMIT%2FApache-blue.svg)
![Tag](https://img.shields.io/github/v/tag/exvacuum/bevy_outline_post_process)
[![Docs](https://img.shields.io/docsrs/bevy_outline_post_process)](https://exvacuum.github.io/bevy_outline_post_process)
A plugin for the [Bevy](https://bevyengine.org) engine which adds an outline post-processing effect.
Note: This is a full-screen post process effect and cannot be enabled/disabled for specific objects.
A plugin for the [Bevy](https://bevyengine.org) engine which adds an outline post-processing effect. Optionally supports adaptive outlining, so darker areas are outlined in white rather than black, based on luminance.
## Screenshots
![](./doc/screenshot.png)
![](./doc/screenshot_smooth.png)
![](https://git.exvacuum.dev/bevy_outline_post_process/plain/doc/screenshot.png)
![](https://git.exvacuum.dev/bevy_outline_post_process/plain/doc/screenshot_smooth.png)
Configuration Used:
```rs
grex_outline_post_process::components::OutlinePostProcessSettings {
weight: 2.0,
threshold: 0.0,
}
bevy_outline_post_process::components::OutlinePostProcessSettings::new(2.0, LinearRgba::BLACK, 0.01, 0.01, 1.0);
```
## Compatibility
| Crate Version | Bevy Version |
|--- |--- |
| 0.1 | 0.13 |
| 0.4-0.5 | 0.15 |
| 0.3 | 0.14 |
| 0.1-0.2 | 0.13 |
## Installation
### crates.io
```toml
[dependencies]
bevy_outline_post_process = "0.5"
```
### Using git URL in Cargo.toml
```toml
[dependencies.grex_outline_post_process]
git = "https://github.com/exvacuum/grex_outline_post_process.git"
[dependencies.bevy_outline_post_process]
git = "https://git.soaos.dev/bevy_outline_post_process"
```
## Usage
@ -34,15 +41,14 @@ git = "https://github.com/exvacuum/grex_outline_post_process.git"
In `main.rs`:
```rs
use bevy::prelude::*;
use grex_outline_post_process;
use bevy_outline_post_process;
fn main() {
App::new()
.add_plugins((
DefaultPlugins,
grex_outline_post_process::OutlinePostProcessPlugin,
bevy_outline_post_process::OutlinePostProcessPlugin,
))
.insert_resource(Msaa::Off)
.run();
}
```
@ -50,14 +56,14 @@ fn main() {
When spawning a camera:
```rs
commands.spawn((
// Camera3dBundle...
NormalPrepass,
grex_outline_post_process::components::OutlinePostProcessSettings {
weight: 2.0,
threshold: 0.0,
}
// Camera3d...
bevy_outline_post_process::components::OutlinePostProcessSettings::default();
));
```
This effect will only run for cameras which contain this component.
## License
This crate is licensed under your choice of 0BSD, Apache-2.0, or MIT license.

View File

@ -2,44 +2,94 @@
struct OutlinePostProcessSettings {
weight: f32,
threshold: f32,
#ifdef SIXTEEN_BYTE_ALIGNMENT
_padding: vec2<f32>,
#endif
color: vec4<f32>,
normal_threshold: f32,
depth_threshold: f32,
adaptive_threshold: f32,
camera_near: f32,
}
@group(0) @binding(0) var screen_texture: texture_2d<f32>;
@group(0) @binding(1) var screen_sampler: sampler;
@group(0) @binding(2) var normal_texture: texture_2d<f32>;
@group(0) @binding(3) var normal_sampler: sampler;
@group(0) @binding(4) var<uniform> settings: OutlinePostProcessSettings;
@group(0) @binding(4) var depth_texture: texture_depth_2d;
@group(0) @binding(5) var depth_sampler: sampler;
@group(0) @binding(6) var<uniform> settings: OutlinePostProcessSettings;
@fragment
fn fragment(
in: FullscreenVertexOutput
) -> @location(0) vec4<f32> {
let screen_color = textureSample(screen_texture, screen_sampler, in.uv);
let outline_width = settings.weight / vec2f(textureDimensions(screen_texture));
let luma = (0.2126 * screen_color.r + 0.7152 * screen_color.g + 0.0722 * screen_color.b);
let outline_width = settings.weight / vec2f(textureDimensions(screen_texture));
let uv_top = vec2f(in.uv.x, in.uv.y - outline_width.y);
let uv_bottom = vec2f(in.uv.x, in.uv.y + outline_width.y);
let uv_right = vec2f(in.uv.x + outline_width.x, in.uv.y);
let uv_left = vec2f(in.uv.x - outline_width.x, in.uv.y);
let uv_top_right = vec2f(in.uv.x + outline_width.x, in.uv.y - outline_width.y);
let normal = textureSample(normal_texture, normal_sampler, in.uv);
let normal_top = textureSample(normal_texture, normal_sampler, uv_top);
let normal_right = textureSample(normal_texture, normal_sampler, uv_right);
let normal_top_right = textureSample(normal_texture, normal_sampler, uv_top_right);
// NORMAL FACTOR {{{
let normal = textureSample(normal_texture, normal_sampler, in.uv).xyz;
let normal_top = textureSample(normal_texture, normal_sampler, uv_top).xyz;
let normal_right = textureSample(normal_texture, normal_sampler, uv_right).xyz;
let normal_top_right = textureSample(normal_texture, normal_sampler, uv_top_right).xyz;
let delta_top = abs(normal - normal_top);
let delta_right = abs(normal - normal_right);
let delta_top_right = abs(normal - normal_top_right);
let normal_delta_top = abs(normal - normal_top);
let normal_delta_right = abs(normal - normal_right);
let normal_delta_top_right = abs(normal - normal_top_right);
let delta_max = max(delta_top, max(delta_right, delta_top_right));
let delta_raw = max(delta_max.x, max(delta_max.y, delta_max.z));
let normal_top_sum = max(normal_delta_top.x, max(normal_delta_top.y, normal_delta_top.z));
let normal_right_sum = max(normal_delta_right.x, max(normal_delta_right.y, normal_delta_right.z));
let normal_top_right_sum = max(normal_delta_top_right.x, max(normal_delta_top_right.y, normal_delta_top_right.z));
let delta_clipped = clamp((delta_raw * 2.0) - settings.threshold, 0.0, 1.0);
var normal_difference = step(settings.normal_threshold, max(normal_top_sum, max(normal_right_sum, normal_top_right_sum)));
let normal_outline = normal_difference;
// }}}
let outline = vec4f(delta_clipped, delta_clipped, delta_clipped, 0.0);
// DEPTH FACTOR {{{
let depth_color = textureSample(depth_texture, depth_sampler, in.uv);
let depth_color_top = textureSample(depth_texture, depth_sampler, uv_top);
let depth_color_bottom = textureSample(depth_texture, depth_sampler, uv_bottom);
let depth_color_right = textureSample(depth_texture, depth_sampler, uv_right);
let depth_color_left = textureSample(depth_texture, depth_sampler, uv_left);
return screen_color - outline;
let depth = linearize_depth(depth_color);
let depth_top = linearize_depth(depth_color_top);
let depth_bottom = linearize_depth(depth_color_bottom);
let depth_right = linearize_depth(depth_color_right);
let depth_left = linearize_depth(depth_color_left);
var depth_delta = 0.0;
depth_delta = depth_delta + depth - depth_top;
depth_delta = depth_delta + depth - depth_bottom;
depth_delta = depth_delta + depth - depth_right;
depth_delta = depth_delta + depth - depth_left;
var depth_outline = step(settings.depth_threshold, depth_delta);
if depth_color == 0.0 {
depth_outline = 0.0;
}
// }}}
var outline = saturate(depth_outline + normal_outline);
if (1.0 - luma) > settings.adaptive_threshold {
outline = outline * -1;
}
if (outline == 1.0) {
return settings.color;
}
if (outline == -1.0) {
return vec4(vec3(1.0) - settings.color.xyz, 1.0);
}
return screen_color;
}
fn linearize_depth(depth: f32) -> f32 {
if depth == 0.0 {
return 0.0;
}
return settings.camera_near / depth;
}

View File

@ -1,9 +1,63 @@
use bevy::{prelude::*, render::{render_resource::ShaderType, extract_component::ExtractComponent}};
use bevy::{
core_pipeline::prepass::{DeferredPrepass, DepthPrepass, NormalPrepass},
prelude::*,
render::{extract_component::ExtractComponent, render_resource::{CachedRenderPipelineId, ShaderType}},
};
#[derive(Component, ShaderType, ExtractComponent, PartialEq, Clone, Default)]
/// Stores the cached pipeline ID for the post processing pass for this camera
#[derive(Component, Deref, DerefMut)]
pub struct OutlinePostProcessPipelineId(pub CachedRenderPipelineId);
/// Component which, when inserted into an entity with a camera and normal prepass, enables an outline effect for that
/// camera.
#[derive(Component, ShaderType, ExtractComponent, PartialEq, Clone)]
#[require(NormalPrepass, DepthPrepass, DeferredPrepass, Msaa(|| Msaa::Off))]
pub struct OutlinePostProcessSettings {
pub weight: f32,
pub threshold: f32,
#[cfg(feature = "webgl2")]
_padding: Vec2,
/// Weight of outlines in pixels.
weight: f32,
/// Color of outlines.
color: LinearRgba,
/// A threshold for normal differences, values below this threshold will not become outlines.
/// Higher values will result in more outlines which may look better on smooth surfaces.
normal_threshold: f32,
/// A threshold for depth differences (in units), values below this threshold will not become outlines.
/// Higher values will result in more outlines which may look better on smooth surfaces.
depth_threshold: f32,
/// Luma threshold to invert outline color. A value of `1.0` means this feature is disabled.
adaptive_threshold: f32,
/// Near plane depth of camera, used for linearization of depth buffer values
pub(crate) camera_near: f32,
}
impl OutlinePostProcessSettings {
/// Create a new instance with the given settings
pub fn new(
weight: f32,
color: LinearRgba,
normal_threshold: f32,
depth_threshold: f32,
adaptive_threshold: f32,
) -> Self {
Self {
weight,
color,
normal_threshold,
depth_threshold,
adaptive_threshold,
camera_near: 0.0,
}
}
}
impl Default for OutlinePostProcessSettings {
fn default() -> Self {
Self {
weight: 1.0,
color: LinearRgba::BLACK,
normal_threshold: 0.01,
depth_threshold: 0.05,
adaptive_threshold: 1.0,
camera_near: 0.0,
}
}
}

View File

@ -1,12 +1,33 @@
use bevy::{prelude::*, render::{RenderApp, extract_component::{UniformComponentPlugin, ExtractComponentPlugin}, render_graph::{RenderGraphApp, ViewNodeRunner}}, asset::embedded_asset, core_pipeline::core_3d::graph::{Core3d, Node3d}};
#![warn(missing_docs)]
//! A plugin for the Bevy game engine which provides an outline post-process effect. The effect
//! makes use of a normal and depth prepass to generate outlines where significant differences in the values
//! occur.
use bevy::{
asset::embedded_asset,
core_pipeline::core_3d::graph::{Core3d, Node3d},
prelude::*,
render::{
extract_component::{ExtractComponentPlugin, UniformComponentPlugin},
render_graph::{RenderGraphApp, ViewNodeRunner},
render_resource::{PipelineCache, SpecializedRenderPipelines, TextureFormat},
view::{ExtractedView, ViewTarget},
Render, RenderApp, RenderSet,
},
};
use components::{OutlinePostProcessPipelineId, OutlinePostProcessSettings};
pub use nodes::OutlineRenderLabel;
use resources::{OutlinePostProcessPipeline, OutlinePostProcessingPipelineKey};
pub struct OutlinePostProcessPlugin;
/// Components used by this plugin.
pub mod components;
mod resources;
mod nodes;
mod resources;
/// Plugin which provides an outline post-processing effect.
pub struct OutlinePostProcessPlugin;
impl Plugin for OutlinePostProcessPlugin {
fn build(&self, app: &mut App) {
@ -15,16 +36,24 @@ impl Plugin for OutlinePostProcessPlugin {
app.add_plugins((
UniformComponentPlugin::<components::OutlinePostProcessSettings>::default(),
ExtractComponentPlugin::<components::OutlinePostProcessSettings>::default(),
));
))
.add_systems(Update, update_shader_clip_planes);
let Ok(render_app) = app.get_sub_app_mut(RenderApp) else {
let Some(render_app) = app.get_sub_app_mut(RenderApp) else {
return;
};
render_app.add_render_graph_node::<ViewNodeRunner<nodes::OutlineRenderNode>>(
render_app
.add_systems(
Render,
prepare_post_processing_pipelines.in_set(RenderSet::Prepare),
)
.init_resource::<SpecializedRenderPipelines<OutlinePostProcessPipeline>>()
.add_render_graph_node::<ViewNodeRunner<nodes::OutlineRenderNode>>(
Core3d,
nodes::OutlineRenderLabel,
).add_render_graph_edges(
)
.add_render_graph_edges(
Core3d,
(
Node3d::Tonemapping,
@ -35,10 +64,48 @@ impl Plugin for OutlinePostProcessPlugin {
}
fn finish(&self, app: &mut App) {
let Ok(render_app) = app.get_sub_app_mut(RenderApp) else {
let Some(render_app) = app.get_sub_app_mut(RenderApp) else {
return;
};
render_app.init_resource::<resources::OutlinePostProcessPipeline>();
}
}
fn prepare_post_processing_pipelines(
mut commands: Commands,
pipeline_cache: Res<PipelineCache>,
mut pipelines: ResMut<SpecializedRenderPipelines<OutlinePostProcessPipeline>>,
post_processing_pipeline: Res<OutlinePostProcessPipeline>,
views: Query<(Entity, &ExtractedView), With<OutlinePostProcessSettings>>,
) {
for (entity, view) in views.iter() {
let pipeline_id = pipelines.specialize(
&pipeline_cache,
&post_processing_pipeline,
OutlinePostProcessingPipelineKey {
texture_format: if view.hdr {
ViewTarget::TEXTURE_FORMAT_HDR
} else {
TextureFormat::bevy_default()
},
},
);
commands
.entity(entity)
.insert(OutlinePostProcessPipelineId(pipeline_id));
}
}
fn update_shader_clip_planes(
mut settings_query: Query<(Ref<Projection>, &mut OutlinePostProcessSettings)>,
) {
for (projection, mut settings) in settings_query.iter_mut() {
if projection.is_changed() {
if let Projection::Perspective(projection) = projection.into_inner() {
settings.camera_near = projection.near;
}
}
}
}

View File

@ -1,8 +1,25 @@
use bevy::{prelude::*, render::{render_graph::{ViewNode, NodeRunError, RenderGraphContext, RenderLabel}, view::ViewTarget, renderer::RenderContext, render_resource::{PipelineCache, BindGroupEntries, RenderPassDescriptor, RenderPassColorAttachment, Operations}, extract_component::ComponentUniforms}, core_pipeline::prepass::ViewPrepassTextures, ecs::query::QueryItem};
use bevy::{
core_pipeline::prepass::ViewPrepassTextures,
ecs::{query::QueryItem, system::lifetimeless::Read},
prelude::*,
render::{
extract_component::ComponentUniforms,
render_graph::{NodeRunError, RenderGraphContext, RenderLabel, ViewNode},
render_resource::{
BindGroupEntries, Operations, PipelineCache, RenderPassColorAttachment,
RenderPassDescriptor,
},
renderer::RenderContext,
view::ViewTarget,
},
};
use crate::components::OutlinePostProcessPipelineId;
use super::components;
use super::resources;
/// Label for outline post-process render node.
#[derive(RenderLabel, Clone, Eq, PartialEq, Hash, Debug)]
pub struct OutlineRenderLabel;
@ -11,26 +28,33 @@ pub struct OutlineRenderNode;
impl ViewNode for OutlineRenderNode {
type ViewQuery = (
&'static ViewTarget,
&'static ViewPrepassTextures,
&'static components::OutlinePostProcessSettings,
Read<ViewTarget>,
Read<ViewPrepassTextures>,
Read<OutlinePostProcessPipelineId>,
Read<components::OutlinePostProcessSettings>,
);
fn run(
&self,
_graph: &mut RenderGraphContext,
render_context: &mut RenderContext,
(view_target, view_prepass_textures, _): QueryItem<Self::ViewQuery>,
(
view_target,
view_prepass_textures,
pipeline_id,
_,
): QueryItem<Self::ViewQuery>,
world: &World,
) -> Result<(), NodeRunError> {
let render_pipeline = world.resource::<resources::OutlinePostProcessPipeline>();
let pipeline_cache = world.resource::<PipelineCache>();
let Some(pipeline) = pipeline_cache.get_render_pipeline(render_pipeline.pipeline_id) else {
let Some(pipeline) = pipeline_cache.get_render_pipeline(**pipeline_id) else {
warn!("Failed to get render pipeline from cache, skipping...");
return Ok(());
};
let uniforms = world.resource::<ComponentUniforms<components::OutlinePostProcessSettings>>();
let uniforms =
world.resource::<ComponentUniforms<components::OutlinePostProcessSettings>>();
let Some(uniform_binding) = uniforms.uniforms().binding() else {
error!("Failed to get settings uniform binding");
return Ok(());
@ -41,6 +65,11 @@ impl ViewNode for OutlineRenderNode {
return Ok(());
};
let Some(depth_buffer_view) = view_prepass_textures.depth_view() else {
error!("Failed to get normal buffer view");
return Ok(());
};
let post_process = view_target.post_process_write();
let bind_group = render_context.render_device().create_bind_group(
@ -51,6 +80,8 @@ impl ViewNode for OutlineRenderNode {
&render_pipeline.screen_sampler,
normal_buffer_view,
&render_pipeline.normal_sampler,
depth_buffer_view,
&render_pipeline.depth_sampler,
uniform_binding,
)),
);

View File

@ -1,4 +1,13 @@
use bevy::{prelude::*, render::{render_resource::{Sampler, BindGroupLayout, BindGroupLayoutEntries, ShaderStages, binding_types::{texture_2d, sampler, uniform_buffer}, TextureSampleType, SamplerBindingType, SamplerDescriptor, PipelineCache, RenderPipelineDescriptor, CachedRenderPipelineId, PrimitiveState, MultisampleState, FragmentState, ColorTargetState, TextureFormat, ColorWrites}, renderer::RenderDevice, texture::BevyDefault}, core_pipeline::fullscreen_vertex_shader::fullscreen_shader_vertex_state};
use bevy::{
core_pipeline::fullscreen_vertex_shader,
prelude::*,
render::{
render_resource::{
binding_types::{sampler, texture_2d, uniform_buffer}, BindGroupLayout, BindGroupLayoutEntries, ColorTargetState, ColorWrites, FragmentState, RenderPipelineDescriptor, Sampler, SamplerBindingType, SamplerDescriptor, ShaderStages, SpecializedRenderPipeline, TextureFormat, TextureSampleType
},
renderer::RenderDevice,
},
};
use super::components;
@ -7,7 +16,8 @@ pub struct OutlinePostProcessPipeline {
pub layout: BindGroupLayout,
pub screen_sampler: Sampler,
pub normal_sampler: Sampler,
pub pipeline_id: CachedRenderPipelineId,
pub depth_sampler: Sampler,
pub shader: Handle<Shader>,
}
impl FromWorld for OutlinePostProcessPipeline {
@ -23,6 +33,8 @@ impl FromWorld for OutlinePostProcessPipeline {
sampler(SamplerBindingType::Filtering),
texture_2d(TextureSampleType::Float { filterable: true }),
sampler(SamplerBindingType::Filtering),
texture_2d(TextureSampleType::Depth),
sampler(SamplerBindingType::NonFiltering),
uniform_buffer::<components::OutlinePostProcessSettings>(false),
),
),
@ -30,34 +42,53 @@ impl FromWorld for OutlinePostProcessPipeline {
let screen_sampler = render_device.create_sampler(&SamplerDescriptor::default());
let normal_sampler = render_device.create_sampler(&SamplerDescriptor::default());
let depth_sampler = render_device.create_sampler(&SamplerDescriptor::default());
let shader = world.resource::<AssetServer>().load::<Shader>("embedded://grex_outline_post_process/../assets/shaders/outline_post_process.wgsl");
let pipeline_id = world.resource_mut::<PipelineCache>().queue_render_pipeline(RenderPipelineDescriptor {
label: Some("outline_post_process_render_pipeline".into()),
layout: vec![layout.clone()],
push_constant_ranges: vec![],
vertex: fullscreen_shader_vertex_state(),
primitive: PrimitiveState::default(),
depth_stencil: None,
multisample: MultisampleState::default(),
fragment: Some(FragmentState {
shader,
shader_defs: vec![],
entry_point: "fragment".into(),
targets: vec![Some(ColorTargetState {
format: TextureFormat::bevy_default(),
blend: None,
write_mask: ColorWrites::ALL,
})],
}),
});
let shader = world.resource::<AssetServer>().load::<Shader>(
"embedded://bevy_outline_post_process/../assets/shaders/outline_post_process.wgsl",
);
Self {
layout,
screen_sampler,
normal_sampler,
pipeline_id,
depth_sampler,
shader,
}
}
}
/// A key that uniquely identifies a built-in postprocessing pipeline.
#[derive(Clone, Copy, PartialEq, Eq, Hash)]
pub struct OutlinePostProcessingPipelineKey {
/// The format of the source and destination textures.
pub texture_format: TextureFormat,
}
impl SpecializedRenderPipeline for OutlinePostProcessPipeline {
type Key = OutlinePostProcessingPipelineKey;
fn specialize(&self, key: Self::Key) -> RenderPipelineDescriptor {
RenderPipelineDescriptor {
label: Some("outline_post_processing".into()),
layout: vec![self.layout.clone()],
vertex: fullscreen_vertex_shader::fullscreen_shader_vertex_state(),
fragment: Some(FragmentState {
shader: self.shader.clone(),
shader_defs: vec![],
entry_point: "fragment".into(),
targets: vec![Some(ColorTargetState {
format: key.texture_format,
blend: None,
write_mask: ColorWrites::ALL,
})],
}),
primitive: default(),
depth_stencil: None,
multisample: default(),
push_constant_ranges: vec![],
zero_initialize_workgroup_memory: false,
}
}
}