Skip to content

Latest commit

 

History

History
executable file
·
437 lines (316 loc) · 14 KB

bevy-shaders-cheatsheet.md

File metadata and controls

executable file
·
437 lines (316 loc) · 14 KB

Bevy-Shader-Cheatsheet.

I've found the whole 'shader-land' story around Bevy to be pretty impenetrable, this is a small attempt to collect some of my notes -- made public in the hope it will alleviate the potential suffering of others.

You will see single-letter-variables in shaders everywhere! Don't contribute to the problem, be part of the solution.


Contents:


uvs:

// 'uv's are in the MeshVertexOutput
#import bevy_pbr::forward_io::VertexOutput

@fragment
fn fragment(in: VertexOutput) -> vec4<f32> {
	// usually you'll see them used in your fragment shaders thusly:
	let uv = in.uv;
  let normalised_uv = (in.uv.xy * 2.0) - 1.0; // If you want 0,0 to be at the 'center' of your Mesh's geometry.

	return vec4f(uv.x, uv.y, 0.0, 1.0);
}

resolution:

You'll usually need this when you go to say, draw a sdCircle or other sdf shape and suddenly realise that it's squashed because the aspect ratio of the window is not being accounted for.

  • Most of what you need is in View
  • Don't forget the @group(0) @binding(0) var<uniform> view: View; to make the uniform available.
#import bevy_render::view View

@group(0) @binding(0) var<uniform> view: View;

@fragment
fn fragment(in: VertexOutput) -> vec4<f32> {
  	// example, shader-toyesk normalise and account for aspect ratio:
    var uv = (in.uv.xy * 2.0) - 1.0;
    let resolution = view.viewport.zw;
    uv.x *= resolution.x / resolution.y;
    uv *= rotate2D(PI / -2.0); // Our uvs are by default a -1,-1 in uppermost left cnr, this rotates you around.

    // your code here.


}

time:

#import bevy_pbr::mesh_view_bindings globals
#import bevy_pbr::forward_io::VertexOutput

const TAU:f32 =  6.28318530718;

@fragment
fn fragment(in: VertexOutput) -> @location(0) vec4<f32> {
    var uv = (in.uv * 2.0) - 1.0;
    var col = vec3f(0.);

    let distance_to_center = vec2(0.25) - uv;
    let angle = atan2(distance_to_center.y, distance_to_center.x);
    let radius = length(distance_to_center) * 2.0;

    col = hsv_to_srgb(vec3f((angle / TAU) + globals.time / 3.0, radius, 1.0));
    return vec4f(col, 1.0);
}

// From the bevy source code
fn hsv_to_srgb(c: vec3<f32>) -> vec3<f32> {
    let K = vec4(1.0, 2.0 / 3.0, 1.0 / 3.0, 3.0);
    let p = abs(fract(c.xxx + K.xyz) * 6.0 - K.www);
    return c.z * mix(K.xxx, clamp(p - K.xxx, vec3(0.0), vec3(1.0)), c.y);
}

NOTE: if you're in 2d, the globals is in a diff spot: #import bevy_sprite::mesh2d_view_bindings globals NOTE: #import bevy_sprite::mesh2d_vertex_output::VertexOutput is also NOT the same as bevy_pbr::forward_io::VertexOutput, make sure you're using the right one!


pixels

  • Want to get a line the width of a single pixel?

from assets/shaders/shadertoy-ports/water-caustic-tileable.wgsl

#import bevy_render::view  View
@group(0) @binding(0) var<uniform> view: View;
    
@fragment
fn fragment(in: VertexOutput) -> @location(0) vec4<f32> {
    let time: f32 = globals.time;
    var uv: vec2<f32> = in.uv;
   
    // Shows a yellow grid, where the lines are 1px, calculated by looking at your the available uvs.
    let pixel: vec2<f32> = vec2<f32>(2.0) / view.viewport.zw;
    uv *= 2.0;
    let f: f32 = floor(time * 0.5 % 2.0);
    let first: vec2<f32> = step(pixel, uv) * f;
    uv = step(fract(uv), pixel);
    colour = mix(colour, vec3<f32>(1.0, 1.0, 0.0), (uv.x + uv.y) * first.x * first.y);

    return vec4<f32>(colour, 1.0);
}

sdf-shapes


noise


uuid-generation:

  • you need uuids for the ECS, whenever you're deriving TypeUuid on stuff you plan on putting in there (which we do plan on doing)
#[derive(AsBindGroup, TypeUuid, TypePath, Debug, Clone)]
#[uuid = "c74e039a-3df7-4f71-bd1d-7fe4b25a2230"]
struct MyShader{}

Here's an online-generator, that I use a bit. if you're feeling lazy here's 25 generated earlier

f6395028-088c-4d0c-bf79-9bb238b79768
83db0271-2867-4477-9cad-afd89bd93abc
688566d8-3ca3-4401-80bf-19db02f7ab8c
42ecf202-ab31-47d0-99e8-62cc59489b29
4b919679-6311-4cc9-a374-131500abc512
ff12d937-9a01-4abe-91cf-13457d33df80
21f7e8f1-c950-409b-a2eb-8457e10d0070
f5f0c9a4-9834-4edf-ae47-071a40f9eafd
2ba4a8b0-d4b4-4f00-a6a0-389afc4014ce
c6d009c9-c5f2-41c9-8068-b6f68fd06969
6907e7a9-6a12-444d-9923-e347f609ca95
c8b1af9a-b54c-42e2-b479-f52e0ed6f4da
87b235fb-4451-4de5-a843-e7782292b9b3
d4e7bf47-b6d3-4ff9-86ce-874b7d32743e
8efd22a3-17b2-43de-bac1-a064142c96ae
2a144e21-4e01-42d7-a2e8-5078793fb16c
29746aa9-e63e-424c-a951-ef14b0466946
ed5396f9-26cc-4f40-9123-2b302d729ecf
c1887ade-4669-4c96-ac08-2a1eb2ebfb33
14d87951-1436-4055-8130-df22881a6774
2b72b9be-4d3f-4f21-bf4f-589032bb43d5
1f74affc-1fd3-49e5-944d-969f4d253e64
cca52d9e-2bd4-4fc0-b058-7bd3ddd1aba5
6feca34e-6fa7-4fa4-a7c8-d914382108ba
fd9ec163-e144-478d-a085-702d028f1149

or, you can use quuidy:

cargo install quuidy
quuidy -n 10

get-data-into-your-shader

If none of that makes sense: I wrote a blog-post, on this. Take this shader code dotted-line-shader Then use this rust-binding-code. (copied below for the lazy)

// in src/YOURMODULE.rs
#[derive(AsBindGroup, TypeUuid, TypePath, Debug, Clone)]
#[uuid = "c74e039a-3df7-4f71-bd1d-7fe4b25a2230"]
struct DottedLineShader {
    #[uniform(100)] // Make the uniform a big(ish) number to avoid clashing with existing bindings.
    uniforms: Holder, //RGBA
}

/// Simplified holding struct to make passing across uniform(n) simpler.
#[derive(ShaderType, Default, Clone, Debug)]
struct Holder {
    tint: Color,
    /// How wide do you want the line as a % of its available uv space: 0.5 would be 50% of the surface of the geometry
    line_width: f32,
    /// How many segments (transparent 'cuts') do you want?
    segments: f32,
    /// How fast do you want the animation to be? set 0.0 to disable.
    phase: f32,
    /// How far spaced apart do you want these lines?
    line_spacing: f32,
}

impl Material for DottedLineShader {
    fn fragment_shader() -> ShaderRef {
        "shaders/dotted_line.wgsl".into()
    }
}

In your main.rs:

app.add_plugins(MaterialPlugin::<YOURMODULEPATH::DottedLineShader>::default());

smoothstep

smoothstep interpolates between two 'edges', leftedge, sometimes called edge0 and rightedge, sometimes called edge1, for a given x. i.e make the shape of the below graph, where all values are clamped between those two edges. So if your x is <= to leftedge smoothstep returns you a 0. if your x is >= to the rightedge, smoothstep returns you a 1.

smoothstep

or,

smoothstep

or, The smoothstep function takes three parameters:

    LeftEdge: The lower edge of the range.
    RightEdge: The upper edge of the range.
    x: The input value that you want to interpolate between those above Edges.

step

fn step(limit, value) any value under the limit will return a 0.0, anything above value a 1.0; available on f32, vecN<T> i.e all vec2/3/4 types with any isize/usize/f32 etc.

can be useful to replace if statements, because a numerical solve is (rumoured to be) superior in performance to a branch:

// From the dotted_line.wgsl mentioned elsewhere in this guide.
    // draw x line
    if abs(uv.x) < 0.025{
        uv += t * 0.5;
        // segment the line, only tint the areas we want
        var uv_segmented: vec2<f32> = fract(uv * 3.0) * 0.05;
        let step_y: f32 = step(0.025, abs(uv_segmented.y)); // you can read this as `if (abs(uv_segmented.y) > 0.025)/*...*/ ``
        col += vec4f(0.23, 0.88, 0.238, 1.0)* step_y;
    }

code


syntax diffs

NOTE: this is not an exhaustive list!

  • glsl has mod but in wgsl you need to use %
  • bindings in glsl by default are mutable, not so in wgsl, use var for mutable let for immutable.
  • in glsl you'll see the in keyword, which do to a similar thing with pointers in wgsl:
fn testing (uv: ptr<function, vec2<f32>>) {
    (*uv).x = 4.0;
}
  • You'll often see glsl functions that take a value to mutate, and they return nothing: this is invalid in wgsl so this:
void pR(inout vec2 p, float a) {
	p = cos(a)*p + sin(a)*vec2(p.y, -p.x);
}

becomes:

fn pR(out_point: vec2<f32>, angle: f32)-> vec2f {
    var point = out_point;
    point = cos(angle) * point + sin(angle) * vec2<f32>(point.y, -point.x);
    return point;
}

from https://www.shadertoy.com/view/4t2cR1

  • in glsl, when declaring structs, you terminate the lines of the fields with ;, in wgsl you use ,so: this:
  struct geometry {
  float dist;
  vec3 hit;
  int iterations;
  };

becomes:

struct Geometry {
    dist: f32,
    hit: vec3<f32>,
    iterations: i32,
}
  • You may be tempted to do multipart assignments on the rhs, like this:

    O += 0.2 / (abs(length(I = p / (r + r - p).y) * 80.0 - i) + 40.0 / r.y) but, this I = ... is INVALID, you will see this a lot in shadertoy, in particular when fancy shader-wizards are attempting to not summon Cthulhu by exceeding the 300char limit, but in .wgsl land you gotta do the assignment outside.

  • vec3 u = 1.-(--f)*f*f*f*-f; Note, the -- is not legal in wgsl, so don't do it instead do:

    f -= vec3<f32>(1.0, 1.0, 1.0); // Decrement each component of the vector by 1
    var u: vec3<f32> = 1.0 - f * f * f * f * -f;

-- single inlined if/else's with an assignment i.e,

let functionSign: f32 = if mp.dist < 0.0 { -1.0 } else { 1.0 };

are not allowed, you need to do this instead:

    var functionSign: f32;
    if mp.dist < 0.0 {
        functionSign = -1.0;
    } else {
        functionSign = 1.0;
    };

importable

how I came about this knowledge

A few things to note on imports:

  1. There are multiple syntaxes supported (I do not know for how long this will stay the case)
  2. The 'new' way, will throw false errors in wgsl-analyzer, which if you're not already using--YOU SHOULD BE!

Example importing:

  • In the shader code YOU ARE WRITING:

source file: src/shader_utils/common.wgsl

#import shadplay::shader_utils::common NEG_HALF_PI, shaderToyDefault, rotate2D

NOTE: this shader source (i.e the .wgsl code you are wanting to be importable), MUST live inside your crate, it cannot live above the src directory your main.rs/lib.rs etc live in.

  • In the shader code that is your library ( the stuff you want to define, and import into the above)
#define_import_path shadplay::shader_utils::common

_it appears, that you can basically name these whatever you like, the convention observed in the bevy engine's codebase is what I've done above.-

The other way:

  • Say the common.wgsl was alongside your shader, in the shadplay repo that may look like this:
/shaders
├──/common.wgsl      <<< YOUR LIBRARY
├──/myshader.wgsl    <<< THAT YOU WANT TO IMPORT HERE
├──/myshader_2d.wgsl <<< THAT YOU WANT TO IMPORT HERE

The asset-path syntax can be used:

  • In the shader code YOU ARE WRITING:

source file: src/shader_utils/common.wgsl

import it using asset-path syntax: #import "shaders/common.wgsl"
  • You don't need to do the #define shinnenagans in the common.wgsl.

  • one may prefer this style if you have the assets/shaders/*.wgsl layout in your project.

Some very useful imports from bevy:

Famous mathematical constants are used prolifically in shaders (at least on shadertoy), there are many in the bevy_pbr::utils module, and there are even more in the shadplay::common::common, you can see examples of these throughout the shaders in assets/shaders.