Seeking Constructive Criticism on a Snake Clone
Forum rules
Before you make a thread asking for help, read this.
Before you make a thread asking for help, read this.
Seeking Constructive Criticism on a Snake Clone
I've always had a passion for programming (developer for a DoD intelligence agency, primarily focused on the web) but have yet to find an engine that really feels right to me. I have licenses for Torque and Torque Game Builder, played with PyGame, UDK, Ogre - I never really progressed anywhere.
Love has been on my radar for awhile but I never really played around with it. Today, I decided to go ahead and give it a go. I read through the LUA PIL for an hour or so while at work, came home and cranked out a Snake clone in about an hour. By no means is this perfect - it actually has a really nasty dependency on FPS that I want to figure out (please don't provide tips on this, I want to figure it out), but I am seeking constructive criticism on the code as a whole. Would you have done things differently?
One area I think could be improved is the collision detection - right now I am using generic for loops, but I think a reverse table would speed things up (if Lua allows using a table as a key for another table). I just didn't feel like playing with that part tonight and wanted to get a working prototype done.
The code is available at http://goo.gl/FlN3 and I welcome your feedback! Looking forward to taking part in this community - seems like a great group of people.
Love has been on my radar for awhile but I never really played around with it. Today, I decided to go ahead and give it a go. I read through the LUA PIL for an hour or so while at work, came home and cranked out a Snake clone in about an hour. By no means is this perfect - it actually has a really nasty dependency on FPS that I want to figure out (please don't provide tips on this, I want to figure it out), but I am seeking constructive criticism on the code as a whole. Would you have done things differently?
One area I think could be improved is the collision detection - right now I am using generic for loops, but I think a reverse table would speed things up (if Lua allows using a table as a key for another table). I just didn't feel like playing with that part tonight and wanted to get a working prototype done.
The code is available at http://goo.gl/FlN3 and I welcome your feedback! Looking forward to taking part in this community - seems like a great group of people.
Re: Seeking Constructive Criticism on a Snake Clone
Welcome! Nice to see proof positive of how easy Lua is to get started with
I'll leave the FPS dependency alone, as you request, but there are a few other things I'd like to point out:
I'll leave the FPS dependency alone, as you request, but there are a few other things I'd like to point out:
- Instead of using table.maxn, you should just use the # operator - table.maxn has its place, but when you're manipulating the table through table.insert and table.remove, # works just fine. Example:
Code: Select all
if #player.points >= player.max_length then
- At line 74, I think you mean math.randomseed(os.time()). It also shouldn't really be necessary to call math.randomseed more than once - I usually do it in love.load(). I guess it is a way of throwing more perceived randomness into the mix, though.
- One of the quirks of Lua's PRNG (and, by extension, rand()) is that on some platforms, the first number is not nearly as random as it ought to be, and will often be the same even with different starting seeds. Thus, to ensure the illusion of randomness, it's generally a good idea to burn a math.random after each math.randomseed call (which is another reason why simply doing it once in love.load is kinda nice).
- Robin
- The Omniscient
- Posts: 6506
- Joined: Fri Feb 20, 2009 4:29 pm
- Location: The Netherlands
- Contact:
Re: Seeking Constructive Criticism on a Snake Clone
Looks good! It's obvious you are already experienced in programming, so there aren't many problems with your code. In addition to the points anjo raised, I only want to point out some things that I'd have done differently, because they fit better with Lua's paradigms.
- Using a single table to contain them is more Luaic:
Code: Select all
UP = {x = 0, y = -1} DOWN = {x = 0, y = 1} LEFT = {x = -1, y = 0} RIGHT = {x = 1, y = 0}
Usually, people do this then:Code: Select all
directions = {up = {x = 0, y = -1}, down = {x = 0, y = 1}, ...}
andCode: Select all
player.dir = 'up'
If you go a bit further, you can replace the first part of love.keypressed with three lines, as a general way of changing direction.Code: Select all
player.x = player.x + directions[player.dir].x * player.speed
- Colors are usually done like:
Code: Select all
apples.color = {r = 255, g = 0, b = 0}
This has the nice implication that you can useCode: Select all
apples.color = {255, 0, 0}
Code: Select all
love.graphics.setColor(unpack(apples.color))
- Also, you use setColor in a loop. You can pull them out of there, you only need to set the color once.
Help us help you: attach a .love.
- bartbes
- Sex machine
- Posts: 4946
- Joined: Fri Aug 29, 2008 10:35 am
- Location: The Netherlands
- Contact:
Re: Seeking Constructive Criticism on a Snake Clone
I know you said you didn't want to hear about FPS independent coding, so I won't tell you anything about a solution, but if you're looking for it, both the wiki and all other good games should contain it.
Re: Seeking Constructive Criticism on a Snake Clone
Thanks for the feedback - I've updated the code with many of the suggestions you offered. Oddly enough this game is running at 300FPS at work and only 60FPS at home (my home computer blows my work computers out of the water - I guess I need to reboot at home). Needless to say the FPS dependency made it a vastly different, and faster, game today. I've yet to include that - going to work on it later on today.
Thanks again everyone!
Thanks - I was unaware of this operator, definitely cleans the code up a bit.anjo wrote:Instead of using table.maxn, you should just use the # operator
I'll have to keep this in mind - thanks for the tip!anjo wrote:One of the quirks of Lua's PRNG (and, by extension, rand()) is that on some platforms, the first number is not nearly as random as it ought to be, and will often be the same even with different starting seeds.
I like this solution much better than the global variable route I was taking. I wasn't too confident in being able to clean up the keypressed() function, because I wanted to eliminate the possibility of death when attempting to move in the opposite direction as the current movement. After a quick look back at the code building the directions table I realized the absolute values of opposite directions were ==. JACKPOT! Thanks for pointing me in this direction - much happier with the keypressed() function now.Robin wrote:Using a single table to contain them is more Luaic
Code: Select all
if key == "up" or key == "down" or key == "left" or key == "right" then
if math.abs(directions[player.dir].x) ~= math.abs(directions[key].x) and math.abs(directions[player.dir].y) ~= math.abs(directions[key].y) then
player.dir = key
end
end
I remembered reading about the unpack() function when reading through the LUA PIL, but didn't make the connection to use them here. This is, syntactically, how I attempted to implement colors initially as it is very similar to Python's tuples.Robin wrote:Colors are usually done like
Thanks again everyone!
Re: Seeking Constructive Criticism on a Snake Clone
Not sure this is how most others have implemented framerate independence, but this is what I came up with - would love to hear your ideas.
In love.update(), I've added the following code at the beginning of the function (well, right after I check to see if the game is still running):
Of course, game.update_time is set to 0 initially within love.load().
Thoughts?
In love.update(), I've added the following code at the beginning of the function (well, right after I check to see if the game is still running):
Code: Select all
game.update_time = game.update_time + dt
if game.update_time < 0.001 then do return end end
game.update_time = 0
Thoughts?
Re: Seeking Constructive Criticism on a Snake Clone
It does, but:walesmd wrote:if Lua allows using a table as a key for another table.
Code: Select all
local t1 = { x = 0, y = 0 }
local t2 = { x = 0, y = 0 }
if t1 == t2 then
-- never happens, t1 and t2 are two distinct tables
end
local t = {}
t[ t1 ] = 1
print( t[ t2 ] ) -- prints nil
Sounds like you have vsync turned on at your home computer.walesmd wrote:Oddly enough this game is running at 300FPS at work and only 60FPS at home
Cheers,
Andre
- bartbes
- Sex machine
- Posts: 4946
- Joined: Fri Aug 29, 2008 10:35 am
- Location: The Netherlands
- Contact:
Re: Seeking Constructive Criticism on a Snake Clone
It makes a lot more sense to change your speeds from pixels/update to pixels/second and then multiply by time (dt).
Re: Seeking Constructive Criticism on a Snake Clone
Yeah, I messed around with that but was coming across issues where the snake would have "blank" spaces within him. Nonetheless, I do agree - a much simpler implementation and for the purposes of this little prototype and learning the language/library I'm calling this project done.
Time to start working on the tutorial I promised my Twitter followers.
Time to start working on the tutorial I promised my Twitter followers.
- Robin
- The Omniscient
- Posts: 6506
- Joined: Fri Feb 20, 2009 4:29 pm
- Location: The Netherlands
- Contact:
Re: Seeking Constructive Criticism on a Snake Clone
The first line could be made even simpler by using something similar to if key in dict: often used in Python: in Lua, if a table doesn't have a value for a certain key, it is nil. Since all the values in the directions table are tables (which never evaluate to false), you could do if directions[key] then.walesmd wrote:Code: Select all
if key == "up" or key == "down" or key == "left" or key == "right" then if math.abs(directions[player.dir].x) ~= math.abs(directions[key].x) and math.abs(directions[player.dir].y) ~= math.abs(directions[key].y) then player.dir = key end end
The second line could be massively simplified as well, but that requires you to complicate other parts of your code, for example adding an "opposite" key to the tables in directions:
Code: Select all
up = {x = 0, y = -1, opposite = 'down'}
Code: Select all
if directions[key] and directions[key].opposite ~= player.dir then
player.dir = key
end
Help us help you: attach a .love.
Who is online
Users browsing this forum: Ahrefs [Bot] and 4 guests