Sprite (Tutorial): Difference between revisions
(→Sprite Tutorial: In the middle of the first sprite-object example.) |
(→Sprite Tutorial: Finished Example 3.1) |
||
Line 22: | Line 22: | ||
# Now we need a control loop that draws the sprite and handles input.<tt> | # Now we need a control loop that draws the sprite and handles input.<tt> | ||
#:: : "drawloop" | #:: : "drawloop" | ||
#:: wait 1 | #:: wait for 1 | ||
#:: put c?? Sprite p00 at "x" "y" | #:: put c?? Sprite p00 at "x" "y" | ||
#:: if uppressed then "up" | #:: if uppressed then "up" | ||
Line 46: | Line 46: | ||
#:: set "spr&local&_height" to 3 | #:: set "spr&local&_height" to 3 | ||
#:: : "drawloop" | #:: : "drawloop" | ||
#:: wait 1 | #:: wait for 1 | ||
#:: put c?? Sprite "local" at "local2" "local3"</tt> | #:: put c?? Sprite "local" at "local2" "local3"</tt> | ||
# We also need some bounds checking. I prefer to define variables for zone boundaries, but we'll use constants with [[expressions]] for now.<tt> | # We also need some bounds checking. I prefer to define variables for zone boundaries, but we'll use constants with [[expressions]] for now.<tt> | ||
Line 65: | Line 65: | ||
#:: dec "local3" by 1 | #:: dec "local3" by 1 | ||
#:: goto "#return"</tt> | #:: goto "#return"</tt> | ||
# One pitfall that comes with using subroutines for movement like this is that it's possible to get the sprite stuck on corners as you move it. This is because the sprite position is not updated after the sprite counters are changed, but only after every move direction has been checked. One way to fix this is to just use the spr#_x and spr#_y counters instead of local counters, since those are directly tied to sprite position. But for reasons we'll discuss later, I recommend against this, or at least keeping copies of them in local counters. The other way is to make sure the sprite is updated (i.e. drawn) after each move. This is accomplished most easily through subroutines as well: | |||
#:: : "#draw" | |||
#:: put c?? Sprite "local" at "local2" "local3" | |||
#:: goto "#return" | |||
#: Then, for every location where the sprite needs updating, that is at the end of all movement subroutines and within the main loop, call the subroutine "#draw" with a goto. In fact, it's not really even necessary to call this in every iteration of the main loop if it's called on movement. It only needs to be called once after all the sprite counters have been initialized. | |||
This leaves us with this final code: | This leaves us with this final code: | ||
lockplayer | lockplayer | ||
Line 74: | Line 79: | ||
set "spr&local&_width" to 3 | set "spr&local&_width" to 3 | ||
set "spr&local&_height" to 3 | set "spr&local&_height" to 3 | ||
goto "#draw" | |||
: "drawloop" | : "drawloop" | ||
if uppressed then "#up" | if uppressed then "#up" | ||
if leftpressed then "#left" | if leftpressed then "#left" | ||
if rightpressed then "#right" | if rightpressed then "#right" | ||
if downpressed then "#down" | if downpressed then "#down" | ||
wait for 1 | |||
goto "drawloop" | goto "drawloop" | ||
: "#up" | : "#up" | ||
if "local3" <= 0 then "#return" | if "local3" <= 0 then "#return" | ||
dec "local3" by 1 | dec "local3" by 1 | ||
goto "#draw" | |||
goto "#return" | goto "#return" | ||
: "#down" | : "#down" | ||
if "local3" >= "(25-'spr&local&_height')" then "#return" | if "local3" >= "(25-'spr&local&_height')" then "#return" | ||
inc "local3" by 1 | inc "local3" by 1 | ||
goto "#draw" | |||
goto "#return" | goto "#return" | ||
: "#left" | : "#left" | ||
if "local2" <= 0 then "#return" | if "local2" <= 0 then "#return" | ||
dec "local2" by 1 | dec "local2" by 1 | ||
goto "#draw" | |||
goto "#return" | goto "#return" | ||
: "#right" | : "#right" | ||
if "local2" <= "(80-'spr&local&_width')" then "#return" | if "local2" <= "(80-'spr&local&_width')" then "#return" | ||
inc "local2" by 1 | inc "local2" by 1 | ||
goto "#draw" | |||
goto "#return" | |||
: "#draw" | |||
put c?? Sprite "local" at "local2" "local3" | |||
goto "#return" | goto "#return" | ||
===The Not So Basics: Sprite Collision=== | ===The Not So Basics: Sprite Collision=== | ||
Line 138: | Line 150: | ||
'''set "spr&local&_cwidth" to 3''' | '''set "spr&local&_cwidth" to 3''' | ||
'''set "spr&local&_cheight" to 1''' | '''set "spr&local&_cheight" to 1''' | ||
goto "#draw" | |||
: "drawloop" | : "drawloop" | ||
if uppressed then "#up" | if uppressed then "#up" | ||
if leftpressed then "#left" | if leftpressed then "#left" | ||
if rightpressed then "#right" | if rightpressed then "#right" | ||
if downpressed then "#down" | if downpressed then "#down" | ||
wait for 1 | |||
goto "drawloop" | goto "drawloop" | ||
: "#up" | : "#up" | ||
Line 150: | Line 162: | ||
'''if c?? Sprite_colliding "local" at 0 -1 then "#return"''' | '''if c?? Sprite_colliding "local" at 0 -1 then "#return"''' | ||
dec "local3" by 1 | dec "local3" by 1 | ||
goto "#draw" | |||
goto "#return" | goto "#return" | ||
: "#down" | : "#down" | ||
Line 155: | Line 168: | ||
'''if c?? Sprite_colliding "local" at 0 1 then "#return"''' | '''if c?? Sprite_colliding "local" at 0 1 then "#return"''' | ||
inc "local3" by 1 | inc "local3" by 1 | ||
goto "#draw" | |||
goto "#return" | goto "#return" | ||
: "#left" | : "#left" | ||
Line 160: | Line 174: | ||
'''if c?? Sprite_colliding "local" at -1 0 then "#return"''' | '''if c?? Sprite_colliding "local" at -1 0 then "#return"''' | ||
dec "local2" by 1 | dec "local2" by 1 | ||
goto "#draw" | |||
goto "#return" | goto "#return" | ||
: "#right" | : "#right" | ||
Line 165: | Line 180: | ||
'''if c?? Sprite_colliding "local" at 1 0 then "#return"''' | '''if c?? Sprite_colliding "local" at 1 0 then "#return"''' | ||
inc "local2" by 1 | inc "local2" by 1 | ||
goto "#draw" | |||
goto "#return" | |||
: "#draw" | |||
put c?? Sprite "local" at "local2" "local3" | |||
goto "#return" | goto "#return" | ||
===The Sprite-Object Model: Active Collision=== | ===The Sprite-Object Model: Active Collision=== | ||
Line 190: | Line 209: | ||
#:: if c?? Sprite_colliding "local" at 0 -1 then "collision"</tt> | #:: if c?? Sprite_colliding "local" at 0 -1 then "collision"</tt> | ||
#: And so on. | #: And so on. | ||
# You need to create some key objects and some door objects, at least one of each, but creating more than one will be really, really easy. Each of these will be sprites, and each of them will have a robot in control. First, draw some graphics over with the player source you had before. Remember what their parameters are. If you want to do something REALLY cool, create a list of constants and put it in the global robot to keep track of these numbers for you. This way, if you ever change them, or do something like making your sprites vlayer based, you'll have a much easier time of things since you'll only have to change the global robot. | # You need to create some key objects and some door objects, at least one of each, but creating more than one will be really, really easy. Each of these will be sprites, and each of them will have a robot in control. First, draw some graphics over with the player source you had before. Remember what their parameters are. If you want to do something REALLY cool, create a list of constants and put it in the global robot to keep track of these numbers for you. This way, if you ever change them, or do something like making your sprites vlayer based, you'll have a much easier time of things since you'll only have to change the global robot. In any case, use what you've learned about sprites so far to create a robot called "key" with a dynamic initialization routine:<tt> | ||
#:: set "local" to "robot_id" | |||
#:: set "local2" to "thisx" | |||
#:: set "local3" to "thisy" | |||
#:: set "local4" to "this_color" | |||
#:: gotoxy "local" 0 | |||
#:: . "@spr&local&" | |||
#:: set "spr&local&_refx" to XCOORD | |||
#:: set "spr&local&_refy" to YCOORD | |||
#:: set "spr&local&_width" to WIDTH | |||
#:: set "spr&local&_height" to HEIGHT | |||
#:: set "spr&local&_cx" to CX | |||
#:: set "spr&local&_cy" to CY | |||
#:: set "spr&local&_cwidth" to CWIDTH | |||
#:: set "spr&local&_cheight" to CHEIGHT | |||
#:: put "local4" Sprite "local" at "local2" "local3" | |||
#:: end</tt> | |||
#: Of course you'll need to provide your own values for those counters, and as I've said, I really recommend assigning them to constant counters whose values you set all in one place. It makes things much easier later, when you want to change the way things are done. Also notice that in addition to reading the coordinates of the key based on where you put the robot, it also reads the color of the key and draws the sprite accordingly. This means we can use exactly the same robot code for different colored keys, without changing a thing but except color of the robot itself. | |||
# Now you need a ":touch" label. In the case of a key, if you touch it you just collect the key and it disappears. So we need a counter to keep track of the key, and we need to make the sprite stop disappear.<tt> | |||
#:: : "touch" | |||
#:: inc "key_&local4&" by 1 | |||
#:: set "spr&local&_off" to 1 | |||
#:: die</tt> | |||
#: What we've done here is increment a counter that is unique to the color of the key. This means we can collect more than one of the same color key, and open just as many doors. The spr#_off counter is a special functional counter which turns off sprite drawing and sprite collision when it is set to something. It does NOT set the sprite's counters to zero, though. | |||
# Now, copy that key robot and make a door robot out of it. I'm serious, the changes you'll be making are so slight that they don't warrant starting from scratch. Just change the values of the sprite initialization to suit, and change the behavior in the touch routine:<tt> | |||
#:: : "touch" | |||
#:: if "key_&local4& <= 0 then "end" | |||
#:: dec "key_&local4&" by 1 | |||
#:: set "spr&local&_off" to 1 | |||
#:: die | |||
#:: : "end" | |||
#:: end</tt> | |||
#: We perform a simple check to make sure that the player collected a key that is the same color as this door. Other than that, it's a mirror image of the key's touch routine. | |||
# Now, copy those two robots all over the board. You may actually want to do this with copyrobot commands, so that you only have to change the code in one place, when you need to change it. Or, if you want to be clever, save the robot code into a file and load from that. Whatever you do, put a bunch of copies of the key and door code onto your board, using as many different colors as you like. Now play around with it. | |||
. "Global robot" | |||
set "spr_p_x" to 80 | |||
set "spr_p_y" to 0 | |||
set "spr_p_w" to 3 | |||
set "spr_p_h" to 3 | |||
set "spr_p_cx" to 0 | |||
set "spr_p_cy" to 2 | |||
set "spr_p_cw" to 3 | |||
set "spr_p_ch" to 1 | |||
set "spr_d_x" to 83 | |||
set "spr_d_y" to 0 | |||
set "spr_d_w" to 3 | |||
set "spr_d_h" to 3 | |||
set "spr_d_cx" to 0 | |||
set "spr_d_cy" to 0 | |||
set "spr_d_cw" to 3 | |||
set "spr_d_ch" to 3 | |||
set "spr_k_x" to 86 | |||
set "spr_k_y" to 0 | |||
set "spr_k_w" to 2 | |||
set "spr_k_h" to 3 | |||
set "spr_k_cx" to 0 | |||
set "spr_k_cy" to 0 | |||
set "spr_k_cw" to 2 | |||
set "spr_k_ch" to 3 | |||
. "Your values may differ, so please change them to suit." | |||
--------------------------------------------------------------- | |||
. "Player robot" | |||
lockplayer | |||
set "local" to '''"robot_id"''' | |||
set "local2" to '''"thisx"''' | |||
set "local3" to '''"thisy"''' | |||
'''gotoxy "local" 0''' | |||
'''. "&spr&local&"''' | |||
set "spr&local&_refx" to '''"spr_p_x"''' | |||
set "spr&local&_refy" to '''"spr_p_y"''' | |||
set "spr&local&_width" to '''"spr_p_w"''' | |||
set "spr&local&_height" to '''"spr_p_h"''' | |||
set "spr&local&_cx" to '''"spr_p_cx"''' | |||
set "spr&local&_cy" to '''"spr_p_cy"''' | |||
set "spr&local&_cwidth" to '''"spr_p_cw"''' | |||
set "spr&local&_cheight" to '''"spr_p_ch"''' | |||
goto "#draw" | |||
: "drawloop" | |||
if uppressed then "#up" | |||
if leftpressed then "#left" | |||
if rightpressed then "#right" | |||
if downpressed then "#down" | |||
wait for 1 | |||
goto "drawloop" | |||
: "#up" | |||
if "local3" <= 0 then "#return" | |||
if c?? Sprite_colliding "local" at 0 -1 then '''"collision"''' | |||
dec "local3" by 1 | |||
goto "#draw" | |||
goto "#return" | |||
: "#down" | |||
if "local3" >= "(25-'spr&local&_height')" then "#return" | |||
if c?? Sprite_colliding "local" at 0 1 then '''"collision"''' | |||
inc "local3" by 1 | |||
goto "#draw" | |||
goto "#return" | |||
: "#left" | |||
if "local2" <= 0 then "#return" | |||
if c?? Sprite_colliding "local" at -1 0 then '''"collision"''' | |||
dec "local2" by 1 | |||
goto "#draw" | |||
goto "#return" | |||
: "#right" | |||
if "local2" <= "(80-'spr&local&_width')" then "#return" | |||
if c?? Sprite_colliding "local" at 1 0 then '''"collision"''' | |||
inc "local2" by 1 | |||
goto "#draw" | |||
goto "#return" | |||
: "#draw" | |||
put c?? Sprite "local" at "local2" "local3" | |||
goto "#return" | |||
''': "collision"''' | |||
'''loop start''' | |||
'''goto "#collide('spr_clist&loopcount&')"''' | |||
'''send "spr('spr_clist&loopcount&')" to "touch"''' | |||
'''loop for "('spr_collisions'-1)''' | |||
'''goto "#return"''' | |||
--------------------------------------------------------------- | |||
. "Key robot" | |||
set "local" to "robot_id" | |||
set "local2" to "thisx" | |||
set "local3" to "thisy" | |||
set "local4" to "this_color" | |||
gotoxy "local" 0 | |||
. "@spr&local&" | |||
set "spr&local&_refx" to "spr_k_x" | |||
set "spr&local&_refy" to "spr_k_y" | |||
set "spr&local&_width" to "spr_k_w" | |||
set "spr&local&_height" to "spr_k_h" | |||
set "spr&local&_cx" to "spr_k_cx" | |||
set "spr&local&_cy" to "spr_k_cy" | |||
set "spr&local&_cwidth" to "spr_k_cw" | |||
set "spr&local&_cheight" to "spr_k_ch" | |||
put "local4" Sprite "local" at "local2" "local3" | |||
end | |||
: "touch" | |||
inc "key_&local4&" by 1 | |||
set "spr&local&_off" to 1 | |||
die | |||
--------------------------------------------------------------- | |||
. "Door robot" | |||
set "local" to "robot_id" | |||
set "local2" to "thisx" | |||
set "local3" to "thisy" | |||
set "local4" to "this_color" | |||
gotoxy "local" 0 | |||
. "@spr&local&" | |||
set "spr&local&_refx" to "spr_d_x" | |||
set "spr&local&_refy" to "spr_d_y" | |||
set "spr&local&_width" to "spr_d_w" | |||
set "spr&local&_height" to "spr_d_h" | |||
set "spr&local&_cx" to "spr_d_cx" | |||
set "spr&local&_cy" to "spr_d_cy" | |||
set "spr&local&_cwidth" to "spr_d_cw" | |||
set "spr&local&_cheight" to "spr_d_ch" | |||
put "local4" Sprite "local" at "local2" "local3" | |||
end | |||
: "touch" | |||
if "key_&local4& <= 0 then "end" | |||
dec "key_&local4&" by 1 | |||
set "spr&local&_off" to 1 | |||
die | |||
: "end" | |||
end</tt> | |||
(More to come...) | (More to come...) | ||
Revision as of 03:47, 21 October 2007
Sprites were introduced to MegaZeux in version 2.65, primarily as a method of making it easier to implement large object representations in the game (for example, engines for handling a multiple character player or enemies). They include many features such as collision detection, easy drawing, and configurable draw order. Despite being designed to make coding easier, many MZXers avoid their use in favor of more traditional methods such as overlay buffering and hand-rolled collision routines. This is largely out of a perception that Sprites are difficult to use.
Sprite Tutorial
Despite the mystery and trepidation that generally surrounds them, sprites are not that hard to work with and are designed to be easy to use, with the right program model. Two common and effective models, which can be used together or alone depending on the requirements of the engine, are the sprite-layer model and the sprite-object model. The structure of your code will depend on the application and the model being used, but there are some basics to cover first.
The Basics: Drawing Sprites
All sprites, regardless of their function, require some basic initialization and setup before they can be used. First, you need to set aside space on the board (or the Vlayer) for the sprite source image. You don't even have to put anything there yet, some advanced techniques construct the image data dynamically. But you need a block of space that is going to be used for sprite data. Next, all sprites need initialization code somewhere that designates this area.
set "spr#_refx" to XCOORD These counters specify the x coordinate, y coordinate, set "spr#_refy" to YCOORD width, and height of the bounding rectangle for the source image set "spr#_width" to WIDTH Replace # with the number of the sprite, from 0 to 255. set "spr#_height" to HEIGHT Counter interpolation is acceptable and in fact common for this.
Finally, all sprites need to be placed somewhere in order to be viewed. While "spr#_x" and "spr#_y" counters do exist, and they can be written to place and move the sprite, we recommend you treat them as read-only and use the "put" command for readability and clarity of purpose:
put c?? Sprite # at X Y # is the number of the sprite to be placed, as a parameter. Can be a counter.
Exercise 1.1: Making a sprite based player
A common application for sprites is to have all actors in the game, including the player, be represented by sprites. This lends itself well to a sprite-object model, but we'll come to that later. Our first exercise will be to make a player sprite that can be moved around with the arrow keys.
- First, start editing a new world, and draw a customblock smiley face on the board. It doesn't really matter what it looks like, or where it is, but for the sake of example put it at (80,0) off the right edge of the screen, and make it 3x3 characters large.
- Create a new robot called "sprite" to handle the sprite drawing and moving code. I like to put my control robots in a horizontal line starting at (1,0), but again, it really doesn't matter. The very first line of the robot should be "lockplayer", just to keep the player from moving around.
- Let's use sprite 0 for our player. So, the next 4 commands should be:
- set "spr0_refx" to 80
- set "spr0_refy" to 0
- set "spr0_width" to 3
- set "spr0_height" to 3
- Now we need a control loop that draws the sprite and handles input.
- : "drawloop"
- wait for 1
- put c?? Sprite p00 at "x" "y"
- if uppressed then "up"
- if leftpressed then "left"
- if rightpressed then "right"
- if downpressed then "down"
- goto "drawloop"
- Note that x and y will default to 0 since they haven't been used yet. It's generally a good idea to keep this information in a pair of local counters and initialize them along with the rest of the sprite data.
- Now, you just need code to modify the values of "x" and "y" for each label "up", "left", "right", and "down".
- : "up"
- dec "y" by 1
- goto "drawloop"
- And the rest should be obvious: inc "y" by 1 for "down", dec "x" by 1 for "left", and inc "x" by 1 for "right".
Exercise 1.2: Improving and generalizing the code
That's it, you can now test your world and move the sprite around. You'll notice that the sprite gets "stuck" if you move it off the top or left of the board, and disappears off the bottom right. That's because we didn't do any bounds checking. In fact, there are a lot of improvements we can make to this code before we move on.
- First let's make the sprite initialization dynamic. This may seem like more work now, but it'll make things much easier when you want to play with a lot of objects later, and decide that you need to reallocate sprite numbers. We'll declare local counters for the sprite draw location while we're at it.
- set "local" to 0
- set "local2" to 5
- set "local3" to 5
- set "spr&local&_refx" to 80
- set "spr&local&_refy" to 0
- set "spr&local&_width" to 3
- set "spr&local&_height" to 3
- : "drawloop"
- wait for 1
- put c?? Sprite "local" at "local2" "local3"
- We also need some bounds checking. I prefer to define variables for zone boundaries, but we'll use constants with expressions for now.
- : "up"
- if "local3" <= 0 then "drawloop"
- dec "local3" by 1
- goto "drawloop"
- : "down"
- if "local3" >= "(25-'spr&local&_height')" then "drawloop"
- inc "local3" by 1
- goto "drawloop"
- Remember that "local3" is "y".
- You also probably noticed that the previous movement routine favored certain directions over others; you can fix this by turning the label calls into subroutines (with diagonal movement as a free bonus!)
- if uppressed then "#up"
- ...
- : "#up"
- if "local3" <= 0 then "#return"
- dec "local3" by 1
- goto "#return"
- One pitfall that comes with using subroutines for movement like this is that it's possible to get the sprite stuck on corners as you move it. This is because the sprite position is not updated after the sprite counters are changed, but only after every move direction has been checked. One way to fix this is to just use the spr#_x and spr#_y counters instead of local counters, since those are directly tied to sprite position. But for reasons we'll discuss later, I recommend against this, or at least keeping copies of them in local counters. The other way is to make sure the sprite is updated (i.e. drawn) after each move. This is accomplished most easily through subroutines as well:
- : "#draw"
- put c?? Sprite "local" at "local2" "local3"
- goto "#return"
- Then, for every location where the sprite needs updating, that is at the end of all movement subroutines and within the main loop, call the subroutine "#draw" with a goto. In fact, it's not really even necessary to call this in every iteration of the main loop if it's called on movement. It only needs to be called once after all the sprite counters have been initialized.
This leaves us with this final code:
lockplayer set "local" to 0 set "local2" to 5 set "local3" to 5 set "spr&local&_refx" to 80 set "spr&local&_refy" to 0 set "spr&local&_width" to 3 set "spr&local&_height" to 3 goto "#draw" : "drawloop" if uppressed then "#up" if leftpressed then "#left" if rightpressed then "#right" if downpressed then "#down" wait for 1 goto "drawloop" : "#up" if "local3" <= 0 then "#return" dec "local3" by 1 goto "#draw" goto "#return" : "#down" if "local3" >= "(25-'spr&local&_height')" then "#return" inc "local3" by 1 goto "#draw" goto "#return" : "#left" if "local2" <= 0 then "#return" dec "local2" by 1 goto "#draw" goto "#return" : "#right" if "local2" <= "(80-'spr&local&_width')" then "#return" inc "local2" by 1 goto "#draw" goto "#return" : "#draw" put c?? Sprite "local" at "local2" "local3" goto "#return"
The Not So Basics: Sprite Collision
So we can now draw a sprite and move it around the screen. This is great, but except for some very limited cases is not particularly useful for gameplay. Simple bounds checking is easy enough to implement, but what if we want to have our player sprite move around inside a defined terrain, with walls and solid objects and impassible barriers? Worse, what if we want to interact with the environment, or with other sprites? Designing this from scratch would involve doing a lot of checking for customblocks, a way to figure out which sprite is at a specific location, and depending on the size of the sprite being moved, would require checking multiple locations each time in order to ensure consistency. It would be difficult to make the system general enough to be transplantable from game to game, and if you did manage it it would be hard to read and understand the code.
Fortunately, MZX provides a way to do all that work with one statement:
if c?? Sprite_colliding p## at X Y then LABEL p## is the number of the sprite you want to move, X and Y are relative coordinates.
One of the reasons people find this simple command so difficult to use is that they don't understand what it actually does. The first important requirement is that the sprite being moved define a collision rectangle. This should be done along with the other sprite initialization counters like so:
set "spr#_cx" to X These X and Y coordinates are relative values. set "spr#_cy" to Y That means you set them relative to (0,0) as the top left corner of the sprite. set "spr#_cwidth" to WIDTH So if you want the collision rectangle to be the same size and area as the sprite itself, set "spr#_cheight" to HEIGHT cx and cy should both be 0, and cwidth and cheight should be the same as width and height.
Most people understand this much. What often gets confused is that the Sprite_colliding object in the if statement is NOT this collision rectangle. Nor does it directly represent the collision rectangle of another sprite, though it is necessary for other sprites to have collision rectangles in order for collision to work. But there isn't an actual sprite_colliding object anywhere on the board, this is simply the syntax used to call collision detection for a sprite in advance of movement. The command says "if I hypothetically move this sprite (specified by the parameter) X by Y from its current location, will it collide with anything." And then it branches depending on whether the answer is true or false.
The two key points about X and Y are that they are relative to the sprite's current location, and they specify the movement of the sprite, not the position of something else. The color term is co-opted to perform a non-intuitive task of specifying relative versus absolute movement. c?? means that X and Y are relative values, and is what you will normally want to use. c00 (or any other absolute color) make X and Y absolute coordinates, but again the statement checks to see what would happen if you put the sprite at that location, not for the presence of some other object at that location. This is vital to understand, since without it you will probably try to do much more work than you need to do, and will probably achieve unexpected and incorrect results for your efforts.
Exercise 2: Adding basic collision detection
Let's see if we can't improve our player sprite code and turn it into something you might actually want to use in a game.
- First, take that board you were working on and draw some walls on it. Sprite collision detection only works with customblocks and other sprites, so in any game where you want to use sprite collision, all of your collidable scenery should be customblock. But if you're really interested in advanced MZX programming and artwork, you should already be doing this anyway.
- In order to create the illusion of depth and a 3/4 camera angle, we'll define the collision rectangle as being only the bottom row of the sprite. The sprite dimensions are 3x3, so that means:
- set "spr&local&_cx" to 0
- set "spr&local&_cy" to 2
- set "spr&local&_cwidth" to 3
- set "spr&local&_cheight" to 1
- Update each of your movement subroutines to include a collision check along with the bounds check:
- : "#up"
- if "local3" <= 0 then "#return"
- if c?? Sprite_colliding "local" at 0 -1 then "#return"
- dec "local3" by 1
- goto "#return"
That's it! That's really all you have to do!
lockplayer set "local" to 0 set "local2" to 5 set "local3" to 5 set "spr&local&_refx" to 80 set "spr&local&_refy" to 0 set "spr&local&_width" to 3 set "spr&local&_height" to 3 set "spr&local&_cx" to 0 set "spr&local&_cy" to 2 set "spr&local&_cwidth" to 3 set "spr&local&_cheight" to 1 goto "#draw" : "drawloop" if uppressed then "#up" if leftpressed then "#left" if rightpressed then "#right" if downpressed then "#down" wait for 1 goto "drawloop" : "#up" if "local3" <= 0 then "#return" if c?? Sprite_colliding "local" at 0 -1 then "#return" dec "local3" by 1 goto "#draw" goto "#return" : "#down" if "local3" >= "(25-'spr&local&_height')" then "#return" if c?? Sprite_colliding "local" at 0 1 then "#return" inc "local3" by 1 goto "#draw" goto "#return" : "#left" if "local2" <= 0 then "#return" if c?? Sprite_colliding "local" at -1 0 then "#return" dec "local2" by 1 goto "#draw" goto "#return" : "#right" if "local2" <= "(80-'spr&local&_width')" then "#return" if c?? Sprite_colliding "local" at 1 0 then "#return" inc "local2" by 1 goto "#draw" goto "#return" : "#draw" put c?? Sprite "local" at "local2" "local3" goto "#return"
The Sprite-Object Model: Active Collision
It's wonderful and all that we can make sprites not do something (i.e. move) if they collide. But what if we want to make them do something else instead? What if we want that thing to be different depending on the object of the collision? When you perform a collision check with "if sprite_colliding", it has the side effect of populating an array-like construct that details what, if anything, was collided with. This array is called "spr_clist", its length is stored in the counter "spr_collisions", and it is accessed as pseudo-arrays usually are in MZX, through counter interpolation. Here is a typical and flexible collision handler:
: "collision" loop start goto "#collide('spr_clist&loopcount&')" This will only goto labels that actually exist, making it easily pluggable. send "spr('spr_clist&loopcount&')" to "touch" This will send to a robot named with the same number as the colliding sprite. loop for "('spr_collisions'-1) Don't forget about the loop termination quirk, where the terminator is inclusive. goto "#return" Short circuits whatever else would have been done had collision not happened. : "#collide-1" -1 is the background, meaning the sprite collided with a customblock. * "~fOuch, I bumped into a wall!" Normally you wouldn't bother with this label though, since a wall is a wall. goto "#return" Continues collision handling. Remember, subroutines go on a call stack.
The send command is the beginning of understanding sprites with an object model, where each sprite is bound to code in a robot specific to that sprite, so that everything relevant to that sprite (including counters you may create) can be accessed with a single number. Here, we use that number to send the sprite (and I find it useful to make no distinction between the sprite and the robot controlling it) a touch label.
You should devise and maintain a convention for what robots that control sprites should be called. "sprite#" or "spr#" are normal, or you could just reference them by number. In order to make things as code driven as possible, so that you only have to change a single line to reassign a sprite number in a robot, you should do this dynamically in your robot setup with a rename command. And there's some other stuff you can do to make it easy to move sprites around just by moving the robots that control them.
set "local" to "robot_id" This makes things very dynamic, since you can copy the same robot around the board with no changes set "local2" to "thisx" and make multiple copies of the sprite. Setting local2 and local3 based on the robot's position set "local3" to "thisy" means that you don't have to configure the initial sprite placement, either. Then you can move the gotoxy "local" 0 robot into a robot bank at the top of the board, and the unique value of local ensures you won't . "@spr&local&" overwrite anything. But this is not always appropriate, sometimes you'll want to choose a specific ID
Example 3.1: Keys and doors with sprites
Now that we have an idea of how to make our sprite player touch things, let's create some things for him to touch.
- First, the player needs a collision handler. You can use the one we just discussed with no modification (though you may want to take out the #collide-1 label since that's just to demonstrate how to do something special based on what you collided with). You'll also need to change the target label for the collision check to "collision":
- if c?? Sprite_colliding "local" at 0 -1 then "collision"
- And so on.
- You need to create some key objects and some door objects, at least one of each, but creating more than one will be really, really easy. Each of these will be sprites, and each of them will have a robot in control. First, draw some graphics over with the player source you had before. Remember what their parameters are. If you want to do something REALLY cool, create a list of constants and put it in the global robot to keep track of these numbers for you. This way, if you ever change them, or do something like making your sprites vlayer based, you'll have a much easier time of things since you'll only have to change the global robot. In any case, use what you've learned about sprites so far to create a robot called "key" with a dynamic initialization routine:
- set "local" to "robot_id"
- set "local2" to "thisx"
- set "local3" to "thisy"
- set "local4" to "this_color"
- gotoxy "local" 0
- . "@spr&local&"
- set "spr&local&_refx" to XCOORD
- set "spr&local&_refy" to YCOORD
- set "spr&local&_width" to WIDTH
- set "spr&local&_height" to HEIGHT
- set "spr&local&_cx" to CX
- set "spr&local&_cy" to CY
- set "spr&local&_cwidth" to CWIDTH
- set "spr&local&_cheight" to CHEIGHT
- put "local4" Sprite "local" at "local2" "local3"
- end
- Of course you'll need to provide your own values for those counters, and as I've said, I really recommend assigning them to constant counters whose values you set all in one place. It makes things much easier later, when you want to change the way things are done. Also notice that in addition to reading the coordinates of the key based on where you put the robot, it also reads the color of the key and draws the sprite accordingly. This means we can use exactly the same robot code for different colored keys, without changing a thing but except color of the robot itself.
- Now you need a ":touch" label. In the case of a key, if you touch it you just collect the key and it disappears. So we need a counter to keep track of the key, and we need to make the sprite stop disappear.
- : "touch"
- inc "key_&local4&" by 1
- set "spr&local&_off" to 1
- die
- What we've done here is increment a counter that is unique to the color of the key. This means we can collect more than one of the same color key, and open just as many doors. The spr#_off counter is a special functional counter which turns off sprite drawing and sprite collision when it is set to something. It does NOT set the sprite's counters to zero, though.
- Now, copy that key robot and make a door robot out of it. I'm serious, the changes you'll be making are so slight that they don't warrant starting from scratch. Just change the values of the sprite initialization to suit, and change the behavior in the touch routine:
- : "touch"
- if "key_&local4& <= 0 then "end"
- dec "key_&local4&" by 1
- set "spr&local&_off" to 1
- die
- : "end"
- end
- We perform a simple check to make sure that the player collected a key that is the same color as this door. Other than that, it's a mirror image of the key's touch routine.
- Now, copy those two robots all over the board. You may actually want to do this with copyrobot commands, so that you only have to change the code in one place, when you need to change it. Or, if you want to be clever, save the robot code into a file and load from that. Whatever you do, put a bunch of copies of the key and door code onto your board, using as many different colors as you like. Now play around with it.
. "Global robot" set "spr_p_x" to 80 set "spr_p_y" to 0 set "spr_p_w" to 3 set "spr_p_h" to 3 set "spr_p_cx" to 0 set "spr_p_cy" to 2 set "spr_p_cw" to 3 set "spr_p_ch" to 1 set "spr_d_x" to 83 set "spr_d_y" to 0 set "spr_d_w" to 3 set "spr_d_h" to 3 set "spr_d_cx" to 0 set "spr_d_cy" to 0 set "spr_d_cw" to 3 set "spr_d_ch" to 3 set "spr_k_x" to 86 set "spr_k_y" to 0 set "spr_k_w" to 2 set "spr_k_h" to 3 set "spr_k_cx" to 0 set "spr_k_cy" to 0 set "spr_k_cw" to 2 set "spr_k_ch" to 3 . "Your values may differ, so please change them to suit." --------------------------------------------------------------- . "Player robot" lockplayer set "local" to "robot_id" set "local2" to "thisx" set "local3" to "thisy" gotoxy "local" 0 . "&spr&local&" set "spr&local&_refx" to "spr_p_x" set "spr&local&_refy" to "spr_p_y" set "spr&local&_width" to "spr_p_w" set "spr&local&_height" to "spr_p_h" set "spr&local&_cx" to "spr_p_cx" set "spr&local&_cy" to "spr_p_cy" set "spr&local&_cwidth" to "spr_p_cw" set "spr&local&_cheight" to "spr_p_ch" goto "#draw" : "drawloop" if uppressed then "#up" if leftpressed then "#left" if rightpressed then "#right" if downpressed then "#down" wait for 1 goto "drawloop" : "#up" if "local3" <= 0 then "#return" if c?? Sprite_colliding "local" at 0 -1 then "collision" dec "local3" by 1 goto "#draw" goto "#return" : "#down" if "local3" >= "(25-'spr&local&_height')" then "#return" if c?? Sprite_colliding "local" at 0 1 then "collision" inc "local3" by 1 goto "#draw" goto "#return" : "#left" if "local2" <= 0 then "#return" if c?? Sprite_colliding "local" at -1 0 then "collision" dec "local2" by 1 goto "#draw" goto "#return" : "#right" if "local2" <= "(80-'spr&local&_width')" then "#return" if c?? Sprite_colliding "local" at 1 0 then "collision" inc "local2" by 1 goto "#draw" goto "#return" : "#draw" put c?? Sprite "local" at "local2" "local3" goto "#return" : "collision" loop start goto "#collide('spr_clist&loopcount&')" send "spr('spr_clist&loopcount&')" to "touch" loop for "('spr_collisions'-1) goto "#return" --------------------------------------------------------------- . "Key robot" set "local" to "robot_id" set "local2" to "thisx" set "local3" to "thisy" set "local4" to "this_color" gotoxy "local" 0 . "@spr&local&" set "spr&local&_refx" to "spr_k_x" set "spr&local&_refy" to "spr_k_y" set "spr&local&_width" to "spr_k_w" set "spr&local&_height" to "spr_k_h" set "spr&local&_cx" to "spr_k_cx" set "spr&local&_cy" to "spr_k_cy" set "spr&local&_cwidth" to "spr_k_cw" set "spr&local&_cheight" to "spr_k_ch" put "local4" Sprite "local" at "local2" "local3" end : "touch" inc "key_&local4&" by 1 set "spr&local&_off" to 1 die --------------------------------------------------------------- . "Door robot" set "local" to "robot_id" set "local2" to "thisx" set "local3" to "thisy" set "local4" to "this_color" gotoxy "local" 0 . "@spr&local&" set "spr&local&_refx" to "spr_d_x" set "spr&local&_refy" to "spr_d_y" set "spr&local&_width" to "spr_d_w" set "spr&local&_height" to "spr_d_h" set "spr&local&_cx" to "spr_d_cx" set "spr&local&_cy" to "spr_d_cy" set "spr&local&_cwidth" to "spr_d_cw" set "spr&local&_cheight" to "spr_d_ch" put "local4" Sprite "local" at "local2" "local3" end : "touch" if "key_&local4& <= 0 then "end" dec "key_&local4&" by 1 set "spr&local&_off" to 1 die : "end" end
(More to come...)
External Links
Saike's Sprite Tutorial - Slightly out of date with regards to other MZX features.