I calculate a time delta since the last animation frame change and check if it exceeds the desired delta each update. If it does, go to the next animation frame and
reset the deltaTime to 0 subtract the expected duration from the actual duration to minimize error (as per K-Bal's suggestion). This results in a smaller margin of error than basing it solely on frame count without the asynchronicity that a timer introduces.
Psuedocode in case wording was unclear:
struct Frame {
int duration;
//...
}
class Sprite {
int frameCount;
Frame *frames;
int frame;
int ticks;
void Update();
//...
};
void Sprite::Update(int deltaTime)
{
ticks += deltaTime;
if(ticks >= frames[frame].duration) //milliseconds
{
ticks -= frames[frame].duration;
//assumes 1-directional looping order (e.g. 1, 2, 3, 1, 2, 3)
//in my actual code this is wrapped in a switch(frameOrder) where enum FrameOrder { None, Loop, Oscillation, ReverseLoop }
frame = ++frame % frameCount;
}
}