Scroll a SpriteMap?

image
Ok after changing the movement method I was able to control it a little better and see the effect you are.
All the lines should stay straight, but its like its rounding some of the sampling depending on fractional values or something.

So its looking more like a sampling problem, we might need someone who knows more about why this would be happening as its not specific to the SpriteMap but would be the Texture sampling its self, I bet if we took a screenshot of this same setup and then had it on a plane that we were moving with the same sampling mode the same jitter would happen.

So I’m actually kind of intrigued now.

@Evgeni_Popov you are the king of nerds, any idea what would be causing this jitter in the texture?

I don’t reproduce, but we recently added a “pixel perfect” mode for sprites, maybe it should be added to sprite map too?

Yeah I think this would needed to be added, sounds like its the same thing.

Ill have to figure out how to apply that to the spriteMap though it seems like it might be a little tougher to do because of how the UV is being split up essentially for the atlas. It will just take a little bit of thought here.

2 Likes

Any update? Thanks

What would be a good way for me to edit/test this file? I have BabylonJS installed with npm i.

You will have to follow these instructions to be able to have a Babylon.js local working repo:

1 Like

Thank you for taking the initiative on this, I have a thousands fires to put out right now and this was put on the backlog for me.

I also don’t have the time but I will have to look at it in a few weeks. I’ll just be flipping switches if I do look at it.

Ohh, then I might end up getting to it first then! If you need some direction let me know, just PM me and ill guide you on the right path.

1 Like

In about 5 days I’ll be forced to learn how to dev this and fix it. Hoping the experts come through before then.

Here’s just layer 5 of a map rendering. See how it looks like a wave is moving through it?

Realized I could replace the shader at runtime. Reduced it to this so far:

#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.));
}

void main() {
  vec4 color = vec4(0.);
  vec2 tileUV = fract(tUV);
  vec2 tileID = floor(tUV);

  for (int i = 5; i < 6; i++) {
    float frameID = texture2D(tileMaps[5], (tileID + 0.5) / stageSize, 0.).x;
    /*
    #define LAYER_ID_SWITCH
    */
    mat4 frameData = getFrameData(frameID);
    vec2 frameSize = (vec2(32,32)) / spriteMapSize;
    vec2 offset = frameData[0].xy / spriteMapSize;

    vec4 nc = texture2D(spriteSheet, tileUV * frameSize + offset);

    if (i == 0) {
      color = nc;
    } else {
      float alpha = min(color.a + nc.a, 1.0);
      vec3 mixed = mix(color.xyz, nc.xyz, nc.a);
      color = vec4(mixed, alpha);
    }
  }

  gl_FragColor = color;
}

I’m starting to think it’s something to do with sampling due to there being no mipmaps as it changes with distance.

You can see it here if you click and drag.


Switch tile texture to linear sampling and it’s almost fixed. now there are lines around the edges of the tiles.

vec2 frameSize = vec2(28,28) / spriteMapSize;
vec2 offset = (frameData[0].xy + vec2(2,2)) * sheetUnits;

My tile width/height is 32.

Making the tile smaller and moving the offset gets rid of the gaps. This isn’t a solution but that’s what it is. sampling gaps around the edge of the tiles.

Is there any way to change vsync? I feel like it’s tearing.

It seems vsync is enabled by default in browser.

Try:

1 Like

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.

Repro!

Just need the seams fixed and it’s pixel perfect!

Actually, the sampler should be bilinear, or you should load the texture by disabling mipmaps (the sprite manager sets trilinear when in pixel perfect mode but because the texture is loaded without mipmaps in the constructor, the sampler is really bilinear).

It does not remove the seams, but helps a little for some of them.

If it can help, I made a repro with only two tiles:

1 Like