I'm recreating Mrs Pacman using Lua, and just recently learned a series of tiles are used for collision and movement; I'm trying to recreate that. Here's the tile-map:
nodemap = {
{1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1},
{1,1,0,0,0,0,0,0,1,1,0,0,0,0,0,0,0,0,0,0,1,1,0,0,0,0,0,0,1,1},
{1,1,0,1,1,1,1,0,1,1,0,1,1,1,1,1,1,1,1,0,1,1,0,1,1,1,1,0,1,1},
{1,1,0,1,1,1,1,0,1,1,0,1,1,1,1,1,1,1,1,0,1,1,0,1,1,1,1,0,1,1},
{1,1,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,1,1},
{1,1,1,1,0,1,1,0,1,1,1,1,1,0,1,1,0,1,1,1,1,1,0,1,1,0,1,1,1,1},
{1,1,1,1,0,1,1,0,1,1,1,1,1,0,1,1,0,1,1,1,1,1,0,1,1,0,1,1,1,1},
{1,1,1,1,0,1,1,0,1,1,1,1,1,0,1,1,0,1,1,1,1,1,0,1,1,0,1,1,1,1},
{1,0,0,0,0,1,1,0,0,0,0,0,0,0,1,1,0,0,0,0,0,0,0,1,1,0,0,0,0,1},
{1,1,1,1,0,1,1,1,1,1,0,1,1,1,1,1,1,1,1,0,1,1,1,1,1,0,1,1,1,1},
{1,1,1,1,0,1,1,1,1,1,0,1,1,1,1,1,1,1,1,0,1,1,1,1,1,0,1,1,1,1},
{1,1,1,1,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,1,1,1,1},
{1,1,1,1,0,1,1,1,1,1,0,1,1,1,0,0,1,1,1,0,1,1,1,1,1,0,1,1,1,1},
{1,1,1,1,0,1,1,1,1,1,0,1,0,0,0,0,0,0,1,0,1,1,1,1,1,0,1,1,1,1},
{1,1,1,1,0,1,1,0,0,0,0,1,0,0,0,0,0,0,1,0,0,0,0,1,1,0,1,1,1,1},
{1,1,1,1,0,1,1,0,1,1,0,1,0,0,0,0,0,0,1,0,1,1,0,1,1,0,1,1,1,1},
{1,1,1,1,0,1,1,0,1,1,0,1,1,1,1,1,1,1,1,0,1,1,0,1,1,0,1,1,1,1},
{1,0,0,0,0,0,0,0,1,1,0,0,0,0,0,0,0,0,0,0,1,1,0,0,0,0,0,0,0,1},
{1,1,1,1,0,1,1,1,1,1,1,1,1,0,1,1,0,1,1,1,1,1,1,1,1,0,1,1,1,1},
{1,1,1,1,0,1,1,1,1,1,1,1,1,0,1,1,0,1,1,1,1,1,1,1,1,0,1,1,1,1},
{1,1,1,1,0,0,0,0,0,0,0,0,0,0,1,1,0,0,0,0,0,0,0,0,0,0,1,1,1,1},
{1,1,1,1,0,1,1,1,1,1,0,1,1,1,1,1,1,1,1,0,1,1,1,1,1,0,1,1,1,1},
{1,1,1,1,0,1,1,1,1,1,0,1,1,1,1,1,1,1,1,0,1,1,1,1,1,0,1,1,1,1},
{1,1,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,1,1},
{1,1,0,1,1,1,1,0,1,1,1,1,1,0,1,1,0,1,1,1,1,1,0,1,1,1,1,0,1,1},
{1,1,0,1,1,1,1,0,1,1,1,1,1,0,1,1,0,1,1,1,1,1,0,1,1,1,1,0,1,1},
{1,1,0,1,1,1,1,0,1,1,0,0,0,0,1,1,0,0,0,0,1,1,0,1,1,1,1,0,1,1},
{1,1,0,1,1,1,1,0,1,1,0,1,1,1,1,1,1,1,1,0,1,1,0,1,1,1,1,0,1,1},
{1,1,0,1,1,1,1,0,1,1,0,1,1,1,1,1,1,1,1,0,1,1,0,1,1,1,1,0,1,1},
{1,1,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,1,1},
{1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1}
}
This creates the maze without error, but the issue I'm having is that Pacman moves smoothly across tiles, without jumping from tile, to tile, to tile... So I have this code to "smoothly" move Pacman:
-- Make Pacmans position consist of a single decimal (because over time a number of 1.899999994 will occur and that's ugly)
mrspacman.x = math.floor(mrspacman.x*10)/10
mrspacman.y = math.floor(mrspacman.y*10)/10
--If an arrowkey was pressed (Think of this as "A new direction was queued")
if (mrspacman.nextDirection) then
-- If Pacman is in the center of a tile, then
if (mrspacman.x == math.floor(mrspacman.x)) and (mrspacman.y == math.floor(mrspacman.y)) then
-- If the tile in front of Pacman is empty, set direction to that
if (nodemap[mrspacman.y-math2.sin(mrspacman.dir)][mrspacman.x+math2.cos(mrspacman.dir)]~=1) then
mrspacman.dz = mrspacman.dir
-- Disable this queue
mrspacman.nextDirection = false
end
end
end
-- If Pacman is NOT in the center of a tile
if (mrspacman.x ~= math.floor(mrspacman.x)) or (mrspacman.y ~= math.floor(mrspacman.y)) then
-- Constantly move forwards
mrspacman.x = mrspacman.x + (math2.cos(mrspacman.dz)*mrspacman.speed)
mrspacman.y = mrspacman.y - (math2.sin(mrspacman.dz)*mrspacman.speed)
else
-- If the tile in front of Pacman is empty, move to that tile
if (nodemap[mrspacman.y-math2.sin(mrspacman.dz)][mrspacman.x+math2.cos(mrspacman.dz)] ~= 1) then
mrspacman.x = mrspacman.x + (math2.cos(mrspacman.dz)*mrspacman.speed)
mrspacman.y = mrspacman.y - (math2.sin(mrspacman.dz)*mrspacman.speed)
end
end
mrspacman.dz = The angle being faced.
mrspacman.dir = The recorded keypress.
mrspacman.speed = .1 (Is a decimal to allow for the smooth movement).
Executing this code results in Pacman freezing when a backwards-key was pressed, and its position becoming messed up when rotating in a corner, going through walls... That can be seen here.
How can this be fixed?
UPDATE
I added a table named math2 consisting of the cos and sin fx's (Only will return the values for the 4 cardinal directions):
math2 = {
cos = function(angle)
vectors = {1,0,-1,0}
return vectors[(angle/90)+1]
end,
sin = function(angle)
vectors = {0,1,0,-1}
return vectors[(angle/90)+1]
end
}
So the question is quite broad one:
The problem is quite vague and the test case in the video is quite big and unfocused. To understand what's going on you need to look in detail at least at some of the steps of the program. Namely, the problem is that mrspacman can go down through the wall, if tile to the left of the wall is empty. This hints that there might be some wrongness with the code that checks for wall presence.
The code in the question is not complete, so errors may originate from outside of it. Nevertheless, in the last hour I've put together a primitive framework to test the code from question. (it is listed at the end of the answer)
Having toyed around with the code, I now believe that the whole pacman was written not only without a single trigonometric function, but without floating point numbers at all.
The thing with float numbers is that they're ill suited for exact calculations. Some compilers and computing systems will even spit curses on you for doing that.
Before the edit, the code in question had expressions with sines and cosines:
I've now evaluated it on my computer and indeed for values of
dir
that are0, 90, 180, 270
it evaluates to0, 1, 0, -1
. But that is on 64bit machine. I'm not sure TI-Nspire can be trusted to yield same precision.During my tests I've run into another problem with floats:
would introduce error in the decimal digit: for
mrspacman.x
equal to4.3
it would eval to4.2
causing the pacwoman to freeze. Strangely enough, when evaluated directly in the prompt, it would behave correctly. Anyways, the common implementation of that function adds some bias before flooring:math.floor(mrspacman.x*10+0.5)/10
.Apart from that, there were minor hiccups with axes orientation, which were detected quickly thanks to all of the debug information dumped by framework.
And that's all. The code works (although, I'll just leave this here without more arguments). If the code still does not behave on your device, you'll need to look for problems elsewhere.
Debug code
I've left your variables global as they presumably were in your program and that probably doesn't matter in a single-file script. But in general it is a good practice to make any variable
local
unless there is a really good reason for otherwise.