-
Notifications
You must be signed in to change notification settings - Fork 40
Quick Start Tutorial
This tutorial will walk you through writing a shader that makes stone (or some other block you choose) have an animated disco glow! It won't teach you GLSL or much about the details, but you'll get a rough idea of what is involved and where to start. Here's what it will look like when we are done:
For this tutorial, you'll be using a resource pack. There are other ways to associate shaders with blocks, but this way requires no Java development.
You'll need Minecraft instance that includes Fabric Loader, Fabric API and Canvas 1.0 or later. In the folder where Minecraft runs, open the resourcepacks
folder and create a new sub folder named shaderpack
. (It can be any name, we're just using that name for the tutorial.)
In the shaderpack
folder, use a text editor to create a filed named pack.mcmeta
with these contents:
{
"pack": {
"pack_format": 5,
"description": "Canvas shader tutorial pack"
}
}
Within the shaderpack
folder create the following subfolders:
assets/minecraft/materials
assets/minecraft/shaders/material
assets/minecraft/materialmaps/block
Start Minecraft and confirm shaderpack
is now in the list of available resource packs and then load it. Because it is empty, loading it will not have any effect.
For this effect, we need a vertex shader and a fragment shader.
The vertex shader collects information from incoming quad vertices and makes it available to the fragment shader, usually via "varying" variables. (Yes, I know, all variables are varying, aren't they? It's just how OpenGL names things. You get used to it.)
Create tutorial.vert
with the code below and place it in assets/minecraft/shaders/material
:
#include canvas:shaders/api/vertex.glsl
#include canvas:shaders/lib/face.glsl
// sends noise coordinates from the vertex shader
varying vec2 v_noise_uv;
// The name of this method is special - Canvas will call it for each vertex associated with your shader
void cv_startVertex(inout cv_VertexData data) {
// 2D noise coordinates are derived from world geometry using a Canvas library function
v_noise_uv = cv_faceUv(data.vertex.xyz, data.normal);
}
The fragment shader modifies individual pixels or sub pixels (the "fragments") before they are drawn to the screen. This is where we will do most of the work for this effect.
Create tutorial.frag
with the code below and place it in assets/minecraft/shaders/material
:
#include canvas:shaders/api/fragment.glsl
#include canvas:shaders/lib/math.glsl
#include canvas:shaders/api/world.glsl
// holds our noise coordinates from the vertex shader
varying vec2 v_noise_uv;
// The name of this method is special - Canvas will call it for each fragment associated with your shader
void cv_startFragment(inout cv_FragmentData fragData) {
// modify appearance where stone texture is lighter in color
if (cv_luminance(fragData.spriteColor.rgb) > 0.5) {
// get a time value we can use for animation
float time = cv_renderSeconds();
// use an animated noise function to mix a pastel blue/red color
float color_weight = cv_noise2dt(v_noise_uv * 2.0, time);
// mix 'em up!
vec4 highlight = mix(vec4(1.0, 0.7, 1.0, 1.0), vec4(0.7, 1.0, 1.0, 1.0), color_weight);
// call animated noise function with noise coordinates scaled and shifted
float blend_weight = cv_noise2dt(v_noise_uv * 4.0 - 16.0, time * 2.0);
// mix with the stone texture colors
fragData.spriteColor = mix(fragData.spriteColor, highlight, blend_weight);
// make these fragments fully lit
fragData.emissivity = 1.0;
fragData.ao = false;
fragData.diffuse = false;
}
}
Now that we have shaders, we need to let Canvas know about them so they can be applied to models. This is done by creating a RenderMaterial
(usually just called "materials") to describe how a quad should be rendered. The Fabric Rendering API includes materials already, but Canvas lets you add shaders to them.
We're going to name our material minecraft:tutorial
for simplicity. For your own materials you should use a different name (typically the name of your mod) instead of minecraft
so that your materials don't have name conflicts with others.
Create tutorial.json
with the text below and place it in assets/minecraft/materials
:
{
"layers": [
{
"vertexSource": "minecraft:shaders/material/tutorial.vert",
"fragmentSource": "minecraft:shaders/material/tutorial.frag"
}
]
}
NOTE: The extensions for adding shaders to render materials and the JSON loader for render are actually part of FREX. But, Canvas always includes FREX, so you don't need to worry about that.
Now that Canvas has a material containing our shaders, we have to let it know what blocks to use it with. There are several ways to do this, but for this tutorial we're going to create a Material Map.
A material maps does what it sounds like: maps specific materials to blocks or items or even to specific textures on a them. Stone only has one texture so the material map for this tutorial is very simple.
The name and location of the material map file must match that of the block or item it handles, similar to the way blockstate model files are named and located.
Create stone.json
with the text below and place it in assets/minecraft/materialmaps/block
:
{
"defaultMaterial": "minecraft:tutorial"
}
That's it! Start up Minecraft and if everything was done correctly your stone blocks should look like those int the video at the top of the page.
For more information on creating shaders with Canvas, return to the home page.