最新消息:雨落星辰是一个专注网站SEO优化、网站SEO诊断、搜索引擎研究、网络营销推广、网站策划运营及站长类的自媒体原创博客

javascript - Converting screen coordinates to isometric map coordinates - Stack Overflow

programmeradmin1浏览0评论

I'm generating an isometric tile map using a diamond pattern:

tileWidth = 128;
tileHeight = 94;

for (var x = 0; x < rows; x++) {
  for (var y = 0; y < cols; y++) {
    var screenX = (x - y) * tileWidthHalf;
    var screenY = (x + y) * tileHeightHalf;
    drawTile(screenX, screenY);
  }
}

This renders correctly, but now I'm having trouble converting screen coordinates (mouse location) back to the isometric coordinates.

I've tried reversing the math:

var x = _.floor(screenY / (tileWidth / 2) - (screenX / tileWidth / 2));
var y = _.floor(screenY / (tileHeight / 2) + (screenX / tileHeight / 2));

It works fine for the 0, 0 tile but fails to produce the right value afterwards.

I'm just unable to e up with the right math - am I missing something trivial or am I just all wrong about the process?

I'm generating an isometric tile map using a diamond pattern:

tileWidth = 128;
tileHeight = 94;

for (var x = 0; x < rows; x++) {
  for (var y = 0; y < cols; y++) {
    var screenX = (x - y) * tileWidthHalf;
    var screenY = (x + y) * tileHeightHalf;
    drawTile(screenX, screenY);
  }
}

This renders correctly, but now I'm having trouble converting screen coordinates (mouse location) back to the isometric coordinates.

I've tried reversing the math:

var x = _.floor(screenY / (tileWidth / 2) - (screenX / tileWidth / 2));
var y = _.floor(screenY / (tileHeight / 2) + (screenX / tileHeight / 2));

It works fine for the 0, 0 tile but fails to produce the right value afterwards.

I'm just unable to e up with the right math - am I missing something trivial or am I just all wrong about the process?

Share Improve this question edited Sep 27, 2016 at 19:01 helion3 asked Sep 27, 2016 at 16:26 helion3helion3 37.5k16 gold badges64 silver badges111 bronze badges 2
  • Future visitors: Nico's solution (the accepted solution) is the only one that works for me out of many different solutions found on the internet. I am following the article clintbellanger/articles/isometric_math - You MUST substitute screenX with (screenX - tileWidthHalf) as suggested in Nico's answer. See my ment in Nico's answer. Be sure to floor the results. – shell Commented Nov 5, 2019 at 18:39
  • Nico's solution is correct, but remember you must use FLOATS for screenx, screeny, tileWidthHalf, and tileHeightHalf or there is not enough precision. I was using ints and it did not work correctly. Changed these to floats and it works perfectly. – chewinggum Commented Apr 13, 2020 at 5:27
Add a ment  | 

2 Answers 2

Reset to default 8

I don't see how you came up with this solution. You have to solve the system of equation, which gives the following solution:

x = 0.5 * ( screenX / tileWidthHalf + screenY / tileHeightHalf)
y = 0.5 * (-screenX / tileWidthHalf + screenY / tileHeightHalf)

If you need the tile index, use floor as in your code.

I can only guess what your alignment of the tiles in the coordinate system looks like. But from the screenshot you posted in the ments, I assume that you need to swap screenX with (screenX - tileWidthHalf) to get accurate values.

I'm writing w for your tileWidth and h for tileHeight. Using this notation, you essentially have

screenX = (x - y)*(w/2) = (w/2)*x + (-w/2)*y
screenY = (x + y)*(h/2) = (h/2)*x + (h/2)*y

This is a linear transformation which you can also write in matrix notation like this:

⎛x⎞ ↦ ⎛w/2 -w/2⎞ ⎛x⎞
⎝y⎠   ⎝h/2  h/2⎠ ⎝y⎠

To reverse the operation you need to invert this. There is a simple formula for the inverse of a 2×2 matrix. Swap the top left and bottom right entries. Negate the other two. Divide everything by the determinant. Use that and you have the inverse transformation:

⎛x⎞ ↦ 2/ ⎛ h/2 w/2⎞ ⎛x⎞ = ⎛ 1/w 1/h⎞ ⎛x⎞
⎝y⎠   wh ⎝-h/2 w/2⎠ ⎝y⎠   ⎝-1/w 1/h⎠ ⎝y⎠

So in your notation you get

x = screenY / tileHeight + screenX / tileWidth
y = screenY / tileHeight - screenX / tileWidth

which is essentially what Nico wrote as well. Turning non-integer coordinates back to integers depends on where you place the reference point for each tile.

发布评论

评论列表(0)

  1. 暂无评论