There are in fact five distinct states: the neko is either waking, moving,
stopping, scratching, or sleeping. The tkneko
states are encoded in a Perl hash with compiled (for efficiency)
regular expressions as keys and code references (the state
processors) as values:
%states = (
qr/AWAKE/ => \&do_awake,
qr/UP|UPRIGHT|RIGHT|DWRIGHT|DOWN|DWLEFT|LEFT|UPLEFT/ => \&do_move,
qr/STOP/ => \&do_stop,
qr/UTOGI|RTOGI|DTOGI|LTOGI/ => \&do_togi,
qr/SLEEP/ => \&do_sleep,
); # neko state table
The states that are dependent on the neko's direction (but
otherwise equivalent) are further divided into substates, described
by a regular expression with alternatives.
go_neko, the animation main loop, is activated by
a repeating 100 millisecond timer event. The subroutine's job
is simply to call a subroutine based on the animation's current
state, $state. The subroutine in turn selects an
appropriate PPM image and displays it on the canvas.
$mw->repeat(100 => \&go_neko);
sub go_neko {
$state_count++; # current state's cycle count
$where = sprintf("state=%-7s state_count=%05d, nx=%04d, ny=%04d",
$state, $state_count, $nx, $ny);
STATES:
foreach my $regex (keys %states) {
next STATES unless $state =~ /^($regex)$/;
&{$states{$regex}}($1);
return;
}
} # end go_neko
sub frame {
$canvas->coords($pixmaps{$pix}, -1000, -1000);
$pix = "$_[0].ppm";
$canvas->coords($pixmaps{$pix}, $nx, $ny);
}
So do_move might make a call such as this to make
the neko run left:
frame 'left' . (($state_count % 2) + 1);
Of course, in the actual program, the direction isn't a
hardcoded string but the back-reference $1 from
the state table's regular expression match.
To make the neko follow the cursor, we use Tk's
pointerxy command to get the cursor's
coordinates, compute a heading from the neko to the cursor, and then
map that value to a new state. $r2d is the
radian-to-degree conversion factor, and $h is the
new heading, in degrees.
($x, $y) = $canvas->pointerxy;
my $h = int( $r2d * atan2( ($y - $ny), ($x - $nx) ) ) % 360;
my($degrees, $dir);
foreach (
[[ 22.5, 67.5], 'DWRIGHT'],
[[ 67.5, 112.5], 'DOWN'],
[[112.5, 157.5], 'DWLEFT'],
[[157.5, 202.5], 'LEFT'],
[[202.5, 247.5], 'UPLEFT'],
[[247.5, 292.5], 'UP'],
[[292.5, 337.5], 'UPRIGHT'],
[[337.5, 22.5], 'RIGHT'],
) {
($degrees, $dir) = ($_->[0], $_->[1]);
last if $h >= $degrees->[0] and $h < $degrees->[1];
} # forend
set_state $dir;