| line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
|
1
|
|
|
|
|
|
|
package Tk::Taxis; |
|
2
|
|
|
|
|
|
|
|
|
3
|
1
|
|
|
1
|
|
21649
|
use 5.008006;
|
|
|
1
|
|
|
|
|
4
|
|
|
|
1
|
|
|
|
|
33
|
|
|
4
|
1
|
|
|
1
|
|
6
|
use strict; |
|
|
1
|
|
|
|
|
1
|
|
|
|
1
|
|
|
|
|
34
|
|
|
5
|
1
|
|
|
1
|
|
4
|
use warnings::register( 'Tk::Taxis' ); |
|
|
1
|
|
|
|
|
6
|
|
|
|
1
|
|
|
|
|
291
|
|
|
6
|
|
|
|
|
|
|
|
|
7
|
|
|
|
|
|
|
our $VERSION = '2.03'; |
|
8
|
|
|
|
|
|
|
|
|
9
|
|
|
|
|
|
|
################################## defaults #################################### |
|
10
|
|
|
|
|
|
|
|
|
11
|
1
|
|
|
1
|
|
5
|
use constant WIDTH => 400; |
|
|
1
|
|
|
|
|
2
|
|
|
|
1
|
|
|
|
|
81
|
|
|
12
|
1
|
|
|
1
|
|
4
|
use constant HEIGHT => 400; |
|
|
1
|
|
|
|
|
1
|
|
|
|
1
|
|
|
|
|
34
|
|
|
13
|
1
|
|
|
1
|
|
5
|
use constant POPULATION => 20; |
|
|
1
|
|
|
|
|
1
|
|
|
|
1
|
|
|
|
|
40
|
|
|
14
|
1
|
|
|
1
|
|
6
|
use constant PREFERENCE => [ 100, 100 ]; |
|
|
1
|
|
|
|
|
1
|
|
|
|
1
|
|
|
|
|
42
|
|
|
15
|
1
|
|
|
1
|
|
5
|
use constant TUMBLE => 0.03; |
|
|
1
|
|
|
|
|
2
|
|
|
|
1
|
|
|
|
|
55
|
|
|
16
|
1
|
|
|
1
|
|
4
|
use constant SPEED => 0.006; |
|
|
1
|
|
|
|
|
1
|
|
|
|
1
|
|
|
|
|
42
|
|
|
17
|
1
|
|
|
1
|
|
4
|
use constant IMAGES => "woodlice"; |
|
|
1
|
|
|
|
|
2
|
|
|
|
1
|
|
|
|
|
52
|
|
|
18
|
1
|
|
|
|
|
35
|
use constant FILL => [ [ 'white', 'gray' ], |
|
19
|
1
|
|
|
1
|
|
4
|
[ 'white', 'gray' ] ]; |
|
|
1
|
|
|
|
|
1
|
|
|
20
|
1
|
|
|
1
|
|
4
|
use constant LEFT_FILL => "white"; # deprecated |
|
|
1
|
|
|
|
|
1
|
|
|
|
1
|
|
|
|
|
39
|
|
|
21
|
1
|
|
|
1
|
|
10
|
use constant RIGHT_FILL => "gray"; # deprecated |
|
|
1
|
|
|
|
|
2
|
|
|
|
1
|
|
|
|
|
102
|
|
|
22
|
|
|
|
|
|
|
use constant CALCULATION => |
|
23
|
|
|
|
|
|
|
sub |
|
24
|
|
|
|
|
|
|
{ |
|
25
|
0
|
|
|
|
|
0
|
my ( $critter ) = @_; |
|
26
|
0
|
|
|
|
|
0
|
my %boundries = $critter->get_boundries(); |
|
27
|
0
|
|
|
|
|
0
|
my ( $x, $y ) = $critter->get_pos(); |
|
28
|
|
|
|
|
|
|
return |
|
29
|
|
|
|
|
|
|
$x / $boundries{ width }, |
|
30
|
0
|
|
|
|
|
0
|
$y / $boundries{ height }; |
|
31
|
1
|
|
|
1
|
|
10
|
}; |
|
|
1
|
|
|
|
|
1
|
|
|
|
1
|
|
|
|
|
50
|
|
|
32
|
|
|
|
|
|
|
|
|
33
|
|
|
|
|
|
|
################################### widget ##################################### |
|
34
|
|
|
|
|
|
|
|
|
35
|
1
|
|
|
1
|
|
417
|
use Tk qw( DoOneEvent DONT_WAIT ); |
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
36
|
|
|
|
|
|
|
use Tk::Taxis::Critter; |
|
37
|
|
|
|
|
|
|
|
|
38
|
|
|
|
|
|
|
require Tk::Frame; |
|
39
|
|
|
|
|
|
|
our @ISA = ( 'Tk::Frame' ); |
|
40
|
|
|
|
|
|
|
|
|
41
|
|
|
|
|
|
|
Tk::Widget->Construct( 'Taxis' ); |
|
42
|
|
|
|
|
|
|
|
|
43
|
|
|
|
|
|
|
sub Populate |
|
44
|
|
|
|
|
|
|
{ |
|
45
|
|
|
|
|
|
|
my ( $taxis, $options ) = @_; |
|
46
|
|
|
|
|
|
|
my $canvas = $taxis->Canvas(); |
|
47
|
|
|
|
|
|
|
$taxis->Advertise( 'canvas' => $canvas ); |
|
48
|
|
|
|
|
|
|
$canvas->pack(); |
|
49
|
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
$taxis->{ _supress_redraw } = 1; # so no multiple redraws on initialisation |
|
51
|
|
|
|
|
|
|
$taxis->images( delete $options->{ -images } || IMAGES ); |
|
52
|
|
|
|
|
|
|
$taxis->preference( delete $options->{ -preference } || PREFERENCE ); |
|
53
|
|
|
|
|
|
|
$taxis->tumble( delete $options->{ -tumble } || TUMBLE ); |
|
54
|
|
|
|
|
|
|
$taxis->speed( delete $options->{ -speed } || SPEED ); |
|
55
|
|
|
|
|
|
|
$taxis->width( delete $options->{ -width } || WIDTH ); |
|
56
|
|
|
|
|
|
|
$taxis->height( delete $options->{ -height } || HEIGHT ); |
|
57
|
|
|
|
|
|
|
$taxis->population( delete $options->{ -population } || POPULATION ); |
|
58
|
|
|
|
|
|
|
$taxis->fill( delete $options->{ -fill } || FILL ); |
|
59
|
|
|
|
|
|
|
$taxis->calculation( delete $options->{ -calculation } || CALCULATION ); |
|
60
|
|
|
|
|
|
|
|
|
61
|
|
|
|
|
|
|
# deprecated options |
|
62
|
|
|
|
|
|
|
if ( $options->{ -left_fill } ) |
|
63
|
|
|
|
|
|
|
{ |
|
64
|
|
|
|
|
|
|
$taxis->left_fill( delete $options->{ -left_fill } || LEFT_FILL ); |
|
65
|
|
|
|
|
|
|
} |
|
66
|
|
|
|
|
|
|
if ( $options->{ -right_fill } ) |
|
67
|
|
|
|
|
|
|
{ |
|
68
|
|
|
|
|
|
|
$taxis->right_fill( delete $options->{ -right_fill } || RIGHT_FILL ); |
|
69
|
|
|
|
|
|
|
} |
|
70
|
|
|
|
|
|
|
|
|
71
|
|
|
|
|
|
|
$taxis->{ _supress_redraw } = 0; |
|
72
|
|
|
|
|
|
|
$taxis->refresh(); |
|
73
|
|
|
|
|
|
|
|
|
74
|
|
|
|
|
|
|
$taxis->ConfigSpecs |
|
75
|
|
|
|
|
|
|
( |
|
76
|
|
|
|
|
|
|
-images => [ 'METHOD', 'images', 'Images', undef ], |
|
77
|
|
|
|
|
|
|
-preference => [ 'METHOD', 'preference', 'Preference', undef ], |
|
78
|
|
|
|
|
|
|
-tumble => [ 'METHOD', 'tumble', 'Tumble', undef ], |
|
79
|
|
|
|
|
|
|
-speed => [ 'METHOD', 'speed', 'Speed', undef ], |
|
80
|
|
|
|
|
|
|
-width => [ 'METHOD', 'width', 'Width', undef ], |
|
81
|
|
|
|
|
|
|
-height => [ 'METHOD', 'height', 'Height', undef ], |
|
82
|
|
|
|
|
|
|
-population => [ 'METHOD', 'population', 'Population', undef ], |
|
83
|
|
|
|
|
|
|
-fill => [ 'METHOD', 'fill', 'Fill', undef ], |
|
84
|
|
|
|
|
|
|
-calculation => [ 'METHOD', 'calculation', 'Calculation', undef ], |
|
85
|
|
|
|
|
|
|
DEFAULT => [ $canvas ], |
|
86
|
|
|
|
|
|
|
); |
|
87
|
|
|
|
|
|
|
$taxis->SUPER::Populate( $options ); |
|
88
|
|
|
|
|
|
|
$taxis->Delegates( DEFAULT => $canvas ); |
|
89
|
|
|
|
|
|
|
} |
|
90
|
|
|
|
|
|
|
|
|
91
|
|
|
|
|
|
|
################################### images ##################################### |
|
92
|
|
|
|
|
|
|
|
|
93
|
|
|
|
|
|
|
sub images |
|
94
|
|
|
|
|
|
|
{ |
|
95
|
|
|
|
|
|
|
my ( $taxis, $images ) = @_; |
|
96
|
|
|
|
|
|
|
if ( $images ) |
|
97
|
|
|
|
|
|
|
{ |
|
98
|
|
|
|
|
|
|
$taxis->{ images } = $images; |
|
99
|
|
|
|
|
|
|
unless ( $taxis->{ image_bank }{ $images } ) |
|
100
|
|
|
|
|
|
|
{ |
|
101
|
|
|
|
|
|
|
$taxis->{ image_bank }{ $images } = |
|
102
|
|
|
|
|
|
|
{ |
|
103
|
|
|
|
|
|
|
n => $taxis->Photo( -file => $taxis->_find_image( "n.gif" ) ), |
|
104
|
|
|
|
|
|
|
ne => $taxis->Photo( -file => $taxis->_find_image( "ne.gif" ) ), |
|
105
|
|
|
|
|
|
|
e => $taxis->Photo( -file => $taxis->_find_image( "e.gif" ) ), |
|
106
|
|
|
|
|
|
|
se => $taxis->Photo( -file => $taxis->_find_image( "se.gif" ) ), |
|
107
|
|
|
|
|
|
|
s => $taxis->Photo( -file => $taxis->_find_image( "s.gif" ) ), |
|
108
|
|
|
|
|
|
|
sw => $taxis->Photo( -file => $taxis->_find_image( "sw.gif" ) ), |
|
109
|
|
|
|
|
|
|
w => $taxis->Photo( -file => $taxis->_find_image( "w.gif" ) ), |
|
110
|
|
|
|
|
|
|
nw => $taxis->Photo( -file => $taxis->_find_image( "nw.gif" ) ), |
|
111
|
|
|
|
|
|
|
0 => $taxis->Photo(), |
|
112
|
|
|
|
|
|
|
}; |
|
113
|
|
|
|
|
|
|
} |
|
114
|
|
|
|
|
|
|
$taxis->image_height |
|
115
|
|
|
|
|
|
|
( |
|
116
|
|
|
|
|
|
|
$taxis->{ image_bank }{ $images }{ n }->height() || 50 |
|
117
|
|
|
|
|
|
|
); |
|
118
|
|
|
|
|
|
|
$taxis->image_width |
|
119
|
|
|
|
|
|
|
( |
|
120
|
|
|
|
|
|
|
$taxis->{ image_bank }{ $images }{ n }->width() || 50 |
|
121
|
|
|
|
|
|
|
); |
|
122
|
|
|
|
|
|
|
$taxis->refresh(); |
|
123
|
|
|
|
|
|
|
} |
|
124
|
|
|
|
|
|
|
return $taxis->{ images }; |
|
125
|
|
|
|
|
|
|
} |
|
126
|
|
|
|
|
|
|
|
|
127
|
|
|
|
|
|
|
sub _find_image |
|
128
|
|
|
|
|
|
|
{ |
|
129
|
|
|
|
|
|
|
my ( $taxis, $file ) = @_; |
|
130
|
|
|
|
|
|
|
my $dir = $taxis->{ images }; |
|
131
|
|
|
|
|
|
|
my $found; |
|
132
|
|
|
|
|
|
|
if ( my ( $path ) = $dir =~ /^\@(.*)$/ ) |
|
133
|
|
|
|
|
|
|
{ |
|
134
|
|
|
|
|
|
|
$found = ( grep { -e $_ } "$path/$file" )[ 0 ]; |
|
135
|
|
|
|
|
|
|
warnings::warn( "No such file $path/$file" ) unless $found; |
|
136
|
|
|
|
|
|
|
} |
|
137
|
|
|
|
|
|
|
else |
|
138
|
|
|
|
|
|
|
{ |
|
139
|
|
|
|
|
|
|
$found = |
|
140
|
|
|
|
|
|
|
( grep { -f $_ } map { "$_/Tk/Taxis/images/$dir/$file" } @INC )[ 0 ]; |
|
141
|
|
|
|
|
|
|
warnings::warn( "No such file \@INC/Tk/Taxis/images/$dir/$file" ) |
|
142
|
|
|
|
|
|
|
unless $found; |
|
143
|
|
|
|
|
|
|
} |
|
144
|
|
|
|
|
|
|
return $found; |
|
145
|
|
|
|
|
|
|
} |
|
146
|
|
|
|
|
|
|
|
|
147
|
|
|
|
|
|
|
sub _create_critter_image |
|
148
|
|
|
|
|
|
|
{ |
|
149
|
|
|
|
|
|
|
my ( $taxis, $critter ) = @_; |
|
150
|
|
|
|
|
|
|
my $canvas = $taxis->Subwidget( 'canvas' ); |
|
151
|
|
|
|
|
|
|
my @pos = $critter->get_pos(); |
|
152
|
|
|
|
|
|
|
my $id = $critter->get_id(); |
|
153
|
|
|
|
|
|
|
my $image = |
|
154
|
|
|
|
|
|
|
$taxis->{ image_bank }{ $taxis->{ images } }{ $critter->get_orient() }; |
|
155
|
|
|
|
|
|
|
if ( defined $id ) |
|
156
|
|
|
|
|
|
|
{ |
|
157
|
|
|
|
|
|
|
$canvas->coords( $id, $pos[ 0 ], $pos[ 1 ] ); |
|
158
|
|
|
|
|
|
|
$canvas->itemconfigure( $id, -image => $image ); |
|
159
|
|
|
|
|
|
|
} |
|
160
|
|
|
|
|
|
|
else |
|
161
|
|
|
|
|
|
|
{ |
|
162
|
|
|
|
|
|
|
my $id = $canvas->create |
|
163
|
|
|
|
|
|
|
( 'image', $pos[ 0 ], $pos[ 1 ], |
|
164
|
|
|
|
|
|
|
-anchor => 'center', -image => $image ); |
|
165
|
|
|
|
|
|
|
$critter->set_id( $id ); |
|
166
|
|
|
|
|
|
|
} |
|
167
|
|
|
|
|
|
|
return $taxis; |
|
168
|
|
|
|
|
|
|
} |
|
169
|
|
|
|
|
|
|
|
|
170
|
|
|
|
|
|
|
sub _hide_critter_image |
|
171
|
|
|
|
|
|
|
{ |
|
172
|
|
|
|
|
|
|
my ( $taxis, $critter ) = @_; |
|
173
|
|
|
|
|
|
|
my $canvas = $taxis->Subwidget( 'canvas' ); |
|
174
|
|
|
|
|
|
|
my $id = $critter->get_id(); |
|
175
|
|
|
|
|
|
|
my $image = $taxis->{ image_bank }{ $taxis->{ images } }{ 0 }; |
|
176
|
|
|
|
|
|
|
if ( defined $id ) |
|
177
|
|
|
|
|
|
|
{ |
|
178
|
|
|
|
|
|
|
$canvas->itemconfigure( $id, -image => $image ); |
|
179
|
|
|
|
|
|
|
} |
|
180
|
|
|
|
|
|
|
return $taxis; |
|
181
|
|
|
|
|
|
|
} |
|
182
|
|
|
|
|
|
|
|
|
183
|
|
|
|
|
|
|
sub image_height |
|
184
|
|
|
|
|
|
|
{ |
|
185
|
|
|
|
|
|
|
my ( $taxis, $image_height ) = @_; |
|
186
|
|
|
|
|
|
|
if ( defined $image_height ) |
|
187
|
|
|
|
|
|
|
{ |
|
188
|
|
|
|
|
|
|
$taxis->{ image_height } = $image_height; |
|
189
|
|
|
|
|
|
|
} |
|
190
|
|
|
|
|
|
|
return $taxis->{ image_height }; |
|
191
|
|
|
|
|
|
|
} |
|
192
|
|
|
|
|
|
|
|
|
193
|
|
|
|
|
|
|
sub image_width |
|
194
|
|
|
|
|
|
|
{ |
|
195
|
|
|
|
|
|
|
my ( $taxis, $image_width ) = @_; |
|
196
|
|
|
|
|
|
|
if ( defined $image_width ) |
|
197
|
|
|
|
|
|
|
{ |
|
198
|
|
|
|
|
|
|
$taxis->{ image_width } = $image_width; |
|
199
|
|
|
|
|
|
|
} |
|
200
|
|
|
|
|
|
|
return $taxis->{ image_width }; |
|
201
|
|
|
|
|
|
|
} |
|
202
|
|
|
|
|
|
|
|
|
203
|
|
|
|
|
|
|
################################## critters #################################### |
|
204
|
|
|
|
|
|
|
|
|
205
|
|
|
|
|
|
|
sub preference |
|
206
|
|
|
|
|
|
|
{ |
|
207
|
|
|
|
|
|
|
my ( $taxis, $preference ) = @_; |
|
208
|
|
|
|
|
|
|
if ( defined $preference ) |
|
209
|
|
|
|
|
|
|
{ |
|
210
|
|
|
|
|
|
|
$preference = [ $preference ] unless ref $preference; |
|
211
|
|
|
|
|
|
|
for my $i ( 0 .. 1 ) |
|
212
|
|
|
|
|
|
|
{ |
|
213
|
|
|
|
|
|
|
if ( defined $preference->[ $i ] ) |
|
214
|
|
|
|
|
|
|
{ |
|
215
|
|
|
|
|
|
|
if ( abs $preference->[ $i ] < 1 ) |
|
216
|
|
|
|
|
|
|
{ |
|
217
|
|
|
|
|
|
|
warnings::warn( "Absolute value of preference must be greater than 1" ); |
|
218
|
|
|
|
|
|
|
${ $preference }[ $i ] = 1; |
|
219
|
|
|
|
|
|
|
} |
|
220
|
|
|
|
|
|
|
} |
|
221
|
|
|
|
|
|
|
else |
|
222
|
|
|
|
|
|
|
{ |
|
223
|
|
|
|
|
|
|
$preference->[ $i ] = 1; |
|
224
|
|
|
|
|
|
|
} |
|
225
|
|
|
|
|
|
|
} |
|
226
|
|
|
|
|
|
|
$taxis->{ preference } = $preference; |
|
227
|
|
|
|
|
|
|
} |
|
228
|
|
|
|
|
|
|
return $taxis->{ preference }; |
|
229
|
|
|
|
|
|
|
} |
|
230
|
|
|
|
|
|
|
|
|
231
|
|
|
|
|
|
|
sub tumble |
|
232
|
|
|
|
|
|
|
{ |
|
233
|
|
|
|
|
|
|
my ( $taxis, $tumble ) = @_; |
|
234
|
|
|
|
|
|
|
if ( defined $tumble ) |
|
235
|
|
|
|
|
|
|
{ |
|
236
|
|
|
|
|
|
|
if ( $tumble > 1 ) |
|
237
|
|
|
|
|
|
|
{ |
|
238
|
|
|
|
|
|
|
warnings::warn( "Tumble value too high, setting to 1" ); |
|
239
|
|
|
|
|
|
|
$tumble = 1; |
|
240
|
|
|
|
|
|
|
} |
|
241
|
|
|
|
|
|
|
elsif ( $tumble < 0 ) |
|
242
|
|
|
|
|
|
|
{ |
|
243
|
|
|
|
|
|
|
warnings::warn( "Tumble value too low, setting to 0" ); |
|
244
|
|
|
|
|
|
|
$tumble = 0; |
|
245
|
|
|
|
|
|
|
} |
|
246
|
|
|
|
|
|
|
$taxis->{ tumble } = $tumble; |
|
247
|
|
|
|
|
|
|
} |
|
248
|
|
|
|
|
|
|
return $taxis->{ tumble }; |
|
249
|
|
|
|
|
|
|
} |
|
250
|
|
|
|
|
|
|
|
|
251
|
|
|
|
|
|
|
sub speed |
|
252
|
|
|
|
|
|
|
{ |
|
253
|
|
|
|
|
|
|
my ( $taxis, $speed ) = @_; |
|
254
|
|
|
|
|
|
|
if ( defined $speed ) |
|
255
|
|
|
|
|
|
|
{ |
|
256
|
|
|
|
|
|
|
my $canvas = $taxis->Subwidget( 'canvas' ); |
|
257
|
|
|
|
|
|
|
my $max_x = $canvas->cget( -width ); |
|
258
|
|
|
|
|
|
|
my $max_y = $canvas->cget( -height ); |
|
259
|
|
|
|
|
|
|
my $min_speed = 2 / sqrt ( $max_x**2 + $max_y**2 ); |
|
260
|
|
|
|
|
|
|
if ( $speed < $min_speed ) |
|
261
|
|
|
|
|
|
|
{ |
|
262
|
|
|
|
|
|
|
warnings::warn( "Speed too low, setting to minimum value of $min_speed" ); |
|
263
|
|
|
|
|
|
|
$speed = $min_speed; |
|
264
|
|
|
|
|
|
|
# or they sit there and spin |
|
265
|
|
|
|
|
|
|
} |
|
266
|
|
|
|
|
|
|
$taxis->{ speed } = $speed; |
|
267
|
|
|
|
|
|
|
} |
|
268
|
|
|
|
|
|
|
return $taxis->{ speed }; |
|
269
|
|
|
|
|
|
|
} |
|
270
|
|
|
|
|
|
|
|
|
271
|
|
|
|
|
|
|
sub calculation |
|
272
|
|
|
|
|
|
|
{ |
|
273
|
|
|
|
|
|
|
my ( $taxis, $calculation ) = @_; |
|
274
|
|
|
|
|
|
|
if ( defined $calculation ) |
|
275
|
|
|
|
|
|
|
{ |
|
276
|
|
|
|
|
|
|
$taxis->{ calculation } = $calculation; |
|
277
|
|
|
|
|
|
|
} |
|
278
|
|
|
|
|
|
|
return $taxis->{ calculation }; |
|
279
|
|
|
|
|
|
|
} |
|
280
|
|
|
|
|
|
|
|
|
281
|
|
|
|
|
|
|
#################################### taxis ##################################### |
|
282
|
|
|
|
|
|
|
|
|
283
|
|
|
|
|
|
|
sub taxis |
|
284
|
|
|
|
|
|
|
{ |
|
285
|
|
|
|
|
|
|
my ( $taxis, $options ) = @_; |
|
286
|
|
|
|
|
|
|
my $canvas = $taxis->Subwidget( 'canvas' ); |
|
287
|
|
|
|
|
|
|
if ( $taxis->{ critters } ) |
|
288
|
|
|
|
|
|
|
{ |
|
289
|
|
|
|
|
|
|
my $critter; |
|
290
|
|
|
|
|
|
|
for my $i ( 1 .. $taxis->{ population } ) |
|
291
|
|
|
|
|
|
|
{ |
|
292
|
|
|
|
|
|
|
$critter = $taxis->{ critters }[ $i ]; |
|
293
|
|
|
|
|
|
|
$critter->move(); |
|
294
|
|
|
|
|
|
|
$taxis->_create_critter_image( $critter ); |
|
295
|
|
|
|
|
|
|
} |
|
296
|
|
|
|
|
|
|
DoOneEvent( DONT_WAIT ); |
|
297
|
|
|
|
|
|
|
} |
|
298
|
|
|
|
|
|
|
return $taxis; |
|
299
|
|
|
|
|
|
|
} |
|
300
|
|
|
|
|
|
|
|
|
301
|
|
|
|
|
|
|
#################################### arena ##################################### |
|
302
|
|
|
|
|
|
|
|
|
303
|
|
|
|
|
|
|
sub population |
|
304
|
|
|
|
|
|
|
{ |
|
305
|
|
|
|
|
|
|
my ( $taxis, $population ) = @_; |
|
306
|
|
|
|
|
|
|
if ( defined $population ) |
|
307
|
|
|
|
|
|
|
{ |
|
308
|
|
|
|
|
|
|
$taxis->{ population } = abs $population; |
|
309
|
|
|
|
|
|
|
$taxis->refresh(); |
|
310
|
|
|
|
|
|
|
} |
|
311
|
|
|
|
|
|
|
if ( wantarray ) |
|
312
|
|
|
|
|
|
|
{ |
|
313
|
|
|
|
|
|
|
my $canvas = $taxis->Subwidget( 'canvas' ); |
|
314
|
|
|
|
|
|
|
my ( $top_left, $top_right, $bottom_left, $bottom_right ) |
|
315
|
|
|
|
|
|
|
= ( 0, 0, 0, 0 ); |
|
316
|
|
|
|
|
|
|
my $vert_limit = $canvas->cget( -height ) / 2; |
|
317
|
|
|
|
|
|
|
my $horiz_limit = $canvas->cget( -width ) / 2; |
|
318
|
|
|
|
|
|
|
for my $i ( 1 .. $taxis->{ population } ) |
|
319
|
|
|
|
|
|
|
{ |
|
320
|
|
|
|
|
|
|
if ( ${ $taxis->{ critters } }[ $i ]{ pos }[ 1 ] |
|
321
|
|
|
|
|
|
|
<= $vert_limit ) |
|
322
|
|
|
|
|
|
|
{ |
|
323
|
|
|
|
|
|
|
${ $taxis->{ critters } }[ $i ]{ pos }[ 0 ] |
|
324
|
|
|
|
|
|
|
<= $horiz_limit ? |
|
325
|
|
|
|
|
|
|
$top_left++ : |
|
326
|
|
|
|
|
|
|
$top_right++; |
|
327
|
|
|
|
|
|
|
} |
|
328
|
|
|
|
|
|
|
else |
|
329
|
|
|
|
|
|
|
{ |
|
330
|
|
|
|
|
|
|
${ $taxis->{ critters } }[ $i ]{ pos }[ 0 ] |
|
331
|
|
|
|
|
|
|
<= $canvas->cget( -width ) / 2 ? |
|
332
|
|
|
|
|
|
|
$bottom_left++ : |
|
333
|
|
|
|
|
|
|
$bottom_right++; |
|
334
|
|
|
|
|
|
|
} |
|
335
|
|
|
|
|
|
|
} |
|
336
|
|
|
|
|
|
|
return |
|
337
|
|
|
|
|
|
|
( |
|
338
|
|
|
|
|
|
|
top => ( $top_left + $top_right ), |
|
339
|
|
|
|
|
|
|
bottom => ( $bottom_left + $bottom_right ), |
|
340
|
|
|
|
|
|
|
left => ( $bottom_left + $top_left ), |
|
341
|
|
|
|
|
|
|
right => ( $bottom_right + $top_right ), |
|
342
|
|
|
|
|
|
|
top_left => $top_left, |
|
343
|
|
|
|
|
|
|
bottom_left => $bottom_left, |
|
344
|
|
|
|
|
|
|
top_right => $top_right, |
|
345
|
|
|
|
|
|
|
bottom_right => $bottom_right, |
|
346
|
|
|
|
|
|
|
total => ( $top_left + $top_right + $bottom_left + $bottom_right ), |
|
347
|
|
|
|
|
|
|
); |
|
348
|
|
|
|
|
|
|
} |
|
349
|
|
|
|
|
|
|
else |
|
350
|
|
|
|
|
|
|
{ |
|
351
|
|
|
|
|
|
|
return $taxis->{ population }; |
|
352
|
|
|
|
|
|
|
} |
|
353
|
|
|
|
|
|
|
} |
|
354
|
|
|
|
|
|
|
|
|
355
|
|
|
|
|
|
|
sub width |
|
356
|
|
|
|
|
|
|
{ |
|
357
|
|
|
|
|
|
|
my ( $taxis, $width ) = @_; |
|
358
|
|
|
|
|
|
|
if ( $width ) |
|
359
|
|
|
|
|
|
|
{ |
|
360
|
|
|
|
|
|
|
$taxis->{ width } = $width; |
|
361
|
|
|
|
|
|
|
$taxis->refresh(); |
|
362
|
|
|
|
|
|
|
} |
|
363
|
|
|
|
|
|
|
return $taxis->{ width }; |
|
364
|
|
|
|
|
|
|
} |
|
365
|
|
|
|
|
|
|
|
|
366
|
|
|
|
|
|
|
sub height |
|
367
|
|
|
|
|
|
|
{ |
|
368
|
|
|
|
|
|
|
my ( $taxis, $height ) = @_; |
|
369
|
|
|
|
|
|
|
if ( $height ) |
|
370
|
|
|
|
|
|
|
{ |
|
371
|
|
|
|
|
|
|
$taxis->{ height } = $height; |
|
372
|
|
|
|
|
|
|
$taxis->refresh(); |
|
373
|
|
|
|
|
|
|
} |
|
374
|
|
|
|
|
|
|
return $taxis->{ height }; |
|
375
|
|
|
|
|
|
|
} |
|
376
|
|
|
|
|
|
|
|
|
377
|
|
|
|
|
|
|
sub fill |
|
378
|
|
|
|
|
|
|
{ |
|
379
|
|
|
|
|
|
|
my ( $taxis, $fill ) = @_; |
|
380
|
|
|
|
|
|
|
if ( defined $fill ) |
|
381
|
|
|
|
|
|
|
{ |
|
382
|
|
|
|
|
|
|
if ( not ref $fill ) |
|
383
|
|
|
|
|
|
|
{ |
|
384
|
|
|
|
|
|
|
$taxis->{ fill } = [ [ $fill, $fill ], [ $fill, $fill ] ]; |
|
385
|
|
|
|
|
|
|
} |
|
386
|
|
|
|
|
|
|
elsif ( ref $fill && |
|
387
|
|
|
|
|
|
|
( not ref $fill->[0] ) && |
|
388
|
|
|
|
|
|
|
( not ref $fill->[1] ) ) |
|
389
|
|
|
|
|
|
|
{ |
|
390
|
|
|
|
|
|
|
$taxis->{ fill } = [ [ $fill->[0], $fill->[1] ], |
|
391
|
|
|
|
|
|
|
[ $fill->[0], $fill->[1] ] ]; |
|
392
|
|
|
|
|
|
|
} |
|
393
|
|
|
|
|
|
|
elsif ( ref $fill->[0] && ref $fill->[1] ) |
|
394
|
|
|
|
|
|
|
{ |
|
395
|
|
|
|
|
|
|
$taxis->{ fill } = [ [ $fill->[0][0], $fill->[0][1] ], |
|
396
|
|
|
|
|
|
|
[ $fill->[1][0], $fill->[1][1] ] ]; |
|
397
|
|
|
|
|
|
|
} |
|
398
|
|
|
|
|
|
|
else |
|
399
|
|
|
|
|
|
|
{ |
|
400
|
|
|
|
|
|
|
warnings::warn( "Invalid argument to fill" ); |
|
401
|
|
|
|
|
|
|
return; |
|
402
|
|
|
|
|
|
|
} |
|
403
|
|
|
|
|
|
|
$taxis->refresh(); |
|
404
|
|
|
|
|
|
|
} |
|
405
|
|
|
|
|
|
|
return $taxis->{ fill }; |
|
406
|
|
|
|
|
|
|
} |
|
407
|
|
|
|
|
|
|
|
|
408
|
|
|
|
|
|
|
sub left_fill |
|
409
|
|
|
|
|
|
|
{ |
|
410
|
|
|
|
|
|
|
my ( $taxis, $left_fill ) = @_; |
|
411
|
|
|
|
|
|
|
if ( $left_fill ) |
|
412
|
|
|
|
|
|
|
{ |
|
413
|
|
|
|
|
|
|
warnings::warn( "left_fill is deprecated, use fill instead" ); |
|
414
|
|
|
|
|
|
|
$taxis->{ fill }[0][0] = $left_fill; |
|
415
|
|
|
|
|
|
|
$taxis->{ fill }[1][0] = $left_fill; |
|
416
|
|
|
|
|
|
|
$taxis->refresh(); |
|
417
|
|
|
|
|
|
|
} |
|
418
|
|
|
|
|
|
|
return $taxis->{ fill }[0][0]; |
|
419
|
|
|
|
|
|
|
} |
|
420
|
|
|
|
|
|
|
|
|
421
|
|
|
|
|
|
|
sub right_fill |
|
422
|
|
|
|
|
|
|
{ |
|
423
|
|
|
|
|
|
|
my ( $taxis, $right_fill ) = @_; |
|
424
|
|
|
|
|
|
|
if ( $right_fill ) |
|
425
|
|
|
|
|
|
|
{ |
|
426
|
|
|
|
|
|
|
warnings::warn( "right_fill is deprecated, use fill instead" ); |
|
427
|
|
|
|
|
|
|
$taxis->{ fill }[0][1] = $right_fill; |
|
428
|
|
|
|
|
|
|
$taxis->{ fill }[1][1] = $right_fill; |
|
429
|
|
|
|
|
|
|
$taxis->refresh(); |
|
430
|
|
|
|
|
|
|
} |
|
431
|
|
|
|
|
|
|
return $taxis->{ fill }[1][1]; |
|
432
|
|
|
|
|
|
|
} |
|
433
|
|
|
|
|
|
|
|
|
434
|
|
|
|
|
|
|
sub refresh |
|
435
|
|
|
|
|
|
|
{ |
|
436
|
|
|
|
|
|
|
my ( $taxis, $options ) = @_; |
|
437
|
|
|
|
|
|
|
return if $taxis->{ _supress_redraw }; |
|
438
|
|
|
|
|
|
|
my $canvas = $taxis->Subwidget( 'canvas' ); |
|
439
|
|
|
|
|
|
|
$canvas->configure( -width => $taxis->width() ); |
|
440
|
|
|
|
|
|
|
$canvas->configure( -height => $taxis->height() ); |
|
441
|
|
|
|
|
|
|
my $max_x = $taxis->{ width }; |
|
442
|
|
|
|
|
|
|
my $max_y = $taxis->{ height }; |
|
443
|
|
|
|
|
|
|
if ( $taxis->{ arena } ) |
|
444
|
|
|
|
|
|
|
{ |
|
445
|
|
|
|
|
|
|
my ( $top_left, $top_right, $bottom_left, $bottom_right ) |
|
446
|
|
|
|
|
|
|
= @{ $taxis->{ arena } }; |
|
447
|
|
|
|
|
|
|
$canvas->coords |
|
448
|
|
|
|
|
|
|
( $top_left, 0, 0, $max_x/2, $max_y/2 ); |
|
449
|
|
|
|
|
|
|
$canvas->itemconfigure( $top_left, -fill => $taxis->{fill}[0][0] ); |
|
450
|
|
|
|
|
|
|
|
|
451
|
|
|
|
|
|
|
$canvas->coords |
|
452
|
|
|
|
|
|
|
( $top_right, $max_x/2, 0, $max_x, $max_y/2 ); |
|
453
|
|
|
|
|
|
|
$canvas->itemconfigure( $top_right, -fill => $taxis->{fill}[0][1] ); |
|
454
|
|
|
|
|
|
|
|
|
455
|
|
|
|
|
|
|
$canvas->coords |
|
456
|
|
|
|
|
|
|
( $bottom_left, 0, $max_y/2, $max_x/2, $max_y); |
|
457
|
|
|
|
|
|
|
$canvas->itemconfigure( $bottom_left, -fill => $taxis->{fill}[1][0] ); |
|
458
|
|
|
|
|
|
|
|
|
459
|
|
|
|
|
|
|
$canvas->coords |
|
460
|
|
|
|
|
|
|
( $bottom_right, $max_x/2, $max_y/2, $max_x, $max_y ); |
|
461
|
|
|
|
|
|
|
$canvas->itemconfigure( $bottom_right, -fill => $taxis->{fill}[1][1] ); |
|
462
|
|
|
|
|
|
|
|
|
463
|
|
|
|
|
|
|
} |
|
464
|
|
|
|
|
|
|
else |
|
465
|
|
|
|
|
|
|
{ |
|
466
|
|
|
|
|
|
|
my $top_left = $canvas->create |
|
467
|
|
|
|
|
|
|
( 'rectangle', 0, 0, $max_x/2, $max_y/2, |
|
468
|
|
|
|
|
|
|
-fill => $taxis->{fill}[0][0] ); |
|
469
|
|
|
|
|
|
|
|
|
470
|
|
|
|
|
|
|
my $top_right = $canvas->create |
|
471
|
|
|
|
|
|
|
( 'rectangle', $max_x/2, 0, $max_x, $max_y/2, |
|
472
|
|
|
|
|
|
|
-fill => $taxis->{fill}[0][1] ); |
|
473
|
|
|
|
|
|
|
|
|
474
|
|
|
|
|
|
|
my $bottom_left = $canvas->create |
|
475
|
|
|
|
|
|
|
( 'rectangle', 0, $max_y/2, $max_x/2, $max_y, |
|
476
|
|
|
|
|
|
|
-fill => $taxis->{fill}[1][0] ); |
|
477
|
|
|
|
|
|
|
|
|
478
|
|
|
|
|
|
|
my $bottom_right = $canvas->create |
|
479
|
|
|
|
|
|
|
( 'rectangle', $max_x/2, $max_y/2, $max_x, $max_y, |
|
480
|
|
|
|
|
|
|
-fill => $taxis->{fill}[1][1] ); |
|
481
|
|
|
|
|
|
|
|
|
482
|
|
|
|
|
|
|
$taxis->{ arena } = [ $top_left, $top_right, $bottom_left, $bottom_right ]; |
|
483
|
|
|
|
|
|
|
} |
|
484
|
|
|
|
|
|
|
my $i; |
|
485
|
|
|
|
|
|
|
for ( $i = 1 ; $i <= $taxis->{ population } ; $i++ ) |
|
486
|
|
|
|
|
|
|
{ |
|
487
|
|
|
|
|
|
|
my $critter = $taxis->{ critters }[ $i ]; |
|
488
|
|
|
|
|
|
|
unless ( $critter ) |
|
489
|
|
|
|
|
|
|
{ |
|
490
|
|
|
|
|
|
|
$critter = Tk::Taxis::Critter->new( -taxis => $taxis ); |
|
491
|
|
|
|
|
|
|
$taxis->{ critters }[ $i ] = $critter; |
|
492
|
|
|
|
|
|
|
} |
|
493
|
|
|
|
|
|
|
$critter->randomise(); |
|
494
|
|
|
|
|
|
|
$taxis->_create_critter_image( $critter ); |
|
495
|
|
|
|
|
|
|
} |
|
496
|
|
|
|
|
|
|
for my $j ( $i .. @{ $taxis->{ critters } } - 1 ) |
|
497
|
|
|
|
|
|
|
{ |
|
498
|
|
|
|
|
|
|
|
|
499
|
|
|
|
|
|
|
# We don't delete the critters from the critters arrayref, |
|
500
|
|
|
|
|
|
|
# we just keep track of the current population size, and |
|
501
|
|
|
|
|
|
|
# grow this as appropriate; we only hide their images from view in the |
|
502
|
|
|
|
|
|
|
# canvas. We do this because we cannot satifactorily |
|
503
|
|
|
|
|
|
|
# delete images from canvases, as this appears to cause memory leakage |
|
504
|
|
|
|
|
|
|
# even if we delete all references, and call the delete method on all |
|
505
|
|
|
|
|
|
|
# widgets. I presume this is a bug in Tk::Canvas, as it works for other |
|
506
|
|
|
|
|
|
|
# imaged widgets. This way we only get as big as the largest population |
|
507
|
|
|
|
|
|
|
# called during the life of the script. |
|
508
|
|
|
|
|
|
|
|
|
509
|
|
|
|
|
|
|
my $critter = $taxis->{ critters }[ $j ]; |
|
510
|
|
|
|
|
|
|
$taxis->_hide_critter_image( $critter ); |
|
511
|
|
|
|
|
|
|
} |
|
512
|
|
|
|
|
|
|
DoOneEvent( DONT_WAIT ); |
|
513
|
|
|
|
|
|
|
return $taxis; |
|
514
|
|
|
|
|
|
|
} |
|
515
|
|
|
|
|
|
|
|
|
516
|
|
|
|
|
|
|
1; |
|
517
|
|
|
|
|
|
|
|
|
518
|
|
|
|
|
|
|
__END__ |