| line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
|
1
|
|
|
|
|
|
|
package Java::Swing; |
|
2
|
8
|
|
|
8
|
|
204905
|
use strict; use warnings; |
|
|
8
|
|
|
8
|
|
18
|
|
|
|
8
|
|
|
|
|
274
|
|
|
|
8
|
|
|
|
|
34
|
|
|
|
8
|
|
|
|
|
18
|
|
|
|
8
|
|
|
|
|
225
|
|
|
3
|
|
|
|
|
|
|
|
|
4
|
8
|
|
|
8
|
|
40
|
use Carp; |
|
|
8
|
|
|
|
|
16
|
|
|
|
8
|
|
|
|
|
789
|
|
|
5
|
8
|
|
|
8
|
|
8355
|
use Inline Java => 'DATA'; |
|
|
8
|
|
|
|
|
194108
|
|
|
|
8
|
|
|
|
|
62
|
|
|
6
|
|
|
|
|
|
|
|
|
7
|
|
|
|
|
|
|
our $VERSION = '0.14'; |
|
8
|
|
|
|
|
|
|
|
|
9
|
|
|
|
|
|
|
my %callbacks; |
|
10
|
|
|
|
|
|
|
my %listeners; |
|
11
|
|
|
|
|
|
|
|
|
12
|
|
|
|
|
|
|
sub import { |
|
13
|
|
|
|
|
|
|
# To add packages to the Java::Swing scheme, see PerlRealPackages.pm. |
|
14
|
|
|
|
|
|
|
# Make your additions in PerlRealLocalPackages.pm or |
|
15
|
|
|
|
|
|
|
# PerlFakeLocalPackages.pm. Use Real for modules that have a .pm file |
|
16
|
|
|
|
|
|
|
# and Fake for modules this package should fabricate. |
|
17
|
|
|
|
|
|
|
|
|
18
|
8
|
|
|
8
|
|
5995
|
use Java::Swing::PerlRealPackages; |
|
|
8
|
|
|
|
|
23
|
|
|
|
8
|
|
|
|
|
199
|
|
|
19
|
8
|
|
|
8
|
|
3814
|
use Java::Swing::PerlRealLocalPackages; |
|
|
8
|
|
|
|
|
19
|
|
|
|
8
|
|
|
|
|
305
|
|
|
20
|
8
|
|
|
8
|
|
68
|
foreach my $package ( |
|
21
|
|
|
|
|
|
|
@Java::Swing::PerlRealPackages::packages, |
|
22
|
|
|
|
|
|
|
@Java::Swing::PerlRealLocalPackages::local_packages, |
|
23
|
|
|
|
|
|
|
) { |
|
24
|
8
|
|
|
8
|
|
41
|
no strict; |
|
|
8
|
|
|
|
|
10
|
|
|
|
8
|
|
|
|
|
420
|
|
|
25
|
24
|
|
|
|
|
42
|
*{"$package\::AUTOLOAD"} = \&module_autoload; |
|
|
24
|
|
|
|
|
201
|
|
|
26
|
|
|
|
|
|
|
} |
|
27
|
|
|
|
|
|
|
|
|
28
|
8
|
|
|
8
|
|
4075
|
use Java::Swing::PerlFakePackages; |
|
|
8
|
|
|
|
|
19
|
|
|
|
8
|
|
|
|
|
212
|
|
|
29
|
8
|
|
|
8
|
|
3619
|
use Java::Swing::PerlFakeLocalPackages; |
|
|
8
|
|
|
|
|
21
|
|
|
|
8
|
|
|
|
|
535
|
|
|
30
|
8
|
|
|
|
|
174
|
my %package_names = ( |
|
31
|
|
|
|
|
|
|
%Java::Swing::PerlFakePackages::names, |
|
32
|
|
|
|
|
|
|
%Java::Swing::PerlFakeLocalPackages::names, |
|
33
|
|
|
|
|
|
|
); |
|
34
|
8
|
|
|
|
|
96
|
foreach my $package ( |
|
35
|
|
|
|
|
|
|
@Java::Swing::PerlFakePackages::packages, |
|
36
|
|
|
|
|
|
|
@Java::Swing::PerlFakeLocalPackages::local_packages, |
|
37
|
|
|
|
|
|
|
) { |
|
38
|
8
|
|
|
8
|
|
37
|
no strict; |
|
|
8
|
|
|
|
|
13
|
|
|
|
8
|
|
|
|
|
1721
|
|
|
39
|
2448
|
|
|
|
|
5571
|
*{"$package\::AUTOLOAD"} = |
|
|
2448
|
|
|
|
|
38421
|
|
|
40
|
|
|
|
|
|
|
gen_fake_module_autoload($package_names{$package}); |
|
41
|
|
|
|
|
|
|
} |
|
42
|
|
|
|
|
|
|
} |
|
43
|
|
|
|
|
|
|
|
|
44
|
|
|
|
|
|
|
sub module_autoload { |
|
45
|
0
|
|
|
0
|
0
|
0
|
my $invocant = $_[0]; |
|
46
|
0
|
|
|
|
|
0
|
my $command = our $AUTOLOAD; |
|
47
|
0
|
|
|
|
|
0
|
my $required = "Java/Swing/$invocant.pm"; |
|
48
|
|
|
|
|
|
|
|
|
49
|
0
|
|
|
|
|
0
|
require $required; |
|
50
|
|
|
|
|
|
|
|
|
51
|
0
|
|
|
|
|
0
|
goto &$command; |
|
52
|
|
|
|
|
|
|
} |
|
53
|
|
|
|
|
|
|
|
|
54
|
|
|
|
|
|
|
sub gen_fake_module_autoload { |
|
55
|
2448
|
|
100
|
2448
|
0
|
8081
|
my $package_name = shift || 'javax.swing'; |
|
56
|
|
|
|
|
|
|
|
|
57
|
|
|
|
|
|
|
return sub { |
|
58
|
0
|
|
|
0
|
|
|
my $invocant = $_[0]; |
|
59
|
0
|
|
|
|
|
|
my $java_invocant = $invocant; |
|
60
|
0
|
|
|
|
|
|
$java_invocant =~ s/::/./g; |
|
61
|
0
|
|
|
|
|
|
my $command = our $AUTOLOAD; |
|
62
|
0
|
|
|
|
|
|
my $studied = "$package_name.$java_invocant"; |
|
63
|
|
|
|
|
|
|
|
|
64
|
0
|
|
|
|
|
|
Inline->bind( |
|
65
|
|
|
|
|
|
|
Java => 'STUDY', |
|
66
|
|
|
|
|
|
|
SHARED_JVM => 1, |
|
67
|
|
|
|
|
|
|
AUTOSTUDY => 1, |
|
68
|
|
|
|
|
|
|
STUDY => [ $studied ], |
|
69
|
|
|
|
|
|
|
); |
|
70
|
|
|
|
|
|
|
|
|
71
|
8
|
|
|
8
|
|
39
|
no strict; |
|
|
8
|
|
|
|
|
12
|
|
|
|
8
|
|
|
|
|
6544
|
|
|
72
|
0
|
|
|
|
|
|
*{"$invocant\::new"} = gen_generic_new($package_name); |
|
|
0
|
|
|
|
|
|
|
|
73
|
|
|
|
|
|
|
|
|
74
|
0
|
|
|
|
|
|
goto &$command; |
|
75
|
|
|
|
|
|
|
} |
|
76
|
2448
|
|
|
|
|
7593
|
} |
|
77
|
|
|
|
|
|
|
|
|
78
|
|
|
|
|
|
|
sub gen_generic_new { |
|
79
|
0
|
|
|
0
|
0
|
|
my $package_name = shift; |
|
80
|
0
|
|
|
|
|
|
$package_name =~ s/\./::/g; |
|
81
|
|
|
|
|
|
|
|
|
82
|
|
|
|
|
|
|
return sub { |
|
83
|
0
|
|
|
0
|
|
|
my $class = shift; |
|
84
|
0
|
|
|
|
|
|
my $inline_class = "Java\::Swing\::$package_name\::$class"; |
|
85
|
|
|
|
|
|
|
|
|
86
|
|
|
|
|
|
|
# if we were passed a hash reference, construct a default object, then |
|
87
|
|
|
|
|
|
|
# call set on each hash key with the value |
|
88
|
0
|
0
|
|
|
|
|
if (ref($_[0]) =~ /HASH/) { |
|
89
|
0
|
|
|
|
|
|
my $attributes = shift; |
|
90
|
0
|
|
|
|
|
|
my $retval; |
|
91
|
|
|
|
|
|
|
|
|
92
|
0
|
0
|
|
|
|
|
if ( defined $attributes->{ Object } ) { |
|
93
|
0
|
|
|
|
|
|
my $init_object = delete $attributes->{ Object }; |
|
94
|
0
|
|
|
|
|
|
$retval = $inline_class->new( $init_object ); |
|
95
|
|
|
|
|
|
|
} |
|
96
|
|
|
|
|
|
|
else { |
|
97
|
0
|
|
|
|
|
|
$retval = $inline_class->new(); |
|
98
|
|
|
|
|
|
|
} |
|
99
|
|
|
|
|
|
|
|
|
100
|
0
|
|
|
|
|
|
foreach my $attribute (keys %$attributes) { |
|
101
|
0
|
|
|
|
|
|
my $setter = "set" . ucfirst($attribute); |
|
102
|
0
|
|
|
|
|
|
eval { $retval->$setter($attributes->{$attribute}); }; |
|
|
0
|
|
|
|
|
|
|
|
103
|
0
|
0
|
|
|
|
|
if ($@) { |
|
104
|
0
|
|
|
|
|
|
croak "Error: '$attribute' is not an attribute of $class"; |
|
105
|
|
|
|
|
|
|
} |
|
106
|
|
|
|
|
|
|
} |
|
107
|
0
|
|
|
|
|
|
return $retval; |
|
108
|
|
|
|
|
|
|
} |
|
109
|
|
|
|
|
|
|
else { # pass args directly to constructor |
|
110
|
0
|
|
|
|
|
|
return $inline_class->new(@_); |
|
111
|
|
|
|
|
|
|
} |
|
112
|
|
|
|
|
|
|
} |
|
113
|
0
|
|
|
|
|
|
} |
|
114
|
|
|
|
|
|
|
|
|
115
|
|
|
|
|
|
|
# ------------------------------------------------------------------- |
|
116
|
|
|
|
|
|
|
# ----- Methods for Java::Swing objects ----- |
|
117
|
|
|
|
|
|
|
# ------------------------------------------------------------------- |
|
118
|
|
|
|
|
|
|
|
|
119
|
|
|
|
|
|
|
sub new { |
|
120
|
0
|
|
|
0
|
0
|
|
my $class = shift; |
|
121
|
0
|
|
|
|
|
|
my $self = {}; |
|
122
|
|
|
|
|
|
|
|
|
123
|
0
|
|
|
|
|
|
$self->{OBJECT} = Java::Swing::Swinger->new(); |
|
124
|
|
|
|
|
|
|
|
|
125
|
0
|
|
|
|
|
|
return bless $self, $class; |
|
126
|
|
|
|
|
|
|
} |
|
127
|
|
|
|
|
|
|
|
|
128
|
|
|
|
|
|
|
sub start { |
|
129
|
0
|
|
|
0
|
0
|
|
my $self = shift; |
|
130
|
|
|
|
|
|
|
|
|
131
|
0
|
|
|
|
|
|
$self->{OBJECT}->StartCallbackLoop(); |
|
132
|
|
|
|
|
|
|
} |
|
133
|
|
|
|
|
|
|
|
|
134
|
|
|
|
|
|
|
sub stop { |
|
135
|
0
|
|
|
0
|
0
|
|
my $self = shift; |
|
136
|
|
|
|
|
|
|
|
|
137
|
0
|
|
|
|
|
|
$self->{OBJECT}->StopCallbackLoop(); |
|
138
|
|
|
|
|
|
|
} |
|
139
|
|
|
|
|
|
|
|
|
140
|
|
|
|
|
|
|
# See Timer.pm for an example of using the delayed approach. To summarize, |
|
141
|
|
|
|
|
|
|
# if you need to pass your listener to the constructor of your sender, |
|
142
|
|
|
|
|
|
|
# call delayed_connect to get your listener. Pass it to the constructor. |
|
143
|
|
|
|
|
|
|
# Then call finish_delayed_connect with all three pieces (listener, sender, |
|
144
|
|
|
|
|
|
|
# and callbacks). |
|
145
|
|
|
|
|
|
|
sub delayed_connect { |
|
146
|
0
|
|
|
0
|
0
|
|
my $invocant = shift; # not used |
|
147
|
0
|
|
|
|
|
|
my $listener_type = shift; |
|
148
|
0
|
|
|
|
|
|
my $studied = "Perl$listener_type"; |
|
149
|
|
|
|
|
|
|
|
|
150
|
0
|
|
|
|
|
|
Inline->bind( |
|
151
|
|
|
|
|
|
|
Java => 'STUDY', |
|
152
|
|
|
|
|
|
|
SHARED_JVM => 1, |
|
153
|
|
|
|
|
|
|
AUTOSTUDY => 1, |
|
154
|
|
|
|
|
|
|
STUDY => [ $studied ], |
|
155
|
|
|
|
|
|
|
); |
|
156
|
|
|
|
|
|
|
|
|
157
|
0
|
|
|
|
|
|
my $listening_class = 'Java::Swing::' |
|
158
|
|
|
|
|
|
|
. "Perl$listener_type"; |
|
159
|
|
|
|
|
|
|
|
|
160
|
0
|
|
|
|
|
|
my $listener = $listening_class->new(); |
|
161
|
|
|
|
|
|
|
|
|
162
|
0
|
|
|
|
|
|
return $listener; |
|
163
|
|
|
|
|
|
|
} |
|
164
|
|
|
|
|
|
|
|
|
165
|
|
|
|
|
|
|
sub finish_delayed_connect { |
|
166
|
0
|
|
|
0
|
0
|
|
my $invocant = shift; # not used |
|
167
|
0
|
|
|
|
|
|
my $listener = shift; |
|
168
|
0
|
|
|
|
|
|
my $sender = shift; |
|
169
|
0
|
|
|
|
|
|
my $callbacks = shift; |
|
170
|
0
|
|
|
|
|
|
my $send_name = ref $sender; # stringify these |
|
171
|
0
|
|
|
|
|
|
my $call_name = ref $callbacks; |
|
172
|
|
|
|
|
|
|
|
|
173
|
0
|
|
|
|
|
|
$callbacks{$send_name}{$call_name} = $callbacks; |
|
174
|
0
|
|
|
|
|
|
$listeners{$send_name}{$call_name} = $listener; |
|
175
|
|
|
|
|
|
|
|
|
176
|
0
|
|
|
|
|
|
$listener->setSender($send_name); |
|
177
|
0
|
|
|
|
|
|
$listener->setCallbacks($call_name); |
|
178
|
|
|
|
|
|
|
} |
|
179
|
|
|
|
|
|
|
|
|
180
|
|
|
|
|
|
|
sub connect { |
|
181
|
0
|
|
|
0
|
0
|
|
my $invocant = shift; # not used |
|
182
|
0
|
|
|
|
|
|
my $listener_type = shift; |
|
183
|
0
|
|
|
|
|
|
my $sender = shift; |
|
184
|
0
|
|
|
|
|
|
my $callbacks = shift; |
|
185
|
0
|
|
|
|
|
|
my $studied = "Perl$listener_type"; |
|
186
|
|
|
|
|
|
|
|
|
187
|
0
|
|
|
|
|
|
Inline->bind( |
|
188
|
|
|
|
|
|
|
Java => 'STUDY', |
|
189
|
|
|
|
|
|
|
SHARED_JVM => 1, |
|
190
|
|
|
|
|
|
|
AUTOSTUDY => 1, |
|
191
|
|
|
|
|
|
|
STUDY => [ $studied ], |
|
192
|
|
|
|
|
|
|
); |
|
193
|
|
|
|
|
|
|
|
|
194
|
0
|
|
|
|
|
|
delegate_connection($listener_type, $sender, $callbacks); |
|
195
|
|
|
|
|
|
|
} |
|
196
|
|
|
|
|
|
|
|
|
197
|
|
|
|
|
|
|
sub delegate_connection { |
|
198
|
0
|
|
|
0
|
0
|
|
my $listener_type = shift; |
|
199
|
0
|
|
|
|
|
|
my $sender = shift; |
|
200
|
0
|
|
|
|
|
|
my $callbacks = shift; |
|
201
|
0
|
|
|
|
|
|
my $send_name = ref $sender; # stringify these |
|
202
|
0
|
|
|
|
|
|
my $call_name = ref $callbacks; |
|
203
|
|
|
|
|
|
|
|
|
204
|
0
|
|
|
|
|
|
my $listening_class = 'Java::Swing::' |
|
205
|
|
|
|
|
|
|
. "Perl$listener_type"; |
|
206
|
|
|
|
|
|
|
|
|
207
|
0
|
|
|
|
|
|
my $listener = $listening_class->new($send_name, $call_name); |
|
208
|
|
|
|
|
|
|
|
|
209
|
0
|
|
|
|
|
|
$callbacks{$send_name}{$call_name} = $callbacks; |
|
210
|
0
|
|
|
|
|
|
$listeners{$send_name}{$call_name} = $listener; |
|
211
|
|
|
|
|
|
|
|
|
212
|
0
|
|
|
|
|
|
my $add_method = "add$listener_type"; |
|
213
|
0
|
|
|
|
|
|
$sender->$add_method($listener); |
|
214
|
|
|
|
|
|
|
} |
|
215
|
|
|
|
|
|
|
|
|
216
|
|
|
|
|
|
|
# XXX not tested |
|
217
|
|
|
|
|
|
|
sub disconnect { |
|
218
|
0
|
|
|
0
|
0
|
|
my $invocant = shift; # not used |
|
219
|
0
|
|
|
|
|
|
my $listener_type = shift; |
|
220
|
0
|
|
|
|
|
|
my $sender = shift; |
|
221
|
0
|
|
|
|
|
|
my $callbacks = shift; |
|
222
|
0
|
|
|
|
|
|
my $listener_package = "Java::Swing::$listener_type"; |
|
223
|
|
|
|
|
|
|
|
|
224
|
0
|
|
|
|
|
|
$listener_package->disconnect($sender, $callbacks); |
|
225
|
|
|
|
|
|
|
} |
|
226
|
|
|
|
|
|
|
|
|
227
|
|
|
|
|
|
|
sub _Listener { # for the private use of our java class friends. |
|
228
|
0
|
|
|
0
|
|
|
my $sender_name = shift; |
|
229
|
0
|
|
|
|
|
|
my $callbacks_name = shift; |
|
230
|
0
|
|
|
|
|
|
my $triggered_method = shift; |
|
231
|
0
|
|
|
|
|
|
my $event = shift; |
|
232
|
|
|
|
|
|
|
|
|
233
|
0
|
0
|
|
|
|
|
my $methods = $callbacks{$sender_name}{$callbacks_name} |
|
234
|
|
|
|
|
|
|
or die "No registered callback for $sender_name $callbacks_name\n"; |
|
235
|
0
|
|
0
|
0
|
|
|
my $method = $methods->{$triggered_method} || sub {}; |
|
|
0
|
|
|
|
|
|
|
|
236
|
|
|
|
|
|
|
|
|
237
|
8
|
|
|
8
|
|
46
|
no strict; |
|
|
8
|
|
|
|
|
17
|
|
|
|
8
|
|
|
|
|
1129
|
|
|
238
|
0
|
|
|
|
|
|
$method->($sender_name, $event); |
|
239
|
|
|
|
|
|
|
} |
|
240
|
|
|
|
|
|
|
|
|
241
|
|
|
|
|
|
|
1; |
|
242
|
|
|
|
|
|
|
|
|
243
|
|
|
|
|
|
|
=head1 NAME |
|
244
|
|
|
|
|
|
|
|
|
245
|
|
|
|
|
|
|
Java::Swing - Perl extension providing direct access to the Java Swing API |
|
246
|
|
|
|
|
|
|
|
|
247
|
|
|
|
|
|
|
=head1 SYNOPSIS |
|
248
|
|
|
|
|
|
|
|
|
249
|
|
|
|
|
|
|
BEGIN { $ENV{CLASSPATH} .= ':/path/to/Java/Swing/java'; } |
|
250
|
|
|
|
|
|
|
|
|
251
|
|
|
|
|
|
|
use Java::Swing; |
|
252
|
|
|
|
|
|
|
my $swinger = Java::Swing->new(); |
|
253
|
|
|
|
|
|
|
|
|
254
|
|
|
|
|
|
|
my $frame = JFrame->new(); |
|
255
|
|
|
|
|
|
|
my $button = JButton->new( { label => 'Press Me' } ); |
|
256
|
|
|
|
|
|
|
|
|
257
|
|
|
|
|
|
|
$frame->getContentPane()->add($button); |
|
258
|
|
|
|
|
|
|
|
|
259
|
|
|
|
|
|
|
$swinger->connect( |
|
260
|
|
|
|
|
|
|
'ActionListener', $button, { actionPerformed => \&myListener } |
|
261
|
|
|
|
|
|
|
); |
|
262
|
|
|
|
|
|
|
|
|
263
|
|
|
|
|
|
|
my $timer = Timer->new(10, { actionPerformed => \&timer_catcher }); |
|
264
|
|
|
|
|
|
|
$timer->start(); |
|
265
|
|
|
|
|
|
|
|
|
266
|
|
|
|
|
|
|
$swinger->start(); |
|
267
|
|
|
|
|
|
|
|
|
268
|
|
|
|
|
|
|
sub mylistener { |
|
269
|
|
|
|
|
|
|
my $generating_object = shift; |
|
270
|
|
|
|
|
|
|
my $event = shift; |
|
271
|
|
|
|
|
|
|
print "Hello, Rob!\n"; |
|
272
|
|
|
|
|
|
|
} |
|
273
|
|
|
|
|
|
|
|
|
274
|
|
|
|
|
|
|
my $count = 1; |
|
275
|
|
|
|
|
|
|
sub timer_catcher { |
|
276
|
|
|
|
|
|
|
print "Timer went off " . $count++; |
|
277
|
|
|
|
|
|
|
} |
|
278
|
|
|
|
|
|
|
|
|
279
|
|
|
|
|
|
|
=head1 ABSTRACT |
|
280
|
|
|
|
|
|
|
|
|
281
|
|
|
|
|
|
|
Provides direct access to the Java Swing toolkit from Perl. |
|
282
|
|
|
|
|
|
|
|
|
283
|
|
|
|
|
|
|
=head1 DESCRIPTION |
|
284
|
|
|
|
|
|
|
|
|
285
|
|
|
|
|
|
|
Though you can write a Java program which is driven by Perl, some people |
|
286
|
|
|
|
|
|
|
may prefer to keep their Perl pure. This package lets you do that in manner |
|
287
|
|
|
|
|
|
|
similar to the way Perl/Tk and Gtk2:: provide access to their underlying |
|
288
|
|
|
|
|
|
|
libraries. This lets us code in our favorite language, while using the |
|
289
|
|
|
|
|
|
|
graphical interface capabilities of Java Swing. |
|
290
|
|
|
|
|
|
|
|
|
291
|
|
|
|
|
|
|
=head1 EXAMPLE |
|
292
|
|
|
|
|
|
|
|
|
293
|
|
|
|
|
|
|
In the example directory of the distribution there is an example called |
|
294
|
|
|
|
|
|
|
calc. Here I will walk through it a bit at a time. To see it in one |
|
295
|
|
|
|
|
|
|
piece look in the untarred distribution. To run it after make |
|
296
|
|
|
|
|
|
|
and before installing Java::Swing type: |
|
297
|
|
|
|
|
|
|
|
|
298
|
|
|
|
|
|
|
perl -I blib/lib example/calc |
|
299
|
|
|
|
|
|
|
|
|
300
|
|
|
|
|
|
|
After installation just use: |
|
301
|
|
|
|
|
|
|
|
|
302
|
|
|
|
|
|
|
perl example/calc |
|
303
|
|
|
|
|
|
|
|
|
304
|
|
|
|
|
|
|
But remember to change the path separators to fit your OS. |
|
305
|
|
|
|
|
|
|
|
|
306
|
|
|
|
|
|
|
use strict; use warnings; |
|
307
|
|
|
|
|
|
|
|
|
308
|
|
|
|
|
|
|
# This program provides an example of a simple Java::Swing application. |
|
309
|
|
|
|
|
|
|
# Type an expression in the top text box, press evaluate, and see the |
|
310
|
|
|
|
|
|
|
# answer in the other box. |
|
311
|
|
|
|
|
|
|
|
|
312
|
|
|
|
|
|
|
As the comment tries to say, this program displays two JTextFields and |
|
313
|
|
|
|
|
|
|
a JButton. When the button is pressed, the expression in the first field |
|
314
|
|
|
|
|
|
|
is eval'd and the result is placed in the second field. |
|
315
|
|
|
|
|
|
|
|
|
316
|
|
|
|
|
|
|
# Note that to make the example work, you must have the directory containing |
|
317
|
|
|
|
|
|
|
# the Java::Swing classes in the classpath. In the distribution this is |
|
318
|
|
|
|
|
|
|
# called java. |
|
319
|
|
|
|
|
|
|
|
|
320
|
|
|
|
|
|
|
BEGIN { $ENV{CLASSPATH} .= ':java' } |
|
321
|
|
|
|
|
|
|
|
|
322
|
|
|
|
|
|
|
No changes to the CLASSPATH will be effective if they come after |
|
323
|
|
|
|
|
|
|
use Java::Swing, so put them in a BEGIN block before that statement. |
|
324
|
|
|
|
|
|
|
The classes in the java directory of the distribution provide support |
|
325
|
|
|
|
|
|
|
for event listeners. |
|
326
|
|
|
|
|
|
|
|
|
327
|
|
|
|
|
|
|
use Java::Swing; |
|
328
|
|
|
|
|
|
|
|
|
329
|
|
|
|
|
|
|
This innocuous looking statement actually sets up the aliases that make it |
|
330
|
|
|
|
|
|
|
easy to refer to Java Swing classes. In particular, it sets up namespaces |
|
331
|
|
|
|
|
|
|
for each Component so you can refer to them directly as shown immediately |
|
332
|
|
|
|
|
|
|
below. It does not load the Java classes until you actually use them. |
|
333
|
|
|
|
|
|
|
|
|
334
|
|
|
|
|
|
|
my $expression = JTextField->new(); |
|
335
|
|
|
|
|
|
|
my $answer = JTextField->new( { columns => 10 } ); |
|
336
|
|
|
|
|
|
|
my $submit = JButton ->new("Evaluate"); |
|
337
|
|
|
|
|
|
|
my $frame = JFrame ->new(); |
|
338
|
|
|
|
|
|
|
my $root_pane = $frame ->getContentPane(); |
|
339
|
|
|
|
|
|
|
my $south_panel = JPanel ->new(); |
|
340
|
|
|
|
|
|
|
|
|
341
|
|
|
|
|
|
|
Once you use Java::Swing, you can refer to javax.swing classes by their |
|
342
|
|
|
|
|
|
|
class name alone as if it name were a Perl package name. All class methods, |
|
343
|
|
|
|
|
|
|
including constructors, can be called as normal through this Perl package |
|
344
|
|
|
|
|
|
|
name. |
|
345
|
|
|
|
|
|
|
|
|
346
|
|
|
|
|
|
|
But, if you like, you may also use Java::Swing named attribute construction, |
|
347
|
|
|
|
|
|
|
as shown for the second JTextField above. Simply supply a hash reference |
|
348
|
|
|
|
|
|
|
whose keys are attributes of the class with the proper values. Your object |
|
349
|
|
|
|
|
|
|
will be constructed by calling the empty argument constructor. Then the |
|
350
|
|
|
|
|
|
|
attribute values will be supplied by calling set accessors. So columns => 10 |
|
351
|
|
|
|
|
|
|
will translate into setColumns(10). |
|
352
|
|
|
|
|
|
|
|
|
353
|
|
|
|
|
|
|
As of version 0.12, you may add an Object attribute to the constructor |
|
354
|
|
|
|
|
|
|
hash. Then Java::Swing will call the constructor on the underlying class |
|
355
|
|
|
|
|
|
|
which expects it, and then call set accessors for any additional attributes. |
|
356
|
|
|
|
|
|
|
For example: |
|
357
|
|
|
|
|
|
|
|
|
358
|
|
|
|
|
|
|
my $label = JLabel->new( |
|
359
|
|
|
|
|
|
|
{ Object => $icon, |
|
360
|
|
|
|
|
|
|
text => 'caption', } |
|
361
|
|
|
|
|
|
|
); |
|
362
|
|
|
|
|
|
|
Thanks to Andreas Puerzer for suggesting this additional sugar. |
|
363
|
|
|
|
|
|
|
|
|
364
|
|
|
|
|
|
|
Continuing with the example: |
|
365
|
|
|
|
|
|
|
|
|
366
|
|
|
|
|
|
|
$south_panel->add(JLabel->new("Answer:"), "West" ); |
|
367
|
|
|
|
|
|
|
$south_panel->add($answer, "Center"); |
|
368
|
|
|
|
|
|
|
$south_panel->add($submit, "East" ); |
|
369
|
|
|
|
|
|
|
|
|
370
|
|
|
|
|
|
|
$root_pane->add($expression, "North"); |
|
371
|
|
|
|
|
|
|
$root_pane->add($south_panel, "South"); |
|
372
|
|
|
|
|
|
|
|
|
373
|
|
|
|
|
|
|
$frame->setSize(300, 100); |
|
374
|
|
|
|
|
|
|
$frame->show(); |
|
375
|
|
|
|
|
|
|
|
|
376
|
|
|
|
|
|
|
These lines perform Component layout. If you are not familiar with layouts in |
|
377
|
|
|
|
|
|
|
Java Swing (which has the same scheme as awt), consult a book (O'Reilly |
|
378
|
|
|
|
|
|
|
has more than one that will do, try the Java Foundation Classes in a Nutshell |
|
379
|
|
|
|
|
|
|
or Java Swing). |
|
380
|
|
|
|
|
|
|
|
|
381
|
|
|
|
|
|
|
my $swinger = Java::Swing->new(); |
|
382
|
|
|
|
|
|
|
|
|
383
|
|
|
|
|
|
|
At some point, you must obtain a Java::Swing object. Through it, you |
|
384
|
|
|
|
|
|
|
stop and start event listening. It also allows you to connect listeners |
|
385
|
|
|
|
|
|
|
directly to Perl code. |
|
386
|
|
|
|
|
|
|
|
|
387
|
|
|
|
|
|
|
$swinger->connect( |
|
388
|
|
|
|
|
|
|
"ActionListener", $submit, { actionPerformed => \&evaluate } |
|
389
|
|
|
|
|
|
|
); |
|
390
|
|
|
|
|
|
|
|
|
391
|
|
|
|
|
|
|
$swinger->connect( |
|
392
|
|
|
|
|
|
|
"WindowListener", $frame, { windowClosing => \&ending } |
|
393
|
|
|
|
|
|
|
); |
|
394
|
|
|
|
|
|
|
|
|
395
|
|
|
|
|
|
|
Call connect through your Java::Swing object passing it the listener type, |
|
396
|
|
|
|
|
|
|
the object to listen to, and a hash reference whose keys are all the events |
|
397
|
|
|
|
|
|
|
you care about. The values in the hash must be code references. These |
|
398
|
|
|
|
|
|
|
will be called when the event is triggered. They will receive the stringified |
|
399
|
|
|
|
|
|
|
name of the sending object (the originator of the event) and the event object. |
|
400
|
|
|
|
|
|
|
If you need the actual sending object, ask the event for it (try getSource). |
|
401
|
|
|
|
|
|
|
You only need to supply the events you care about. Leave others out, |
|
402
|
|
|
|
|
|
|
default no-ops will be called for them. |
|
403
|
|
|
|
|
|
|
|
|
404
|
|
|
|
|
|
|
If you plan to disconnect, you need to store the hash reference in a |
|
405
|
|
|
|
|
|
|
variable, so that you can pass EXACTLY the same arguments to the disconnect |
|
406
|
|
|
|
|
|
|
method. It is not enough to have the same data in the hash reference, it |
|
407
|
|
|
|
|
|
|
must be the same reference. |
|
408
|
|
|
|
|
|
|
|
|
409
|
|
|
|
|
|
|
If you want multiple routines called for the same event, call connect |
|
410
|
|
|
|
|
|
|
repeatedly. Give it a different hash reference each time, even if that |
|
411
|
|
|
|
|
|
|
reference refers to the same name/code ref pairs. |
|
412
|
|
|
|
|
|
|
|
|
413
|
|
|
|
|
|
|
$swinger->start(); |
|
414
|
|
|
|
|
|
|
|
|
415
|
|
|
|
|
|
|
After everything is ready, call start on your Java::Swing object. Once you |
|
416
|
|
|
|
|
|
|
do this Java takes over control with its event loop. This probably makes |
|
417
|
|
|
|
|
|
|
Java::Swing incompatible with other packages that want to manage the main |
|
418
|
|
|
|
|
|
|
loop, like POE (if you can make such packages cooperate, please advise me |
|
419
|
|
|
|
|
|
|
on how it is done). |
|
420
|
|
|
|
|
|
|
|
|
421
|
|
|
|
|
|
|
sub evaluate { |
|
422
|
|
|
|
|
|
|
my $sender_name = shift; |
|
423
|
|
|
|
|
|
|
my $event = shift; |
|
424
|
|
|
|
|
|
|
|
|
425
|
|
|
|
|
|
|
$answer->setText(eval $expression->getText()); |
|
426
|
|
|
|
|
|
|
} |
|
427
|
|
|
|
|
|
|
|
|
428
|
|
|
|
|
|
|
Here, evaluate pays no attention to the arguments it receives, but they |
|
429
|
|
|
|
|
|
|
are included so you can see how they come in. Instead, the text in the |
|
430
|
|
|
|
|
|
|
expression box is passed directly to eval (yes, security is important here, |
|
431
|
|
|
|
|
|
|
don't set uid). That answer is directly placed in the answer field via |
|
432
|
|
|
|
|
|
|
setText. |
|
433
|
|
|
|
|
|
|
|
|
434
|
|
|
|
|
|
|
sub ending { |
|
435
|
|
|
|
|
|
|
$swinger->stop(); |
|
436
|
|
|
|
|
|
|
} |
|
437
|
|
|
|
|
|
|
|
|
438
|
|
|
|
|
|
|
When you receive control via an event, you can stop the Java event loop by |
|
439
|
|
|
|
|
|
|
calling stop on your Java::Swing object. Here that happens when the user |
|
440
|
|
|
|
|
|
|
closes the window. When you stop the event loop, your program terminates. |
|
441
|
|
|
|
|
|
|
|
|
442
|
|
|
|
|
|
|
=head1 EXTENDING |
|
443
|
|
|
|
|
|
|
|
|
444
|
|
|
|
|
|
|
While I have tried to provide all the Swing you will ever need, there |
|
445
|
|
|
|
|
|
|
are inevitably some things I have not gotten to. Further, you may have |
|
446
|
|
|
|
|
|
|
in house code which you would like to incoroprate into this scheme. |
|
447
|
|
|
|
|
|
|
This section explains the pieces needed to use other code as part of |
|
448
|
|
|
|
|
|
|
Java::Swing. |
|
449
|
|
|
|
|
|
|
|
|
450
|
|
|
|
|
|
|
=head2 Adding Components or Families of Them |
|
451
|
|
|
|
|
|
|
|
|
452
|
|
|
|
|
|
|
While most of the Components from Swing are implemented, AWT, SWT and other |
|
453
|
|
|
|
|
|
|
kits are not implemented (though parts of AWT are). Most Components have |
|
454
|
|
|
|
|
|
|
the same basic plan. They are part of Swing. They have a no argument |
|
455
|
|
|
|
|
|
|
constructor and accessors for all of their attributes. If that describes |
|
456
|
|
|
|
|
|
|
your widget, there is only one thing to do: |
|
457
|
|
|
|
|
|
|
|
|
458
|
|
|
|
|
|
|
Add the base name of your class to the @packages list in |
|
459
|
|
|
|
|
|
|
Swing::PerlFakeLocalPackages |
|
460
|
|
|
|
|
|
|
|
|
461
|
|
|
|
|
|
|
If your widget is not at the top level of its package include the |
|
462
|
|
|
|
|
|
|
subdirectories leading from the package to it like so: |
|
463
|
|
|
|
|
|
|
|
|
464
|
|
|
|
|
|
|
text::html::AccessibleHTML |
|
465
|
|
|
|
|
|
|
|
|
466
|
|
|
|
|
|
|
If the widget has all of the above traits, but is in a different package, |
|
467
|
|
|
|
|
|
|
you should still add it to C<@packages> in C. |
|
468
|
|
|
|
|
|
|
But, you also need to add an entry for it in C<%names> like so: |
|
469
|
|
|
|
|
|
|
|
|
470
|
|
|
|
|
|
|
YourWidget => 'com.yourcompany.package', |
|
471
|
|
|
|
|
|
|
|
|
472
|
|
|
|
|
|
|
If you are adding lots of widgets, you'll want to automate this, but |
|
473
|
|
|
|
|
|
|
I don't have a lot of advice on how to do so. I did it once and forgot |
|
474
|
|
|
|
|
|
|
the scripts. I don't think they were complex. |
|
475
|
|
|
|
|
|
|
|
|
476
|
|
|
|
|
|
|
If the Java class is standard (for some sense of standard), send the module |
|
477
|
|
|
|
|
|
|
to me so I can add it to future distributions. |
|
478
|
|
|
|
|
|
|
|
|
479
|
|
|
|
|
|
|
=head2 Adding Listeners |
|
480
|
|
|
|
|
|
|
|
|
481
|
|
|
|
|
|
|
In Java::Swing, events are handled through callbacks to Perl code. To make |
|
482
|
|
|
|
|
|
|
the callbacks happen, you need a Java class which implements the |
|
483
|
|
|
|
|
|
|
listener interface. As of version 0.10, you no longer need a corresponding |
|
484
|
|
|
|
|
|
|
Perl module, the code from those modules is now implemented once in |
|
485
|
|
|
|
|
|
|
Swing.pm. It is possible to hand code the Java listener implementation, |
|
486
|
|
|
|
|
|
|
but it is a pain. |
|
487
|
|
|
|
|
|
|
|
|
488
|
|
|
|
|
|
|
There are two real possibilities. Either you have a single listener or |
|
489
|
|
|
|
|
|
|
you have several. In the first case, you can use listener_generator. |
|
490
|
|
|
|
|
|
|
This script is not installed, but can be found in the Swing/Generate |
|
491
|
|
|
|
|
|
|
subdirectory of the distribution. Details follow. |
|
492
|
|
|
|
|
|
|
|
|
493
|
|
|
|
|
|
|
=head3 One Listener at a Time |
|
494
|
|
|
|
|
|
|
|
|
495
|
|
|
|
|
|
|
Create a file describing the listener with a valid Perl hash reference in it |
|
496
|
|
|
|
|
|
|
like this: |
|
497
|
|
|
|
|
|
|
|
|
498
|
|
|
|
|
|
|
{ |
|
499
|
|
|
|
|
|
|
'listener' => 'TreeWillExpandListener', |
|
500
|
|
|
|
|
|
|
'methods' => [ |
|
501
|
|
|
|
|
|
|
{ |
|
502
|
|
|
|
|
|
|
'name' => 'treeWillExpand', |
|
503
|
|
|
|
|
|
|
'type' => 'javax.swing.event.TreeExpansionEvent', |
|
504
|
|
|
|
|
|
|
'throws' => 'javax.swing.tree.ExpandVetoException' |
|
505
|
|
|
|
|
|
|
}, |
|
506
|
|
|
|
|
|
|
{ |
|
507
|
|
|
|
|
|
|
'name' => 'treeWillCollapse', |
|
508
|
|
|
|
|
|
|
'type' => 'javax.swing.event.TreeExpansionEvent', |
|
509
|
|
|
|
|
|
|
'throws' => 'javax.swing.tree.ExpandVetoException' |
|
510
|
|
|
|
|
|
|
} |
|
511
|
|
|
|
|
|
|
], |
|
512
|
|
|
|
|
|
|
'full_name' => 'javax.swing.event.TreeWillExpandListener' |
|
513
|
|
|
|
|
|
|
} |
|
514
|
|
|
|
|
|
|
|
|
515
|
|
|
|
|
|
|
Always use all three keys: |
|
516
|
|
|
|
|
|
|
|
|
517
|
|
|
|
|
|
|
=over 4 |
|
518
|
|
|
|
|
|
|
|
|
519
|
|
|
|
|
|
|
=item full_name |
|
520
|
|
|
|
|
|
|
|
|
521
|
|
|
|
|
|
|
The fully qualified name of the listener interface. |
|
522
|
|
|
|
|
|
|
|
|
523
|
|
|
|
|
|
|
=item listener |
|
524
|
|
|
|
|
|
|
|
|
525
|
|
|
|
|
|
|
The interface name (short form, not qualified). |
|
526
|
|
|
|
|
|
|
|
|
527
|
|
|
|
|
|
|
=item methods |
|
528
|
|
|
|
|
|
|
|
|
529
|
|
|
|
|
|
|
An anonymous list of hash references. Each hash needs two keys: |
|
530
|
|
|
|
|
|
|
|
|
531
|
|
|
|
|
|
|
=over 4 |
|
532
|
|
|
|
|
|
|
|
|
533
|
|
|
|
|
|
|
=item name (required) |
|
534
|
|
|
|
|
|
|
|
|
535
|
|
|
|
|
|
|
The name of the method. |
|
536
|
|
|
|
|
|
|
|
|
537
|
|
|
|
|
|
|
=item type (required) |
|
538
|
|
|
|
|
|
|
|
|
539
|
|
|
|
|
|
|
The type of event the method receives. |
|
540
|
|
|
|
|
|
|
|
|
541
|
|
|
|
|
|
|
=item throws (optional) |
|
542
|
|
|
|
|
|
|
|
|
543
|
|
|
|
|
|
|
The type of event the method throws. |
|
544
|
|
|
|
|
|
|
|
|
545
|
|
|
|
|
|
|
=back |
|
546
|
|
|
|
|
|
|
|
|
547
|
|
|
|
|
|
|
=back |
|
548
|
|
|
|
|
|
|
|
|
549
|
|
|
|
|
|
|
Once you have a file describing your listener, run C |
|
550
|
|
|
|
|
|
|
with that file as the only command line argument. This writes to standard |
|
551
|
|
|
|
|
|
|
out. Save the resulting file as C. When you |
|
552
|
|
|
|
|
|
|
compile the java file, include the Inline classes in the CLASSPATH, these |
|
553
|
|
|
|
|
|
|
will be in a directory like |
|
554
|
|
|
|
|
|
|
|
|
555
|
|
|
|
|
|
|
/cpan/modules/Inline-Java-0.47/Java/classes |
|
556
|
|
|
|
|
|
|
|
|
557
|
|
|
|
|
|
|
(During run time, Inline::Java makes sure these are in the CLASSPATH.) |
|
558
|
|
|
|
|
|
|
|
|
559
|
|
|
|
|
|
|
=head3 Families of Listeners |
|
560
|
|
|
|
|
|
|
|
|
561
|
|
|
|
|
|
|
When adding a whole new toolkit (like swt) you need to add all the listeners. |
|
562
|
|
|
|
|
|
|
To do this use the following steps (all scripts mentioned are in the |
|
563
|
|
|
|
|
|
|
Swing/Generate directory of the distribution): |
|
564
|
|
|
|
|
|
|
|
|
565
|
|
|
|
|
|
|
=over 4 |
|
566
|
|
|
|
|
|
|
|
|
567
|
|
|
|
|
|
|
=item 1 |
|
568
|
|
|
|
|
|
|
|
|
569
|
|
|
|
|
|
|
Use C, giving it the package name and the directory where |
|
570
|
|
|
|
|
|
|
source and class files live. (Note that it relies on a hard coded path |
|
571
|
|
|
|
|
|
|
to your rt.jar, change that to the correct location.) |
|
572
|
|
|
|
|
|
|
(If you don't have sources, you'll have to change the script substantially.) |
|
573
|
|
|
|
|
|
|
The output comes to standard out, store it in a file. |
|
574
|
|
|
|
|
|
|
|
|
575
|
|
|
|
|
|
|
=item 2 |
|
576
|
|
|
|
|
|
|
|
|
577
|
|
|
|
|
|
|
Use C, giving it the name of the file from step one along with |
|
578
|
|
|
|
|
|
|
directory for the java pieces. |
|
579
|
|
|
|
|
|
|
|
|
580
|
|
|
|
|
|
|
=item 3 |
|
581
|
|
|
|
|
|
|
|
|
582
|
|
|
|
|
|
|
Compile the java programs. Remember to include the Inline::Java classes |
|
583
|
|
|
|
|
|
|
in the CLASSPATH for compilation (they are supplied for you at run time). |
|
584
|
|
|
|
|
|
|
On my machine these are in /usr/src/Inline-Java-0.47/Java/classes. |
|
585
|
|
|
|
|
|
|
|
|
586
|
|
|
|
|
|
|
=item 4 |
|
587
|
|
|
|
|
|
|
|
|
588
|
|
|
|
|
|
|
Make sure that the classes from step 3 are in the CLASSPATH for all |
|
589
|
|
|
|
|
|
|
scripts that need them. |
|
590
|
|
|
|
|
|
|
|
|
591
|
|
|
|
|
|
|
=back |
|
592
|
|
|
|
|
|
|
|
|
593
|
|
|
|
|
|
|
=head2 Adding Constant Interfaces |
|
594
|
|
|
|
|
|
|
|
|
595
|
|
|
|
|
|
|
Since C is based on C it is inherently a remote |
|
596
|
|
|
|
|
|
|
procedure call system. Among other things this means that only methods |
|
597
|
|
|
|
|
|
|
can be called from one language to another. Constants cannot be seen. |
|
598
|
|
|
|
|
|
|
Therefore, if you have constants in an interface, or even in a class, |
|
599
|
|
|
|
|
|
|
you must provide methods for these, typically in both java and perl. |
|
600
|
|
|
|
|
|
|
|
|
601
|
|
|
|
|
|
|
To see how to do this, consult C in the java |
|
602
|
|
|
|
|
|
|
directory of the distribution and its pair C in |
|
603
|
|
|
|
|
|
|
the Swing directory. |
|
604
|
|
|
|
|
|
|
|
|
605
|
|
|
|
|
|
|
There is not currently an automated way to build these wrappers. |
|
606
|
|
|
|
|
|
|
|
|
607
|
|
|
|
|
|
|
=head2 EXPORT |
|
608
|
|
|
|
|
|
|
|
|
609
|
|
|
|
|
|
|
None. |
|
610
|
|
|
|
|
|
|
|
|
611
|
|
|
|
|
|
|
=head1 SEE ALSO |
|
612
|
|
|
|
|
|
|
|
|
613
|
|
|
|
|
|
|
The documentation above is, of course, incomplete. It gives you the |
|
614
|
|
|
|
|
|
|
spirit of using the kit. The real documentation is the official |
|
615
|
|
|
|
|
|
|
Java documentation for the version of the jdk you have installed. |
|
616
|
|
|
|
|
|
|
|
|
617
|
|
|
|
|
|
|
Particular Java::Swing:: modules may have additional Perl specific |
|
618
|
|
|
|
|
|
|
documentation. See the Swing directory for these modules. |
|
619
|
|
|
|
|
|
|
|
|
620
|
|
|
|
|
|
|
Questions about this module may be addressed to the author or posted to |
|
621
|
|
|
|
|
|
|
the Inline mailing list to which the author and other interested parties |
|
622
|
|
|
|
|
|
|
subscribe. |
|
623
|
|
|
|
|
|
|
|
|
624
|
|
|
|
|
|
|
=head1 AUTHOR |
|
625
|
|
|
|
|
|
|
|
|
626
|
|
|
|
|
|
|
Phil Crow, Ephilcrow2000@yahoo.comE |
|
627
|
|
|
|
|
|
|
|
|
628
|
|
|
|
|
|
|
=head1 COPYRIGHT AND LICENSE |
|
629
|
|
|
|
|
|
|
|
|
630
|
|
|
|
|
|
|
Copyright 2004 by Philip Crow |
|
631
|
|
|
|
|
|
|
|
|
632
|
|
|
|
|
|
|
This library is free software; you can redistribute it and/or modify |
|
633
|
|
|
|
|
|
|
it under the same terms as Perl 5.8.0 itself. |
|
634
|
|
|
|
|
|
|
|
|
635
|
|
|
|
|
|
|
=cut |
|
636
|
|
|
|
|
|
|
|
|
637
|
|
|
|
|
|
|
__DATA__ |