Renamed + Added docs

This commit is contained in:
Silas Bartha 2024-06-03 21:10:39 -04:00
parent 6dec61d81f
commit 369cac3c32
Signed by: soaos
GPG Key ID: 9BD3DCC0D56A09B2
9 changed files with 153 additions and 43 deletions

50
.github/workflows/docs.yml vendored Normal file
View File

@ -0,0 +1,50 @@
name: Docs
on:
push:
branches: [master]
permissions:
contents: read
pages: write
id-token: write
concurrency:
group: deploy
cancel-in-progress: false
jobs:
build:
name: Build
runs-on: ubuntu-latest
steps:
- name: Checkout repository
uses: actions/checkout@v4
- name: Install Dependencies
run: sudo apt-get update; sudo apt-get install --no-install-recommends libasound2-dev libudev-dev
- name: Setup Rust
uses: dtolnay/rust-toolchain@stable
- name: Configure cache
uses: Swatinem/rust-cache@v2
- name: Setup pages
id: pages
uses: actions/configure-pages@v4
- name: Clean docs folder
run: cargo clean --doc
- name: Build docs
run: cargo doc --no-deps
- name: Add redirect
run: echo '<meta http-equiv="refresh" content="0;url=bevy_dither_post_process/index.html">' > target/doc/index.html
- name: Remove lock file
run: rm target/doc/.lock
- name: Upload artifact
uses: actions/upload-pages-artifact@v3
with:
path: target/doc
deploy:
name: Deploy
environment:
name: github-pages
url: ${{ steps.deployment.outputs.page_url }}
runs-on: ubuntu-latest
needs: build
steps:
- name: Deploy to GitHub Pages
id: deployment
uses: actions/deploy-pages@v4

24
.github/workflows/rust.yml vendored Normal file
View File

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

View File

@ -1,6 +1,6 @@
[package] [package]
name = "grex_dither_post_process" name = "bevy_dither_post_process"
version = "0.1.3" version = "0.1.4"
edition = "2021" edition = "2021"
[dependencies.bevy] [dependencies.bevy]

View File

@ -1,4 +1,4 @@
# grex_dither_post_process # bevy_dither_post_process
A plugin for the [Bevy](https://bevyengine.org) engine which adds a dither post-processing effect. A plugin for the [Bevy](https://bevyengine.org) engine which adds a dither post-processing effect.
@ -10,7 +10,7 @@ The effect is implemented as a bilevel ordered dither using a Bayer matrix with
![](./doc/screenshot_plant.png) ![](./doc/screenshot_plant.png)
Configuration Used: Configuration Used:
```rs ```rs
grex_dither_post_process::components::DitherPostProcessSettings::new(3, &asset_server); bevy_dither_post_process::components::DitherPostProcessSettings::new(3, &asset_server);
``` ```
## Compatibility ## Compatibility
@ -22,8 +22,8 @@ grex_dither_post_process::components::DitherPostProcessSettings::new(3, &asset_s
### Using git URL in Cargo.toml ### Using git URL in Cargo.toml
```toml ```toml
[dependencies.grex_dither_post_process] [dependencies.bevy_dither_post_process]
git = "https://github.com/exvacuum/grex_dither_post_process.git" git = "https://github.com/exvacuum/bevy_dither_post_process.git"
``` ```
## Usage ## Usage
@ -31,13 +31,13 @@ git = "https://github.com/exvacuum/grex_dither_post_process.git"
In `main.rs`: In `main.rs`:
```rs ```rs
use bevy::prelude::*; use bevy::prelude::*;
use grex_dither_post_process; use bevy_dither_post_process;
fn main() { fn main() {
App::new() App::new()
.add_plugins(( .add_plugins((
DefaultPlugins, DefaultPlugins,
grex_dither_post_process::DitherPostProcessPlugin, bevy_dither_post_process::DitherPostProcessPlugin,
)) ))
.run(); .run();
} }
@ -47,7 +47,7 @@ When spawning a camera:
```rs ```rs
commands.spawn(( commands.spawn((
// Camera3dBundle... // Camera3dBundle...
grex_dither_post_process::components::DitherPostProcessSettings::new(level, &asset_server); bevy_dither_post_process::components::DitherPostProcessSettings::new(level, &asset_server);
)); ));
``` ```

View File

@ -9,16 +9,16 @@
fn fragment( fn fragment(
in: FullscreenVertexOutput in: FullscreenVertexOutput
) -> @location(0) vec4<f32> { ) -> @location(0) vec4<f32> {
let screen_size = vec2f(textureDimensions(screen_texture)); let screen_size = vec2i(textureDimensions(screen_texture));
let threshold_map_size = vec2f(textureDimensions(threshold_map_texture)); let threshold_map_size = vec2i(textureDimensions(threshold_map_texture));
let pixel_position = floor(in.uv * screen_size); let pixel_position = vec2i(floor(in.uv * vec2f(screen_size)));
let map_position = (pixel_position % threshold_map_size) / threshold_map_size; let map_position = vec2f(pixel_position % threshold_map_size) / vec2f(threshold_map_size);
let threshold = textureSample(threshold_map_texture, threshold_map_sampler, map_position).r; let threshold = textureSample(threshold_map_texture, threshold_map_sampler, map_position).r;
let base_color = textureSample(screen_texture, screen_sampler, in.uv); let base_color = textureSample(screen_texture, screen_sampler, in.uv);
let luma = (0.2126 * base_color.r + 0.7152 * base_color.g + 0.0722 * base_color.b); let luma = (0.2126 * base_color.r + 0.7152 * base_color.g + 0.0722 * base_color.b);
let value = f32(luma > threshold); let value = f32(luma >= threshold);
return vec4f(value, value, value, 1.0); return vec4f(value, value, value, 1.0);
} }

View File

@ -1,19 +1,21 @@
use bevy::{ use bevy::{
prelude::*, prelude::*,
render::{ render::{
extract_component::ExtractComponent,
render_asset::RenderAssetUsages, render_asset::RenderAssetUsages,
render_resource::{Extent3d, TextureDimension, TextureFormat, TextureUsages}, extract_component::ExtractComponent, render_resource::{Extent3d, TextureDimension, TextureFormat, TextureUsages},
}, },
}; };
/// Component which, when inserted into an entity with a camera, enables the dither post-processing
/// effect.
#[derive(Component, ExtractComponent, Clone)] #[derive(Component, ExtractComponent, Clone)]
pub struct DitherPostProcessSettings(Handle<Image>); pub struct DitherPostProcessSettings(Handle<Image>);
impl DitherPostProcessSettings { impl DitherPostProcessSettings {
pub fn new( /// Constructs a new instance ofthis component, enabling the dither effect using a bayer
level: u32, /// matrix of the given level. A given level *n* will generate a square bayer matrix with a size of *(n+1)^2*.
asset_server: &AssetServer, pub fn new(level: u32, asset_server: &AssetServer) -> Self {
) -> Self {
let power = level + 1; let power = level + 1;
let map_size: u32 = 1 << power; let map_size: u32 = 1 << power;
let mut buffer = Vec::<u8>::new(); let mut buffer = Vec::<u8>::new();
@ -38,7 +40,6 @@ impl DitherPostProcessSettings {
let value = ((result as f32 / map_size.pow(2) as f32) * 255.0) as u8; let value = ((result as f32 / map_size.pow(2) as f32) * 255.0) as u8;
buffer.push(value); buffer.push(value);
} }
} }
let mut image = Image::new( let mut image = Image::new(
@ -52,15 +53,16 @@ impl DitherPostProcessSettings {
TextureFormat::R8Unorm, TextureFormat::R8Unorm,
RenderAssetUsages::RENDER_WORLD, RenderAssetUsages::RENDER_WORLD,
); );
image.texture_descriptor.usage = image.texture_descriptor.usage = TextureUsages::COPY_DST
TextureUsages::COPY_DST | TextureUsages::STORAGE_BINDING | TextureUsages::TEXTURE_BINDING; | TextureUsages::STORAGE_BINDING
| TextureUsages::TEXTURE_BINDING;
let handle = asset_server.add(image); let handle = asset_server.add(image);
Self(handle) Self(handle)
} }
/// Gets the handle of the texture representing this component's Bayer matrix
pub fn handle(&self) -> Handle<Image> { pub fn handle(&self) -> Handle<Image> {
self.0.clone() self.0.clone()
} }
} }

View File

@ -1,40 +1,57 @@
use bevy::{prelude::*, render::{RenderApp, render_graph::{RenderGraphApp, ViewNodeRunner}, extract_component::ExtractComponentPlugin}, asset::embedded_asset, core_pipeline::core_3d::graph::{Core3d, Node3d}}; #![warn(missing_docs)]
//! A plugin for the Bevy game engine which provides a black and white dither post-process effect
//! using Bayer ordered dithering.
use bevy::{
asset::embedded_asset,
core_pipeline::core_3d::graph::{Core3d, Node3d},
prelude::*,
render::{
extract_component::ExtractComponentPlugin,
render_graph::{RenderGraphApp, ViewNodeRunner},
RenderApp,
},
};
use crate::components::DitherPostProcessSettings; use crate::components::DitherPostProcessSettings;
pub use nodes::DitherRenderLabel; pub use nodes::DitherRenderLabel;
/// Plugin which provides dither post-processing functionality
pub struct DitherPostProcessPlugin; pub struct DitherPostProcessPlugin;
/// Components used by this plugin
pub mod components; pub mod components;
mod resources;
mod nodes; mod nodes;
mod resources;
impl Plugin for DitherPostProcessPlugin { impl Plugin for DitherPostProcessPlugin {
fn build(&self, app: &mut App) { fn build(&self, app: &mut App) {
embedded_asset!(app, "../assets/shaders/dither_post_process.wgsl"); embedded_asset!(app, "../assets/shaders/dither_post_process.wgsl");
app.add_plugins(( app.add_plugins((ExtractComponentPlugin::<DitherPostProcessSettings>::default(),));
ExtractComponentPlugin::<DitherPostProcessSettings>::default(),
));
let Ok(render_app) = app.get_sub_app_mut(RenderApp) else { let Ok(render_app) = app.get_sub_app_mut(RenderApp) else {
return; return;
}; };
render_app.add_render_graph_node::<ViewNodeRunner<nodes::DitherRenderNode>>( render_app
Core3d, .add_render_graph_node::<ViewNodeRunner<nodes::DitherRenderNode>>(
nodes::DitherRenderLabel, Core3d,
).add_render_graph_edges(
Core3d,
(
Node3d::Tonemapping,
nodes::DitherRenderLabel, nodes::DitherRenderLabel,
Node3d::EndMainPassPostProcessing, )
), .add_render_graph_edges(
); Core3d,
(
Node3d::Tonemapping,
nodes::DitherRenderLabel,
Node3d::EndMainPassPostProcessing,
),
);
} }
fn finish(&self, app: &mut App) { fn finish(&self, app: &mut App) {
let Ok(render_app) = app.get_sub_app_mut(RenderApp) else { let Ok(render_app) = app.get_sub_app_mut(RenderApp) else {
return; return;

View File

@ -1,8 +1,22 @@
use bevy::{prelude::*, render::{render_graph::{ViewNode, NodeRunError, RenderGraphContext, RenderLabel}, view::ViewTarget, renderer::RenderContext, render_resource::{PipelineCache, BindGroupEntries, RenderPassDescriptor, RenderPassColorAttachment, Operations}, render_asset::RenderAssets}, ecs::query::QueryItem}; use bevy::{
ecs::query::QueryItem,
prelude::*,
render::{
render_asset::RenderAssets,
render_graph::{NodeRunError, RenderGraphContext, RenderLabel, ViewNode},
render_resource::{
BindGroupEntries, Operations, PipelineCache, RenderPassColorAttachment,
RenderPassDescriptor,
},
renderer::RenderContext,
view::ViewTarget,
},
};
use super::components; use super::components;
use super::resources; use super::resources;
/// Label for dither post-process effect render node.
#[derive(RenderLabel, Clone, Eq, PartialEq, Hash, Debug)] #[derive(RenderLabel, Clone, Eq, PartialEq, Hash, Debug)]
pub struct DitherRenderLabel; pub struct DitherRenderLabel;
@ -31,7 +45,10 @@ impl ViewNode for DitherRenderNode {
let post_process = view_target.post_process_write(); let post_process = view_target.post_process_write();
let Some(threshold_map) = world.resource::<RenderAssets<Image>>().get(dither_post_process_settings.handle()) else { let Some(threshold_map) = world
.resource::<RenderAssets<Image>>()
.get(dither_post_process_settings.handle())
else {
warn!("Failed to get threshold map, skipping..."); warn!("Failed to get threshold map, skipping...");
return Ok(()); return Ok(());
}; };

View File

@ -43,7 +43,7 @@ impl FromWorld for DitherPostProcessPipeline {
let threshold_map_sampler = render_device.create_sampler(&SamplerDescriptor::default()); let threshold_map_sampler = render_device.create_sampler(&SamplerDescriptor::default());
let shader = world.resource::<AssetServer>().load::<Shader>( let shader = world.resource::<AssetServer>().load::<Shader>(
"embedded://grex_dither_post_process/../assets/shaders/dither_post_process.wgsl", "embedded://bevy_dither_post_process/../assets/shaders/dither_post_process.wgsl",
); );
let pipeline_id = let pipeline_id =