Renamed + Added docs
This commit is contained in:
parent
6dec61d81f
commit
369cac3c32
50
.github/workflows/docs.yml
vendored
Normal file
50
.github/workflows/docs.yml
vendored
Normal 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
24
.github/workflows/rust.yml
vendored
Normal 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
|
@ -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]
|
||||||
|
14
README.md
14
README.md
@ -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
|
|||||||

|

|
||||||
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);
|
||||||
));
|
));
|
||||||
```
|
```
|
||||||
|
|
||||||
|
@ -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);
|
||||||
}
|
}
|
||||||
|
@ -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()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
49
src/lib.rs
49
src/lib.rs
@ -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;
|
||||||
|
21
src/nodes.rs
21
src/nodes.rs
@ -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(());
|
||||||
};
|
};
|
||||||
|
@ -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 =
|
||||||
|
Loading…
x
Reference in New Issue
Block a user