Search code examples
algorithmgraphics2dp5.jsimage-rotation

How to smoothly align rotated bitmaps side by side without jerkiness?


My current program draw a rotated bitmap (64x64) and tile it on the screen by drawing it again but adding an offset based on the computed position of the bitmap top right corner (after rotation), it works fine but i experience some jerkiness of the grid in motion.

The jerkiness doesn't appear if i do the same thing with canvas transforms.

Here is an example which compare both : https://editor.p5js.org/onirom/sketches/A5D-0nxBp

Move mouse to the left part of the canvas for the custom rotation algorithm and to the right part for the canvas one.

It seems that some tile are misplaced by a single pixel which result in the grid jerkiness.

custom rotation algorithm grid jerkiness vs canvas rotation

Is there a way to remove the grid jerkiness without doing it as a single pass and keeping the same interpolation scheme ?

Is it a sub pixels correctness issue ?

Here is some code :

let tileImage = null
function preload() {
  tileImage = loadImage('')
}

function setup() {
  createCanvas(512, 512)
  
  frameRate(14)
  
  tileImage.loadPixels()
}

function computeRotatedPoint(c, s, x, y) {
  return { x: x * c - y * s, y: x * s + y * c }
}

currentTileWidth = 0
currentTileHeight = 0

// draw a rotated bitmap at screen position ox, oy
function drawRotatedBitmap(c, s, ox, oy) {
  let dcu = s
  let dcv = c
  let dru = dcv
  let drv = -dcu
  let su = (tileImage.width / 2.0) - (currentTileWidth_d2 * dcv + currentTileHeight_d2 * dcu)
  let sv = (tileImage.height / 2.0) - (currentTileWidth_d2 * drv + currentTileHeight_d2 * dru)
  let ru = su
  let rv = sv

  for (let y = 0; y < currentTileHeight; y += 1) {
    let u = ru
    let v = rv
    for (let x = 0; x < currentTileWidth; x += 1) {
      let ui = u
      let vi = v
      
      if (ui >= 0 && ui < tileImage.width) {
        let index1 = (floor(ui) + floor(vi) * tileImage.width) * 4
        let index2 = (x + ox + (y + oy) * width) * 4
        pixels[index2 + 0] = tileImage.pixels[index1 + 0]
        pixels[index2 + 1] = tileImage.pixels[index1 + 1]
        pixels[index2 + 2] = tileImage.pixels[index1 + 2]
      }
      
      u += dru
      v += drv
    }
    ru += dcu
    rv += dcv
  }
}

let angle = 0

function draw() {
  background(0)

  const s = sin(angle / 256 * PI * 2)
  const c = cos(angle / 256 * PI * 2)

  // compute rotated tile width / height
  let tw = tileImage.width
  let th = tileImage.height
  if (angle % 128 < 64) {
    currentTileWidth = abs(tw * c + th * s)
    currentTileHeight = abs(tw * s + th * c)
  } else {
    currentTileWidth = abs(tw * c - th * s)
    currentTileHeight = abs(tw * s - th * c)
  }

  currentTileWidth_d2 = (currentTileWidth / 2.0)
  currentTileHeight_d2 = (currentTileHeight / 2.0)
  
  // compute rotated point
  const rp = computeRotatedPoint(c, s, tw, 0)

  // draw tiles
  loadPixels()
  for (let i = -3; i <= 3; i += 1) {
    // compute center
    const cx = width / 2 - currentTileWidth_d2
    const cy = height / 2 - currentTileHeight_d2
    // compute tile position
    const ox = rp.x * i
    const oy = rp.y * i
    drawRotatedBitmap(c, s, round(cx + ox), round(cy + oy))
  }
  updatePixels()
  
  angle += 0.5
}
<!DOCTYPE html>
<html lang="en">
  <head>
    <script src="https://cdnjs.cloudflare.com/ajax/libs/p5.js/1.5.0/p5.js"></script>
    <script src="https://cdnjs.cloudflare.com/ajax/libs/p5.js/1.5.0/addons/p5.sound.min.js"></script>
    <link rel="stylesheet" type="text/css" href="style.css">
    <meta charset="utf-8" />
  </head>
  <body>
    <main>
    </main>
  </body>
</html>


Solution

  • I have found a solution which does not use the same algorithm but use the same interpolation scheme.

    Solution with a three-shear method

    This solution use a three-pass shear method and the solution to fix the tiles jerkiness is to add the tile offset before the rotation and then round coordinates once everything is ready to be drawn :

    smooth three-shear rotation tiling

    /**
     * Bitmap rotation + stable tiling with 3-shearing method
     * The 3-shearing method is stable between -PI / 2 and PI / 2 only, that is why a flip is needed for a full rotation
     *
     * https://www.ocf.berkeley.edu/~fricke/projects/israel/paeth/rotation_by_shearing.html
     */
    let tex = null
    let tile = []
    function preload() {
      tile = loadImage('')
    
    }
    
    function setup() {
      createCanvas(512, 512)
      
      frameRate(12)
    }
    
    function computeRotatedPoint(c, s, x, y) {
      return { x: x * c - y * s, y: x * s + y * c }
    }
    
    function _shearX(t, x, y) {
      return x - y * t
    }
    
    function _shearY(s, x, y) {
      return x * s + y
    }
    
    currentTileWidth = 0
    currentTileHeight = 0
    
    currentTileLookupFunction = tileLookup1
    
    // regular lookup
    function tileLookup1(x, y) {
      return (x + y * tile.width) * 4
    }
    
    // flipped lookup
    function tileLookup2(x, y) {
      return ((tile.width - 1 - x) + (tile.height - 1 - y) * tile.width) * 4
    }
    
    // draw a rotated bitmap at offset ox,oy with cx,cy as center of rotation offset
    function drawRotatedBitmap(c, s, t, ox, oy, cx, cy) {
      for (let ty = 0; ty < tile.height; ty += 1) {
        for (let tx = 0; tx < tile.width; tx += 1) {
          // center of rotation
          let scx = tile.width - tx - tile.width / 2
          let scy = tile.height - ty - tile.height / 2
          
          // this is key to a stable rotation without any jerkiness
          scx += cx
          scy += cy
          
          // shear
          let ux = round(_shearX(t, scx, scy))
          let uy = round(_shearY(s, ux, scy))
          ux = round(_shearX(t, ux, uy))
          
          // translate again
          ux = currentTileWidth_d2 - ux
          uy = currentTileHeight_d2 - uy
    
          // plot with offset
          let index1 = currentTileLookupFunction(tx, ty)
          let index2 = (ox + ux + (oy + uy) * width) * 4
    
          pixels[index2 + 0] = tile.pixels[index1 + 0]
          pixels[index2 + 1] = tile.pixels[index1 + 1]
          pixels[index2 + 2] = tile.pixels[index1 + 2]
        }
      }
    }
    
    let angle = -3.141592653 / 2
    function draw() {
      const s = sin(angle)
      const c = cos(angle)
      const t = tan(angle / 2)
    
      tile.loadPixels()
    
      background(0)
    
      // compute rotated tile width / height
      let tw = tile.width
      let th = tile.height
    
      currentTileWidth = abs(tw * c + th * s)
      currentTileHeight = abs(tw * s + th * c)
    
      currentTileWidth_d2 = round(currentTileWidth / 2.0)
      currentTileHeight_d2 = round(currentTileHeight / 2.0)
    
      // draw tiles
      loadPixels()
      for (let j = -2; j <= 2; j += 1) {
        for (let i = -2; i <= 2; i += 1) {
          let ox = round(width / 2 - currentTileWidth_d2)
          let oy = round(height / 2 - currentTileHeight_d2)
          drawRotatedBitmap(c, s, t, ox, oy, i * tw, j * tw)
        }
      }
      updatePixels()
      
      angle += 0.025
      if (angle >= PI / 2) {
        angle -= PI
        
        if (currentTileLookupFunction === tileLookup2) {
          currentTileLookupFunction = tileLookup1
        } else {
          currentTileLookupFunction = tileLookup2
        }
      }
    }
    <!DOCTYPE html>
    <html lang="en">
      <head>
        <script src="https://cdnjs.cloudflare.com/ajax/libs/p5.js/1.5.0/p5.js"></script>
        <script src="https://cdnjs.cloudflare.com/ajax/libs/p5.js/1.5.0/addons/p5.sound.min.js"></script>
        <link rel="stylesheet" type="text/css" href="style.css">
        <meta charset="utf-8" />
      </head>
      <body>
        <main>
        </main>
      </body>
    </html>


    I cannot say technically why it works but it is probably related to an error accumulation issue / rounding since i can reproduce the question issue completely with the three-shear method if i add the tile offset after rotation and round the offset and shear pass independently such as :

    function drawRotatedBitmap(c, s, t, ox, oy, cx, cy) {
      cx = round(_shearX(t, cx, cy))
      cy = round(_shearY(s, cx, cy))
      cx = round(_shearX(t, cx, cy))
      for (let ty = 0; ty < tex.height; ty += 1) {
        ...
        let ux = round(_shearX(t, scx, scy))
        let uy = round(_shearY(s, ux, scy))
        ux = round(_shearX(t, ux, uy))
        ...
        let index2 = (cx + ux + ox + (cy + uy + oy) * width) * 4
        ...
      }
    }
    

    three-shear rotation tiling with tiles jerkiness


    The issue become clearly visible if i round the offset and the shearing result at the same time which result in missing pixels in the final image such as :

    function drawRotatedBitmap(c, s, t, ox, oy, cx, cy) {
      cx = _shearX(t, cx, cy)
      cy = _shearY(s, cx, cy)
      cx = _shearX(t, cx, cy)
      for (let ty = 0; ty < tex.height; ty += 1) {
        ...
        let ux = _shearX(t, scx, scy)
        let uy = _shearY(s, ux, scy)
        ux = _shearX(t, ux, uy)
        ...
        let index2 = (round(cx + ux) + ox + (round(cy + uy) + oy) * width) * 4
        ...
      }
    }
    

    three-shear rotation tiling with holes


    I would still like a detailed explanation of the jerkiness behavior and to know if there is a smooth solution by adding the tile offset after the rotation, it seems that the jiggling is due to the center of rotation being off one or two pixels depending on the angle.