评论

收藏

Pixi.js 渲染 Tilemap

游戏开发 游戏开发 发布于:2021-07-18 10:13 | 阅读数:378 | 评论:0

源码 预览
游戏开发中或多或少都有接触过Tilemap,在Tiled编辑器里编辑好地图之后,导出数据,然后在游戏引擎(Cocos...)里就可以直接使用了,确实很方便。
由于Pixi.js自身并不支持Tilemap的解析渲染,所以我一直在尝试了解背后的机制。最初能想到的实现就是,在一个Container里放一堆Sprite把地图拼出来,Pixi.js的Texture支持以一个BaseTexture为基础,分块读取,所以实现一个也还可以。代码差不多就是这样:
const rect = new PIXI.Rectangle(0, 0, this.square, this.square)
for (let i = 0, sn = 0; i < this.size.height; i++) {
  for (let j = 0; j < this.size.width; j++) {
    sn = this.data[i * this.size.width + j]
    if (!sn) continue
    sn--
    rect.x = (sn % this.tilesets.columns) * this.square
    rect.y = (~~(sn / this.tilesets.columns)) * this.square
// 分块读取
    const tile = new PIXI.Sprite(new PIXI.Texture(this.tilesheet, rect))
    tile.position.set(j * this.square, i * this.square)
    tile.anchor.set(.5)
    this.addChild(tile)
  }
}
一天看到Pixi.js的作者在Codepen上的代码后,于是有了今天这篇文章。
实现思路
上面是v5的实现,v4实现起来略麻烦。
利用shader去渲染Tilemap。我们从Tilemap导出的JSON数据可以知道,主要的数据其实就是地图元素(瓦片)在图集中的索引。如何在着色器里拿到索引数据呢?
如果你看了上面Codepen的代码,或许你就知道了:
const bitmap = new PIXI.Graphics()
for (let i = 0; i < layer.height; i++) {
  for (let j = 0; j < layer.width; j++) {
    const
      index = layer.data[i * layer.width + j] - 1,
      column = this.mapData.tilesets[0].columns,
      x = index % column,
      y = Math.floor(index / column)
    bitmap.beginFill((x << 16) + (y << 8), index !== -1 ? 1 : 0)
    bitmap.drawRect(j, i, 1, 1)
    bitmap.endFill()
  }
}
// 生成纹理后面会传入纹理单元 一定要设置:PIXI.SCALE_MODES.NEAREST
core.renderer.generateTexture(bitmap, PIXI.SCALE_MODES.NEAREST)
创建一个bitmap,把地图的索引数据变成颜色值储存在这个bitmap里。
还需要一段着色器代码:
precision highp float;
uniform sampler2D tilesheet, bitmap;
uniform float tileSize, tileColumn;
uniform vec2 tilesheetSize, mapSize;
varying vec2 vTextureCoord, vVertexPosition;
void main() {
  vec4 color = texture2D(bitmap, vTextureCoord) * 255.0;
  if (color.a == 0.0) discard;
  vec2 coord = (vec2(color.r, color.g) * tileSize
    + mod(vTextureCoord * mapSize, tileSize)) / tilesheetSize;
  gl_FragColor = texture2D(tilesheet, coord);
}
这里还有一些Pixi.js的操作,我就不写出来了。弄明白原理,就行。感觉和法线贴图是一个道理,虽然我并没有研究过。主要就是把索引数据变成纹理上传到GPU,然后在着色器代码里读取出来。

关注下面的标签,发现更多相似文章