Arce 发表于 2021-7-18 10:13:05

Pixi.js 渲染 Tilemap

源码 预览
游戏开发中或多或少都有接触过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
      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 - 1,
            column = this.mapData.tilesets.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,然后在着色器代码里读取出来。

文档来源:开源中国社区https://my.oschina.net/u/945111/blog/1840237
页: [1]
查看完整版本: Pixi.js 渲染 Tilemap