SNES Mosaic effect

Today we look into implementing a real classic effect which gained popularity by it’s usage in Nintendo’s famous Super Mario World on the SNES.

It’s called pixelation – and is surely more known by it’s colloquial term mosaic. As the name implies it transitions an image from a mosaic-like bunch of huge pixels which decrease in size until the source image is revealed.
Here’s an impression as an animated gif:
mario
The programming language for today will be JavaScript. You will be surprised – though that surely wasn’t the original intention – the canvas object itself offers a method to fake this effect.
We can abuse the .drawImage() method to draw a shrinked version of the target image at the target image’s dimensions!

So let’s warm up! Grab the following image and save it on your harddrive as marioStill.png.

marioStill

Now use this html template and save it as mario.html.

First we need to load the actual image file. This is done by creating a new Image element.

After loading finished it will invoke the pixelate function we’ll work on next.

The parameter e of this function is a reference to the element which fired the load event – our image. To get it’s content we need to call e.target
Let’s draw the image to a canvas element the size of the image.

Nothing too fancy happened yet. We created a canvas element, made it the size of the image, draw the image onto and finally appended the canvas to the DOM.
If you run this inside the browser it’s just like we simply loaded the image directly.

We need to draw a scaled down version of the image to an invisible buffer canvas and ultimately draw this onto the ‘real’ canas at the original dimensions.
For now remove the line

add the following lines right after document.body.appendChild(canvas);

and insert the following new function:

Preview this in your browser and you’ll see something like this:

marioBlurry

Not really looking pixelated in any way, huh? In fact that looks more like a blurry mess! Well this is caused by the browser trying to smooth the image.
We need to disable these filters!
Right below var context = canvas.getContext(“2d”); add these lines

and preview it in the browser again:

marioBlocky

Muuuch better! We got some decent blocks now but what’s actually determining the size of those? Yeah – it’s: var percentage = 0.05; we’ve initialized at the beginning. This makes the little invisible buffer representation of the image a percentage of the original image.
Now imagine – if we dynamically change this number from 0.05 to 1 in small steps and call the draw function periodically using setTimeout() we’re done!

Here’s the final output:

 

Leave a Reply

Your email address will not be published. Required fields are marked *