fixed dynamic layer rendering

This commit is contained in:
Svipal 2020-02-04 04:57:28 +01:00
parent fd29e96a5d
commit a8b60cf946
10 changed files with 4348 additions and 3913 deletions

File diff suppressed because it is too large Load diff

File diff suppressed because one or more lines are too long

4047
dist/phaser.js vendored

File diff suppressed because it is too large Load diff

2
dist/phaser.min.js vendored

File diff suppressed because one or more lines are too long

View file

@ -22,6 +22,7 @@ var SnapCeil = require('../../math/snap/SnapCeil');
*/ */
var CullTiles = function (layer, camera, outputArray, renderOrder) var CullTiles = function (layer, camera, outputArray, renderOrder)
{ {
if (outputArray === undefined) { outputArray = []; } if (outputArray === undefined) { outputArray = []; }
if (renderOrder === undefined) { renderOrder = 0; } if (renderOrder === undefined) { renderOrder = 0; }
@ -44,104 +45,201 @@ var CullTiles = function (layer, camera, outputArray, renderOrder)
var drawRight = mapWidth; var drawRight = mapWidth;
var drawTop = 0; var drawTop = 0;
var drawBottom = mapHeight; var drawBottom = mapHeight;
var inIsoBounds = function (x,y) { return true;}
if (!tilemapLayer.skipCull && tilemapLayer.scrollFactorX === 1 && tilemapLayer.scrollFactorY === 1) {
if (layer.orientation == "orthogonal") {
// Camera world view bounds, snapped for scaled tile size
// Cull Padding values are given in tiles, not pixels
if (!tilemapLayer.skipCull && tilemapLayer.scrollFactorX === 1 && tilemapLayer.scrollFactorY === 1) var boundsLeft = SnapFloor(camera.worldView.x - tilemapLayer.x, tileW, 0, true) - tilemapLayer.cullPaddingX;
{ var boundsRight = SnapCeil(camera.worldView.right - tilemapLayer.x, tileW, 0, true) + tilemapLayer.cullPaddingX;
// Camera world view bounds, snapped for scaled tile size var boundsTop = SnapFloor(camera.worldView.y - tilemapLayer.y, tileH, 0, true) - tilemapLayer.cullPaddingY;
// Cull Padding values are given in tiles, not pixels var boundsBottom = SnapCeil(camera.worldView.bottom - tilemapLayer.y, tileH, 0, true) + tilemapLayer.cullPaddingY;
var boundsLeft = SnapFloor(camera.worldView.x - tilemapLayer.x, tileW, 0, true) - tilemapLayer.cullPaddingX; drawLeft = Math.max(0, boundsLeft);
var boundsRight = SnapCeil(camera.worldView.right - tilemapLayer.x, tileW, 0, true) + tilemapLayer.cullPaddingX; drawRight = Math.min(mapWidth, boundsRight);
var boundsTop = SnapFloor(camera.worldView.y - tilemapLayer.y, tileH, 0, true) - tilemapLayer.cullPaddingY; drawTop = Math.max(0, boundsTop);
var boundsBottom = SnapCeil(camera.worldView.bottom - tilemapLayer.y, tileH, 0, true) + tilemapLayer.cullPaddingY; drawBottom = Math.min(mapHeight, boundsBottom);
} else if (layer.orientation == "isometric") {
drawLeft = Math.max(0, boundsLeft); inIsoBounds = function (x,y){
drawRight = Math.min(mapWidth, boundsRight); var pos = tilemapLayer.tileToWorldXY(x,y,undefined,camera)
drawTop = Math.max(0, boundsTop); return (pos.x > camera.worldView.x && pos.x < camera.worldView.right) && (pos.y > camera.worldView.y && pos.y < camera.worldView.bottom)
drawBottom = Math.min(mapHeight, boundsBottom); }
}
} }
var x; var x;
var y; var y;
var tile; var tile;
if (renderOrder === 0)
{
// right-down
for (y = drawTop; y < drawBottom; y++) if (layer.orientation == "orthogonal") {
if (renderOrder === 0)
{ {
for (x = drawLeft; mapData[y] && x < drawRight; x++) // right-down
for (y = drawTop; y < drawBottom; y++)
{ {
tile = mapData[y][x]; for (x = drawLeft; mapData[y] && x < drawRight; x++)
if (!tile || tile.index === -1 || !tile.visible || tile.alpha === 0)
{ {
continue; tile = mapData[y][x];
}
outputArray.push(tile); if (!tile || tile.index === -1 || !tile.visible || tile.alpha === 0)
{
continue;
}
outputArray.push(tile);
}
} }
} }
} else if (renderOrder === 1)
else if (renderOrder === 1)
{
// left-down
for (y = drawTop; y < drawBottom; y++)
{ {
for (x = drawRight; mapData[y] && x >= drawLeft; x--) // left-down
for (y = drawTop; y < drawBottom; y++)
{ {
tile = mapData[y][x]; for (x = drawRight; mapData[y] && x >= drawLeft; x--)
if (!tile || tile.index === -1 || !tile.visible || tile.alpha === 0)
{ {
continue; tile = mapData[y][x];
}
outputArray.push(tile); if (!tile || tile.index === -1 || !tile.visible || tile.alpha === 0)
{
continue;
}
outputArray.push(tile);
}
} }
} }
} else if (renderOrder === 2)
else if (renderOrder === 2)
{
// right-up
for (y = drawBottom; y >= drawTop; y--)
{ {
for (x = drawLeft; mapData[y] && x < drawRight; x++) // right-up
for (y = drawBottom; y >= drawTop; y--)
{ {
tile = mapData[y][x]; for (x = drawLeft; mapData[y] && x < drawRight; x++)
if (!tile || tile.index === -1 || !tile.visible || tile.alpha === 0)
{ {
continue; tile = mapData[y][x];
}
outputArray.push(tile); if (!tile || tile.index === -1 || !tile.visible || tile.alpha === 0)
{
continue;
}
outputArray.push(tile);
}
} }
} }
} else if (renderOrder === 3)
else if (renderOrder === 3)
{
// left-up
for (y = drawBottom; y >= drawTop; y--)
{ {
for (x = drawRight; mapData[y] && x >= drawLeft; x--) // left-up
for (y = drawBottom; y >= drawTop; y--)
{ {
tile = mapData[y][x]; for (x = drawRight; mapData[y] && x >= drawLeft; x--)
if (!tile || tile.index === -1 || !tile.visible || tile.alpha === 0)
{ {
continue; tile = mapData[y][x];
}
outputArray.push(tile); if (!tile || tile.index === -1 || !tile.visible || tile.alpha === 0)
{
continue;
}
outputArray.push(tile);
}
} }
} }
} else if (layer.orientation == "isometric") {
if (renderOrder === 0)
{
// right-down
for (y = drawTop; y < drawBottom; y++)
{
for (x = drawLeft; mapData[y] && x < drawRight; x++)
{
if (inIsoBounds(x,y)) {
tile = mapData[y][x];
if (!tile || tile.index === -1 || !tile.visible || tile.alpha === 0)
{
continue;
}
outputArray.push(tile);
}
}
}
}
else if (renderOrder === 1)
{
// left-down
for (y = drawTop; y < drawBottom; y++)
{
for (x = drawRight; mapData[y] && x >= drawLeft; x--)
{
if (inIsoBounds(x,y)) {
tile = mapData[y][x];
if (!tile || tile.index === -1 || !tile.visible || tile.alpha === 0)
{
continue;
}
outputArray.push(tile);
}
}
}
}
else if (renderOrder === 2)
{
// right-up
for (y = drawBottom; y >= drawTop; y--)
{
for (x = drawLeft; mapData[y] && x < drawRight; x++)
{
if (inIsoBounds(x,y)) {
tile = mapData[y][x];
if (!tile || tile.index === -1 || !tile.visible || tile.alpha === 0)
{
continue;
}
outputArray.push(tile);
}
}
}
}
else if (renderOrder === 3)
{
// left-up
for (y = drawBottom; y >= drawTop; y--)
{
for (x = drawRight; mapData[y] && x >= drawLeft; x--)
{
if (inIsoBounds(x,y)) {
tile = mapData[y][x];
if (!tile || tile.index === -1 || !tile.visible || tile.alpha === 0)
{
continue;
}
outputArray.push(tile);
}
}
}
}
} }
tilemapLayer.tilesDrawn = outputArray.length; tilemapLayer.tilesDrawn = outputArray.length;
tilemapLayer.tilesTotal = mapWidth * mapHeight; tilemapLayer.tilesTotal = mapWidth * mapHeight;

View file

@ -7,6 +7,7 @@
var GetTilesWithin = require('./GetTilesWithin'); var GetTilesWithin = require('./GetTilesWithin');
var WorldToTileX = require('./WorldToTileX'); var WorldToTileX = require('./WorldToTileX');
var WorldToTileY = require('./WorldToTileY'); var WorldToTileY = require('./WorldToTileY');
var WorldToTileXY = require('./WorldToTileXY');
/** /**
* Gets the tiles in the given rectangular area (in world coordinates) of the layer. * Gets the tiles in the given rectangular area (in world coordinates) of the layer.

View file

@ -30,15 +30,35 @@ var TileToWorldXY = function (tileX, tileY, point, camera, layer)
var orientation = layer.orientation; var orientation = layer.orientation;
var tileWidth = layer.baseTileWidth; var tileWidth = layer.baseTileWidth;
var tileHeight = layer.baseTileHeight; var tileHeight = layer.baseTileHeight;
var tilemapLayer = layer.tilemapLayer;
if (point === undefined) { point = new Vector2(0, 0); } if (point === undefined) { point = new Vector2(0, 0); }
if (orientation === "orthogonal") { if (orientation === "orthogonal") {
point.x = TileToWorldX(tileX, camera, layer, orientation); point.x = TileToWorldX(tileX, camera, layer, orientation);
point.y = TileToWorldY(tileY, camera, layer, orientation); point.y = TileToWorldY(tileY, camera, layer, orientation);
} else if (orientation === "isometric") { } else if (orientation === "isometric") {
point.x = (tileX - tileY) * (tileWidth/2);
point.y = (tileX + tileY) * (tileHeight/2); var layerWorldX = 0;
var layerWorldY = 0;
if (tilemapLayer)
{
if (camera === undefined) { camera = tilemapLayer.scene.cameras.main; }
layerWorldX = tilemapLayer.x + camera.scrollX * (1 - tilemapLayer.scrollFactorX);
tileWidth *= tilemapLayer.scaleX;
layerWorldY = (tilemapLayer.y + camera.scrollY * (1 - tilemapLayer.scrollFactorY));
tileHeight *= tilemapLayer.scaleY;
}
point.x = layerWorldX + (tileX - tileY) * (tileWidth/2);
point.y = layerWorldY + (tileX + tileY) * (tileHeight/2);
} }

View file

@ -46,8 +46,9 @@ var WorldToTileXY = function (worldX, worldY, snapToFloor, point, camera, layer)
// Find the world position relative to the static or dynamic layer's top left origin, // Find the world position relative to the static or dynamic layer's top left origin,
// factoring in the camera's vertical scroll // factoring in the camera's vertical scroll
// console.log(1,worldY)
worldY = worldY - (tilemapLayer.y + camera.scrollY * (1 - tilemapLayer.scrollFactorY)); worldY = worldY - (tilemapLayer.y + camera.scrollY * (1 - tilemapLayer.scrollFactorY));
// console.log(worldY)
tileHeight *= tilemapLayer.scaleY; tileHeight *= tilemapLayer.scaleY;
// Find the world position relative to the static or dynamic layer's top left origin, // Find the world position relative to the static or dynamic layer's top left origin,
@ -62,8 +63,8 @@ var WorldToTileXY = function (worldX, worldY, snapToFloor, point, camera, layer)
: ((worldX/(tileWidth/2) + worldY/(tileHeight/2))/2); : ((worldX/(tileWidth/2) + worldY/(tileHeight/2))/2);
point.y = snapToFloor point.y = snapToFloor
? Math.floor(worldY/(tileHeight/2) - (worldX/(tileWidth/2))/2) ? Math.floor((worldY/(tileHeight/2) - worldX/(tileWidth/2))/2)
: (worldY/(tileHeight/2) - (worldX/(tileWidth/2))/2); : ((worldY/(tileHeight/2) - worldX/(tileWidth/2))/2);
} }

View file

@ -89,8 +89,19 @@ var DynamicTilemapLayerCanvasRenderer = function (renderer, src, interpolationPe
if (tileTexCoords) if (tileTexCoords)
{ {
var halfWidth = tile.width / 2; var width = 0;
var halfHeight = tile.height / 2; var height = 0;
if (src.layer.orientation === "isometric") {
// here we use the tileset width and height to fix problems with isometric map types
width = tileset.tileWidth;
width = tileset.tileHeight;
} else {
width = tile.width;
height = tile.width
}
halfWidth = width / 2;
halfHeight = height / 2;
ctx.save(); ctx.save();
@ -111,9 +122,9 @@ var DynamicTilemapLayerCanvasRenderer = function (renderer, src, interpolationPe
ctx.drawImage( ctx.drawImage(
image, image,
tileTexCoords.x, tileTexCoords.y, tileTexCoords.x, tileTexCoords.y,
tile.width, tile.height, width, height,
-halfWidth, -halfHeight, -halfWidth, -halfHeight,
tile.width, tile.height width, height
); );
ctx.restore(); ctx.restore();

View file

@ -74,15 +74,29 @@ var DynamicTilemapLayerWebGLRenderer = function (renderer, src, interpolationPer
{ {
continue; continue;
} }
if (src.layer.orientation === "isometric") {
// here we use the tileset width and height to fix problems with isometric map types
var frameWidth = tile.width; var frameWidth = tileset.tileWidth;
var frameHeight = tile.height; var frameHeight = tileset.tileHeight;
var frameX = tileTexCoords.x;
var frameY = tileTexCoords.y;
var tw = tileset.tileWidth * 0.5;
var th = tileset.tileHeight * 0.5;
} else {
var frameWidth = tile.width;
var frameHeight = tile.height;
var frameX = tileTexCoords.x;
var frameY = tileTexCoords.y;
var tw = tile.width * 0.5;
var th = tile.height * 0.5;
}
var frameX = tileTexCoords.x;
var frameY = tileTexCoords.y;
var tw = tile.width * 0.5;
var th = tile.height * 0.5;
var tint = getTint(tile.tint, alpha * tile.alpha); var tint = getTint(tile.tint, alpha * tile.alpha);
@ -91,7 +105,7 @@ var DynamicTilemapLayerWebGLRenderer = function (renderer, src, interpolationPer
texture, texture,
texture.width, texture.height, texture.width, texture.height,
x + ((tw + tile.pixelX) * sx), y + ((th + tile.pixelY) * sy), x + ((tw + tile.pixelX) * sx), y + ((th + tile.pixelY) * sy),
tile.width, tile.height, frameWidth, frameHeight,
sx, sy, sx, sy,
tile.rotation, tile.rotation,
tile.flipX, tile.flipY, tile.flipX, tile.flipY,