Compare commits

...

14 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
12 changed files with 964 additions and 1663 deletions

2
.cargo/config.toml Normal file
View File

@ -0,0 +1,2 @@
[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

View File

@ -1,24 +0,0 @@
name: Rust
on:
push:
branches: [ "master" ]
pull_request:
branches: [ "master" ]
env:
CARGO_TERM_COLOR: always
jobs:
build:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
- name: Install Dependencies
run: sudo apt-get update; sudo apt-get install --no-install-recommends libasound2-dev libudev-dev
- name: Build
run: cargo build --verbose
- name: Run tests
run: cargo test --verbose

2230
Cargo.lock generated

File diff suppressed because it is too large Load Diff

View File

@ -1,7 +1,12 @@
[package]
name = "bevy_outline_post_process"
version = "0.2.0"
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,33 +1,39 @@
# bevy_outline_post_process
![License](https://img.shields.io/badge/license-MIT%2FApache-blue.svg)
[![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)
![Build](https://img.shields.io/github/actions/workflow/status/exvacuum/bevy_outline_post_process/rust.yml)
[![Docs](https://img.shields.io/website?url=https%3A%2F%2Fexvacuum.github.io%2Fbevy_outline_post_process%2F&label=docs)](https://exvacuum.github.io/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. Optionally supports adaptive outlining, so darker areas are outlined in white rather than black, based on luminance.
Note: This is a full-screen post process effect and cannot be enabled/disabled for specific objects.
## 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
bevy_outline_post_process::components::OutlinePostProcessSettings::new(2.0, 0.0, false);
bevy_outline_post_process::components::OutlinePostProcessSettings::new(2.0, LinearRgba::BLACK, 0.01, 0.01, 1.0);
```
## Compatibility
| Crate Version | Bevy Version |
|--- |--- |
| 0.2 | 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.bevy_outline_post_process]
git = "https://github.com/exvacuum/bevy_outline_post_process.git"
git = "https://git.soaos.dev/bevy_outline_post_process"
```
## Usage
@ -43,7 +49,6 @@ fn main() {
DefaultPlugins,
bevy_outline_post_process::OutlinePostProcessPlugin,
))
.insert_resource(Msaa::Off)
.run();
}
```
@ -51,11 +56,14 @@ fn main() {
When spawning a camera:
```rs
commands.spawn((
// Camera3dBundle...
NormalPrepass,
bevy_outline_post_process::components::OutlinePostProcessSettings::new(2.0, 0.0, false);
// 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,46 +2,94 @@
struct OutlinePostProcessSettings {
weight: f32,
threshold: f32,
adaptive: u32,
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;
// }}}
var outline = vec4f(delta_clipped, delta_clipped, delta_clipped, 0.0);
let luma = (0.2126 * screen_color.r + 0.7152 * screen_color.g + 0.0722 * screen_color.b);
if settings.adaptive != 0 && luma < 0.5 {
// 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);
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;
}
return screen_color - outline;
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,38 +1,63 @@
use bevy::{
core_pipeline::prepass::{DeferredPrepass, DepthPrepass, NormalPrepass},
prelude::*,
render::{extract_component::ExtractComponent, render_resource::ShaderType},
render::{extract_component::ExtractComponent, render_resource::{CachedRenderPipelineId, ShaderType}},
};
/// 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 {
/// 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.
threshold: f32,
/// Whether to use adaptive outlines. White outlines will be drawn around darker objects, while black ones will be drawn around lighter ones.
adaptive: u32,
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, threshold: f32, adaptive: bool) -> Self {
pub fn new(
weight: f32,
color: LinearRgba,
normal_threshold: f32,
depth_threshold: f32,
adaptive_threshold: f32,
) -> Self {
Self {
weight,
threshold,
adaptive: adaptive as u32,
color,
normal_threshold,
depth_threshold,
adaptive_threshold,
camera_near: 0.0,
}
}
}
}
impl Default for OutlinePostProcessSettings {
fn default() -> Self {
Self {
Self {
weight: 1.0,
threshold: 0.0,
adaptive: 0
color: LinearRgba::BLACK,
normal_threshold: 0.01,
depth_threshold: 0.05,
adaptive_threshold: 1.0,
camera_near: 0.0,
}
}
}
}

View File

@ -1,7 +1,7 @@
#![warn(missing_docs)]
//! A plugin for the Bevy game engine which provides an outline post-process effect. The effect
//! makes use of a normal prepass to generate outlines where differences in the normal buffer
//! makes use of a normal and depth prepass to generate outlines where significant differences in the values
//! occur.
use bevy::{
@ -11,11 +11,15 @@ use bevy::{
render::{
extract_component::{ExtractComponentPlugin, UniformComponentPlugin},
render_graph::{RenderGraphApp, ViewNodeRunner},
RenderApp,
render_resource::{PipelineCache, SpecializedRenderPipelines, TextureFormat},
view::{ExtractedView, ViewTarget},
Render, RenderApp, RenderSet,
},
};
use components::{OutlinePostProcessPipelineId, OutlinePostProcessSettings};
pub use nodes::OutlineRenderLabel;
use resources::{OutlinePostProcessPipeline, OutlinePostProcessingPipelineKey};
/// Components used by this plugin.
pub mod components;
@ -32,13 +36,19 @@ 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_systems(
Render,
prepare_post_processing_pipelines.in_set(RenderSet::Prepare),
)
.init_resource::<SpecializedRenderPipelines<OutlinePostProcessPipeline>>()
.add_render_graph_node::<ViewNodeRunner<nodes::OutlineRenderNode>>(
Core3d,
nodes::OutlineRenderLabel,
@ -54,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,6 +1,6 @@
use bevy::{
core_pipeline::prepass::ViewPrepassTextures,
ecs::query::QueryItem,
ecs::{query::QueryItem, system::lifetimeless::Read},
prelude::*,
render::{
extract_component::ComponentUniforms,
@ -14,6 +14,8 @@ use bevy::{
},
};
use crate::components::OutlinePostProcessPipelineId;
use super::components;
use super::resources;
@ -26,21 +28,27 @@ 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(());
};
@ -57,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(
@ -67,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,16 +1,11 @@
use bevy::{
core_pipeline::fullscreen_vertex_shader::fullscreen_shader_vertex_state,
core_pipeline::fullscreen_vertex_shader,
prelude::*,
render::{
render_resource::{
binding_types::{sampler, texture_2d, uniform_buffer},
BindGroupLayout, BindGroupLayoutEntries, CachedRenderPipelineId, ColorTargetState,
ColorWrites, FragmentState, MultisampleState, PipelineCache, PrimitiveState,
RenderPipelineDescriptor, Sampler, SamplerBindingType, SamplerDescriptor, ShaderStages,
TextureFormat, TextureSampleType,
binding_types::{sampler, texture_2d, uniform_buffer}, BindGroupLayout, BindGroupLayoutEntries, ColorTargetState, ColorWrites, FragmentState, RenderPipelineDescriptor, Sampler, SamplerBindingType, SamplerDescriptor, ShaderStages, SpecializedRenderPipeline, TextureFormat, TextureSampleType
},
renderer::RenderDevice,
texture::BevyDefault,
},
};
@ -21,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 {
@ -37,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),
),
),
@ -44,39 +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://bevy_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,
})],
}),
});
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,
}
}
}