Actually I tried using your uvPixelPerfect and it’s ALMOST fixed.
#if defined(WEBGL2) || defined(WEBGPU) || defined(NATIVE)
#define TEXTUREFUNC(s, c, l) texture2DLodEXT(s, c, l)
#else
#define TEXTUREFUNC(s, c, b) texture2D(s, c, b)
#endif
precision highp float;
varying vec3 vPosition;
varying vec2 vUV;
varying vec2 tUV;
uniform float time;
uniform float spriteCount;
uniform sampler2D spriteSheet;
uniform vec2 spriteMapSize;
uniform vec2 outputSize;
uniform vec2 stageSize;
uniform sampler2D frameMap;
uniform sampler2D tileMaps[LAYERS];
uniform sampler2D animationMap;
uniform vec3 colorMul;
float mt;
const float fdStep = 1. / 4.;
const float aFrameSteps = 1. / MAX_ANIMATION_FRAMES;
mat4 getFrameData(float frameID) {
float fX = frameID / spriteCount;
return mat4(texture2D(frameMap, vec2(fX, 0.), 0.),
texture2D(frameMap, vec2(fX, fdStep * 1.), 0.),
texture2D(frameMap, vec2(fX, fdStep * 2.), 0.), vec4(0.));
}
// see iq comment here: https://www.shadertoy.com/view/MllBWf
vec2 uvPixelPerfect(vec2 uv) {
vec2 res = vec2(textureSize(spriteSheet, 0));
uv = uv * res;
vec2 seam = floor(uv + 0.5);
uv = seam + clamp((uv-seam) / fwidth(uv), -0.5, 0.5);
return uv / res;
}
void main() {
vec4 color = vec4(0.);
vec2 tileUV = fract(tUV);
vec2 tileID = floor(tUV);
vec2 sheetUnits = 1. / spriteMapSize;
float spriteUnits = 1. / spriteCount;
vec2 stageUnits = 1. / stageSize;
for (int i = 0; i < LAYERS; i++) {
float frameID;
#define LAYER_ID_SWITCH
vec4 animationData =
TEXTUREFUNC(animationMap, vec2((frameID + 0.5) / spriteCount, 0.), 0.);
if (animationData.y > 0.) {
mt = mod(time * animationData.z, 1.0);
for (float f = 0.; f < MAX_ANIMATION_FRAMES; f++) {
if (animationData.y > mt) {
frameID = animationData.x;
break;
}
animationData = TEXTUREFUNC(
animationMap, vec2((frameID + 0.5) / spriteCount, aFrameSteps * f),
0.);
}
}
mat4 frameData = getFrameData(frameID + 0.5);
vec2 frameSize = vec2(32,32) / spriteMapSize;
vec2 offset = (frameData[0].xy) * sheetUnits;
vec4 nc = texture2D(spriteSheet, uvPixelPerfect(tileUV * frameSize + offset));
float alpha = min(color.a + nc.a, 1.0);
vec3 mixed = mix(color.xyz, nc.xyz, nc.a);
color = vec4(mixed, alpha);
}
color.xyz *= colorMul;
gl_FragColor = color;
}
That’s what I’m working with. It’s reduced from the one in the package.
vec4 nc = texture2D(spriteSheet, uvPixelPerfect(tileUV * frameSize + offset));
Here’s where I use the pixel perfect function.
With the spritesheet set to Texture.TRILINEAR_SAMPLINGMODE
Now this renders tiles PERFECTLY except for the edges.
So close… The edges flicker.