Skip to content

Latest commit

 

History

History
55 lines (31 loc) · 6.1 KB

Readme.md

File metadata and controls

55 lines (31 loc) · 6.1 KB

Non-Pixel-Perfect Tilemap Rendering

This project countains the sources for new nodes and accompanying shaders that improve the visual clarity of tilemaps when used in non-pixel perfect projects. The nodes were originally created for pixel art games that didn't use pixel-perfect alignment, but it should provide some benefits for tilemaps with higher resolution art as well.

This tool is still a work in progress, and while it is mostly finished, there are still a few improvements I'm working and likely some bugs to find and fix still.

The source code for the node and helper methods are written in C#.

Examples

When rendering a tilemap that is aligned to the screen grid and has an integer scaling, this method should look no different than the default. The differences should only be noticeable when the tilemap is not perfectly aligned to the screen or is scaled/rotated.

Example video with the original tilemap (left) and mesh node (right) inside of a subviewport. The nodes are scaled by an integer number (3) and are slowly moved in a circular motion. As the default tilemap moves, the entire tilemap snaps as which texels are nearest to a given fragment changes over time. There is occasional flickering and texture bleeding as a result of the sub-pixel movement. The mesh render node doesn't use nearest sampling and so the tilemap moves smoothly along the path of motion, and the shader algorithm used prevents color bleed from neighboring tiles on the atlas texture.

integer_scaled.mp4

The same scene, but using a non-integer scale (2.9). Since the amount of pixels inside of a single tile is no longer uniform across the entire tilemap, the tiles flicker as they move. This is a result of the amount of fragments that map to a given texel changes over time, causing that texel to appear to grow and shrink depending on its screen position.

non-integer_scaled.mp4

The example scenes in these videos are included in the repository.

Specifics

The following list is all the differences between this and the default tilemap rendering method.

  • Fragments are checked if they lie on the border of multiple tiles or the inner region of a single tile before sampling.
  • Fragments inside a tile contained inside a single texel will be that texels color, otherwise it will be the linear blend of all texels it overlaps.
    • The algorithm used for modifying UV coordinates is slightly modified from the work of CptPotato. His original algorithm can be found here.
  • Fragments on the border of multiple tiles are a linear blend of the texels from each tile they overlap.
  • Fragments that sample from texture regions with entirely transparent texels use an average of the color channels of all non-blank texels when linearly filtering.
    • Godot has a texture import setting "Fix Alpha Border" that fixes this on non-atlas textures, but it produces visual artifacts on atlas textures due to lacking knowledge about which texture regions belong to which tile. This can result in colors bleeding in from neighboring tile regions.
  • If you choose to use mipmaps, the provided compute shader generates mipmaps for each tile in the atlas while ensuring it does not sampling outside a given tile's region, preventing color bleeding.

When using these nodes, ensure that the tileset textures are imported with "Fix Alpha Border" off and use the compute shader to generate mipmaps if they are desired. Additionally, the shaders expect the texture origin to be in the top-left-most portion of tiles that happen to span more than a 1x1 area on the tileset atlas.

Files and Classes

The relevant files are in the source directory. There are four C# classes for the nodes and a helper class, and two shaders.

The TilemapDisplay node is an abstract class implemented by TilemapMeshDisplay and TilemapCanvasItemDisplay. These nodes have a TileMapLayer property which tells them what to display. They do not replace the original tilemap, they only render a copy of it. I recommend assigning the TileMapLayer node as a child of the display node and disabling its visibility. This way you can still take advantage of the physics and other properties of the tilemap while using the mesh-based rendering.

The TilemapMeshDisplay node draws the tilemap using a multimesh, with each mesh instance representing one tile. Because multimeshes are drawn as a single primitive, they do not work as expected when Y-sort is enabled since the entire tilemap is a single canvas item. TilemapCanvasItemDisplay instead draws each tile as its own individual canvas item using the rendering server. Use this node if you need Y-sort enabled on the tilemap.

The TilemapDisplay.gdshader shader is used by the display nodes. The node will automatically create and set a material with this shader based on the properties of the tilemap and display nodes.

The GenerateCustomTilesetMipmaps.comp is a compute shader that generates tileset mipmaps while ensuring the different tile atlas regions do not bleed into each other. If you want to use mipmaps, you should generate them with this shader.

The TilesetHelper class is a utility class with a few exposed public methods for taking in a TileSetAtlasSource and computing a new texture with mipmaps from the source's original texture. In order to use this, you must first set up the atlas source and properly define all of the tiles in the atlas texture. The shader only generates mipmaps for defined tiles, and needs to know where each tile is in order to avoid accidentally sampling from neighboring tile regions in order to prevent bleeding.

License

The source code in this repository is under the Apache 2.0 License. See LICENSE and NOTICE for additional details.

Additionally, the example scenes in this repository use a tileset image (tileset/Standard Dungeon.png) originally authored by 0x72 under the CC-0 license.