Repeated texture overlay for tilemaps

This shader can be used to apply a repeated texture to a tilemap. It will only apply the texture to fully red parts of the tiles, so that edges can be preserved if you wish. Check out the demo project for the full set up.

Technically you can use this shader for any sprites, not just tiles, but tiles would be the most common use case.

Shader code
shader_type canvas_item;

uniform sampler2D overlay_tex: repeat_enable, filter_nearest;
uniform float scale = 0.006944444; // calculated by 1/texture size e.g. 1/144
varying vec2 world_position;

void vertex(){
	// calculate the world position for use in the fragment shader
	world_position = (MODEL_MATRIX * vec4(VERTEX, 0.0, 1.0)).xy;
}

void fragment() {
	// only apply overlay_tex on the fully red parts of the original tiles
	float mix_amount = floor(COLOR.r);
	
	// sample the overlay_tex using worldPos
	vec4 overlay_color = texture(overlay_tex, world_position * scale);
	
	// combine original color and overlay color together
	COLOR = mix(COLOR, overlay_color, mix_amount);
}
Live Preview
Tags
grass, grass tiles, repeated texture, seamless, tilemap
The shader code and all code snippets in this post are under CC0 license and can be used freely without the author's permission. Images and videos, and assets depicted in those, do not fall under this license. For more info, see our License terms.

More from jess.codes

Related shaders

guest

4 Comments
Oldest
Newest Most Voted
Inline Feedbacks
View all comments
Russell
Russell
1 year ago

this is very nice, thanks for sharing

Sebastian
Sebastian
1 year ago

I love your videos. Can you show us the pixelart water shader sometime?

Zinic
Zinic
6 months ago

If you get texture bleeding (texture is applied over a large area when a red pixel is adjacent to a transparent pixel) then I found there are a couple ways to address this:

  • If the red pixel(s) is from a tile and is adjacent to another tile that has transparent pixels, then you can fix the bleeding by padding the tiles with a colored stripe first and importing the atlas and configuring it to have 1 pixel separation within Godot.
  • Fill transparent pixels with a color set to an opacity of 20%. This will prevent bleeding onto the transparent pixel. This may be an acceptable solution if you know the color below the transparent layer and setting the color to match the color below it. This can make the opacity nearly unnoticeable.
  • Use the double texture method from the video. Set the transparent pixels to blue and then just convert it back to transparent.
Advisor
Advisor
3 months ago
Reply to  Zinic

use built-in method of Image.fix_alpha_edges()

everything is done already

EDIT: unrelated to shader (just raw image data of texture)

Last edited 3 months ago by Advisor