# Timer class for Games::Irrlicht - it represents on timer that is checked on
# each frame
package Games::Irrlicht::Timer;
# (C) by Tels
use strict;
require Exporter;
use Games::Irrlicht::Thingy;
use vars qw/@ISA $VERSION/;
@ISA = qw/Games::Irrlicht::Thingy Exporter/;
$VERSION = '0.04';
sub _init
{
my $self = shift;
# times can be negative, for instance when clock goes backwards!
$self->{time} = shift;
$self->{count} = shift; # count < 0 => infitite
$self->{delay} = shift || $self->{time};
$self->{rand} = abs(shift || 0); # wiggle firing
if (($self->{rand} > abs($self->{time})) ||
($self->{rand} > abs($self->{delay})))
{
require Carp;
Carp::croak ("Timer rand value cannot be greater than time or delay");
}
$self->{start} = shift; # time we started to live
$self->{code} = shift; # callback
if (ref($self->{code}) ne 'CODE')
{
require Carp; Carp::croak ("Timer needs a coderef as callback!");
}
$self->{args} = [ @_ ]; # additional arguments
$self->{due} = 0; # not yet
$self->{next_shot} = $self->{start} + $self->{time};
# add in random delay if necc.
if ($self->{rand} != 0)
{
$self->{next_shot} += int(rand($self->{rand})- $self->{rand} / 2);
}
$self->{overdue} = 0; # when we fire, we are late
# by this amount
if ($self->{time} == 0) # well, maybe due right now
{
$self->_should_fire($self->{start});
$self->fire();
}
$self;
}
sub next_shot
{
# when will next shot be fired?
my $self = shift;
$self->{next_shot};
}
sub fired
{
# did this timer fire?
my $self = shift;
$self->{due};
}
sub count
{
# how many shots left?
my $self = shift;
$self->{count};
}
sub _should_fire
{
# fire the timer
my ($self,$now) = @_;
$self->{due} = 1;
$self->{overdue} = $now - $self->{next_shot}; # we are so late
# our next shot will be then (regardless of when this shot was fired)
$self->{next_shot} += $self->{delay};
if ($self->{rand} != 0)
{
$self->{next_shot} += int(rand($self->{rand}) - $self->{rand} / 2);
}
$self->{count}-- if $self->{count} > 0; # one shot less
$self->{time} = $self->{delay}; # allow a positive
# time and then
# negative delays
$self->{overdue};
}
sub fire
{
# when the timer determined that it was due, this routine must be called
# to make it really fire
my ($self,$overdue) = @_;
$overdue = $self->{overdue} if !defined $overdue; # FPS.pm sets it
# fire timer now
&{$self->{code}}( $self->{app}, $self, $overdue, @{$self->{args} });
}
sub due
{
# check whether this timer is due or not, return 0 for not due, otherwise
# returns 1
my ($self,$now,$time_warp) = @_;
$self->{due} = 0; # not yet
return 0
if ($time_warp == 0) || ($self->{count} == 0) || ($self->{active} == 0);
# freeze backwards looking timers if time goes forward and vice versa
if ($self->{time} > 0)
{
return 0 if $time_warp < 0 || $now < $self->{next_shot};
}
else
{
return 0 if $time_warp > 0 || $now > $self->{next_shot};
}
$self->_should_fire($now);
$self->fire($now);
1;
}
sub is_due
{
# check whether this timer is due or not, return undef for not due, otherwise
# returns the overdue time (e.g. the time in ms the timer should have fired
# ago). If it returns something else than -1,
my ($self,$now,$time_warp) = @_;
$self->{due} = 0; # not yet
return
if ($time_warp == 0) || ($self->{count} == 0) || ($self->{active} == 0);
# freeze backwards looking timers if time goes forward and vice versa
if ($self->{time} > 0)
{
return if $time_warp < 0 || $now < $self->{next_shot};
}
else
{
return if $time_warp > 0 || $now > $self->{next_shot};
}
$self->_should_fire($now);
}
1;
__END__
=pod
=head1 NAME
Games::Irrlicht::Timer - a timer class for Games::Irrlicht
=head1 SYNOPSIS
my $timer = Games::Irrlicht::Timer->new(
$time_to_first_shot,
$count,
$delay_between_shots,
$randomize,
$now,
$callback,
@arguments_to_callback
);
=head1 DESCRIPTION
This package provides a timer class. It is used by Games::Irrlicht and you
should not to use it directly.
=head1 CALLBACK
Once the timer has expired, the callback code (CODE ref) is called with the
following parameters:
&{$callback}($self,$timer,$overdue,@arguments);
C<$self> is the app the timer resides in (e.g. the object of type
Games::Irrlicht), C<$timer> is the timer itself, C<$overdue> is the time the
timer is late (e.g. it fires now, but should have fired C<-$overdue> ms ago)
and the additional arguments are whatever was passed when the timer was
created.
=head1 METHODS
=over 2
=item new()
my $timer = Games::Irrlicht::Timer->new(
$time_to_first_shot,
$count,
$delay_between_shots,
$randomize,
$now,
$callback,
@arguments_to_callback
);
Creates a new timer that will first fire at $time_to_first_shot from $now on
(assuming $now is the current time), fire at most of $count times (negative
counts means infinitely often), and between each shot will wait for $delay.
Both initial time and delay are in ms.
The randomize parameter should be smaller than the time and/or delay, it
will be used to randomize the delay times.
If you setup a timer with a delay of 1000 ms and a randomize value of 100, the
earliest time it will fire is 900 ms, and the latest time would be 1100 ms.
=item next_shot()
$timer->next_shot();
Returns the absolute time when the timer will fire the next time.
=item due()
Check whether the time is due or not. If is ise due (or overdue), it will
fire.
=item id()
Returns the timers unique id.
=item fired()
Returns whether the timer fired or not. Use only after calling C.
=item count()
Returns the number of 'shots' left. Negative value means the timer will
fire infinitely often.
=item group()
my $group = $timer->group();
Returns the ref to the group this timer belongs to, or undef.
=item is_active()
$timer->is_active();
Returns true if the timer is active, or false for inactive. Inactive
timers do not fire.
=item activate()
Set the timer to active. Newly created ones are always active.
=item deactivate()
Set the timer to inactive. Newly created ones are always active.
=back
=head1 BUGS
C does ignore when the timer should have fire multiple times between
it was started and the time it is checked. E.g. when then timer is due in 100
ms, and should fire 3 times, and then is checked the first time after 1000 ms,
it should imidiately fire 3 times, each time having a different overdue time.
Currently it fires only once.
=head1 AUTHORS
(c) 2002, 2003, 2004, Tels
=head1 SEE ALSO
L
=cut