textfiles/programming/spranim.pro

448 lines
15 KiB
Prolog
Raw Normal View History

2021-04-15 11:31:59 -07:00
From umcaner0@cc.umanitoba.ca Wed Sep 14 20:16:09 1994
From: umcaner0@cc.umanitoba.ca (Richard Theodore Caners)
Newsgroups: rec.games.programmer
Subject: A method for sprite animation (long)
Date: 13 Sep 1994 23:01:51 GMT
Organization: The University of Manitoba
NNTP-Posting-Host: mira.cc.umanitoba.ca
Hi, what follows is a text file I wrote describing a method of
sprite animation I am using in a game. I hope people find it
informative.
A SIMPLE METHOD FOR SPRITE ANIMATION
By Darren Gyles
* [Shameless Plug]
Author of HOOP
( Avaliable at ftp.funet.fi /pub/msdos/games/cards/hoop11.zip )
Note : The following implementation of sprite animation uses linked lists
fairly extensively. If you are not familiar with this concept, get any
first year computer science textbook and look into it.
I have recently been programming a simple Galaxian type shoot-em-up and
needed a method for sprite animation. This is the solution that I came up
with. I was fairly pleased with the results. I'm sure that this method is
not unique, but I haven't seen anybody mention it before so I will take
a stab at it.
I am assuming that you already have your sprite graphics routines written.
I will not cover machine specific graphics routines, just algorithms for
implementing sprites.
This document describes a simple sprite system and then expands this
system to include animation. Be forewarned that even though there is
quite a bit of code given, you will never get it to work unless you
understand the methods completely. It is really not very hard, the
most confusing thing are the linked lists. Good luck.
THE BASIC IDEA BEHIND SPRITE ANIMATION
The basic idea behind sprite animation (or any animation, for that matter)
is:
Show frame 1 -> wait delay period -> Show frame 2 -> delay -> etc..
Pretty simple, but how do you implement this?
- IMPLEMENTATION -
GENERAL SPRITE STUFF
For this discussion I will assume you have a linked list of sprites, and
each node in the list represents a sprite and contains all the information
relevant to that sprite (of course you could use an array for this as well
with some simple modifications). I am using C type pseudo code - I hope
that everybody can understand it.
Each node may look like:
struct sprite {
int xpos, ypos; // X and Y coordinates for sprite
int xspeed, yspeed; // X and Y speed of sprite
int type; // Type of sprite (ie 1 = ship, 2 = bullet etc..)
// These types will be defined by you
char background[256];
// Stores the piece of the background
// that this sprite covers
.
. // Various other parameters describing the sprite that you may
. // require.
. // (ie does the sprite collide with other sprites?,
. // does the sprite bounce around or just disappear at edge of
. // the screen? etc.. )
.
struct sprite *next; // Pointer to next sprite (NULL means end of list)
};
And you will have a pointer to the top of this list:
struct sprite *sprite_list;
So you will have a list that looks like this:
sprite_list --> Sprite Data /----> Sprite Data /-->NULL
Next ---------/ Next ---------/
NOTE: For simplicity, I am assuming that all sprites are the same size.
Which is a dumb assumption, but it should be pretty easy to alter
these functions to work with variable sized elements.
You should have an add_sprite() function and a delete_sprite() function
(which are not given here, but are pretty simple) which add and delete
elements to and from the linked list. And you will have a function that
looks like this:
// This function goes through the sprite list and updates all sprites
void do_sprites(void) {
struct sprite *current;
current = sprite_list;
while (current != NULL) {
// This function restores the background, by placing the
// data pointed to by current->background at current->xpos,
// current->ypos. You write this one.
RESTORE_BACKGROUND(current->background, current->xpos, current->ypos);
// Update x and y coordinates
current->xpos = current->xpos + current->xspeed;
current->ypos = current->ypos + current->yspeed;
// Do whatever bounds checking or whatever you want here
// This function will make a copy of the background that will be
// covered by the sprite in the array pointed to by
// current->background. You have to write this one too.
CUT_BACKGROUND(current->background, current->xpos, current->ypos);
// Draw sprite - This function draws the sprite at xpos, ypos
// the current->type would indicate what data to use when drawing.
// You'll have to write this one also.
// Uses the sprite_data[] array which contains the bitmaps for the
// sprites, see below for more details.
DRAWSPRITE(sprite_data[current->type], current->xpos, current->ypos);
current = current->next;
}
}
You would call this function every time you want to move the sprites (ie
every time you go through your main loop). Note that this is very simple
and you don't have to do it this way, I am only providing this example
to make the discussion of the animation more understandable.
To draw the sprite, you send the drawing function the address where the
sprite data starts. This data would be stored in an array that looks
something like:
unsigned char sprite_data[MAX_NO_OF_SPRITES][MAX_SIZE_OF_SPRITES];
So the data for sprite type 10 starts at sprite_data[10][0]. Remember that
I am assuming all sprites are the same size.
So, when do we animate? Right now.
ANIMATION
For animation I use linked lists. You could use arrays instead, but
I like linked lists because they can be referenced faster and more neatly.
The downside is that they are a little messy to set up and to deallocate when
finished.
Each particular sprite animation has its own linked list and each node
in that list represents a frame. Each node has a structure that looks like:
struct frame{
unsigned char *data; // Pointer to the bitmap data for this frame.
// If you are using compiled bitmaps, you could put
// a pointer to the compiled bitmap function here.
int delay; // This value indicates how long to wait before
// going to the next frame.
struct frame *next; // Pointer to the next frame
}
So a sprite that just does the same thing over and over again. Say an
enemy ship that has blinking lights would have a linked list that looks
like :
/--> FRAME 1 ----> FRAME2 -----\
| (Lights on) (Lights off) |
| |
| |
\---------------------------------------------/
Note that this animation only has two frames, but you could have as many as
you like. Also note that each frame has its own delay time so you can
make some frames appear for longer than others.
Now this is where linked lists get a little messy, to create a linked
list that had the above format it would look like:
struct frame *current, *last;
...
current = allocate( sizeof(struct frame) );
// We store the pointer to the start of the animation in the array
// ANIMATIONS, see below for more details.
animations[0] = current;
current->data = POINTER_TO_FRAME1_DATA;
current->delay = 20;
last = current;
current = allocate( sizeof(struct frame) );
current->data = POINTER_TO_FRAME2_DATA;
current->delay = 20;
current->next = last;
last->next = current;
...
Now assuming that we have several animation lists set up, how
do we keep track of them? Simply use an array. We can use our TYPE
field to index this array. For example:
struct frame *animations[MAX_ANIMATIONS];
So say that element 0 in the animations[] array is a ship and element
1 is a bullet, then the array would look like:
animations[0] --> Animation linked list for ship.
animations[1] --> Animation linked list for bullet.
So suppose we are creating a sprite that has type 4. We know that the
animation starts at animations[4].
Now we must add a couple of fields to the sprite struct that was defined
a little earlier in this document to help with our animation.
These fields are:
struct sprite {
...
struct frame *a_frame; // The animation frame that we are currently on
int a_count; // A counter that keeps track of the delay
...
};
So for each sprite we have these extra two fields. Now when we are adding
a sprite to the sprite_list, in addition to initializing xpos, ypos, xspeed,
yspeed etc.. we have the following two lines:
new_sprite->a_frame = animations[new_sprite->type];
new_sprite->a_count = new_sprite->a_frame->delay;
So for each sprite, we now will know what animation frame we are currently
on and how long we have left on this frame.
Each time we update the sprite, we subtract one from the a_count value.
When a_count == 0, we move to the next frame.
Here is a new version of do_sprites() using animation:
void do_sprites(void) {
struct sprite *current;
current = sprite_list;
while (current != NULL) {
RESTORE_BACKGROUND(current->background, current->xpos, current->ypos);
// NEW PART STARTS HERE
current->a_count--;
if (current->a_count ==0) {
// If we have reached end of animation, move to next frame and
// reset a_count.
current->a_frame = current->a_frame->next;
current->a_count = current->a_frame->delay;
}
// NEW PART ENDS HERE
current->xpos = current->xpos + current->xspeed;
current->ypos = current->ypos + current->yspeed;
// Do whatever bounds checking or whatever you want here
CUT_BACKGROUND(current->background, current->xpos, current->ypos);
DRAWSPRITE(sprite_data[current->type], current->xpos, current->ypos);
current = current->next;
}
}
Once you have set up the animation linked list and added the sprite to
the sprite linked list, every time you call do_sprites the animation is done
automatically.
FINITE ANIMATIONS
Finite animations are animations that don't have a loop, that is, they end.
Say for example an explosion, may have three frames: A small explosion,
a bigger explosion and the final biggest explosion. Once this animation
has ended we want to delete the explosion from sprite_list. This is easy
to do.
The animation linked list for the described explosion would look like:
EXPLOSION -----> EXPLOSION -----> EXPLOSION -----> NULL
FRAME # 1 FRAME # 2 FRAME # 3
The NEXT field of the last frame will be a NULL pointer. So when you are
advancing an animation, simply check to see if the next frame if NULL, if
it is, then delete the sprite.
This alteration would look like:
void do_sprites(void)
{
...
current->a_count--;
if (current->a_count ==0) {
// If we have reached end of animation, move to next frame and
// reset a_count.
if (current-a_frame->next == NULL)
// This function deletes the sprite from sprite_list
delete_sprite(current);
else {
current->a_frame = current->a_frame->next;
current->a_count = current->a_frame->delay;
}
}
...
}
Two things I must mention:
1. I didn't bother to show it in the above code fragment, but if you
delete the sprite, make sure that you jump over the following lines
that update the x and y positions and draw the sprite. Since the node
no longer exists, these operations are no longer valid.
2. This has nothing to do with sprites, but it is important to
realize that on MS-DOS machines comparing two pointers will
not always work because of the segmented architecture.
This is important when deleting a node. For example suppose
CURRENT is the node to be deleted, and LAST is the node directly
before CURRENT in the list. To delete CURRENT we would do the
following:
last->next = current->next;
free(current);
So we need to know which node precedes the node to be deleted.
Since the nodes don't keep track of the previous node (only the next)
we must traverse the list and find the node that precedes the node
to be deleted. But since we cannot compare pointers, we cannot simply
do the following: (current is a pointer to node to be deleted)
last = sprite_list;
while (last->next != current)
last = last->next
What I did was to give each node a unique nodeid, so to delete
a node, do the following:
last = sprite_list;
while (last->next->nodeid != current->nodeid)
last = last->next
This version would work. Remember that each node must have a UNIQUE
nodeid value. I hope this makes sense, if you are familiar with
linked lists it should. Although this isn't directly related to
sprite animation you must understand this problem if you wish to
implement the system as described in this document.
CLEANING UP THE ANIMATIONS
When the program is done executing, we must deallocate all the memory
that is used up by the animation linked lists. As I mentioned earlier,
this is a little messy, but not too bad. (Also remember to deallocate all
the nodes in the sprite_list linked list. This is very easy to do.)
Remember that all the animations are pointed to by the array called
(not surprisingly) animations[]. For each element in this array, we
delete the animation that is pointed to.
This is pretty simple, except that some of the animations have loops in them
so we cannot simply do the following.
struct frame *current, *next;
current = animations[0];
while (current!=NULL) {
next = current->next;
free(current);
current=next;
}
What we must do is first check to see if the list is finite. So what I do
is follow the list for 10 frames and see if a NULL pointer is found, if we
find one, the list does not loop and we can use the above method. If we
don't find an end, (and we are assuming that no animations have 10 or
more frames) then we "break" the loop, turning the animation int a non-
looped list and delete the animation as above. This "breaking" looks
like:
struct frame *current, *next;
// Break the loop
next = animations[0];
current = next->current
next->next = NULL;
// Deallocate list
while (current!=NULL) {
next = current->next;
free(current);
current=next;
}
It is good practice to always deallocate all dynamic memory before
terminating your program, so always try to do it, even if it is a pain.
CONCLUSION
Well, that's pretty much it. The basic idea is pretty simple, but I
complicated it with all the implementation details. I hope this makes
sense to people.
If you have any questions/comments/criticisims please feel free to E-mail
me at umcaner0@ccu.umantioba.ca
Also feel free to send me any programs you write that uses this technique,
I would like to see if anybody can actually decipher this mess and
get something working.