1 |
#!/usr/bin/perl -I../lib/ |
2 |
|
3 |
#use PApp::Util qw(dumpval); # debug only |
4 |
|
5 |
use Gtk; |
6 |
use Gtk::Gdk; |
7 |
#use Gtk::Gdk::Pixbuf; |
8 |
use Gtk::Gdk::ImlibImage; |
9 |
|
10 |
use Gnome; |
11 |
|
12 |
use KGS::Protocol; |
13 |
use KGS::Listener::Debug; |
14 |
|
15 |
use IO::Socket::INET; |
16 |
|
17 |
use Errno; |
18 |
|
19 |
init Gnome "kgsueme"; |
20 |
|
21 |
$HACK = 1; # do NEVER enable. ;) |
22 |
|
23 |
our $config; |
24 |
our $IMGDIR = "images"; |
25 |
|
26 |
sub load_img { |
27 |
# new_from_file Gtk::Gdk::Pixbuf "$IMGDIR/$_[0]" |
28 |
load_image Gtk::Gdk::ImlibImage "$IMGDIR/$_[0]" |
29 |
or die "$IMGDIR/$_[0]: $!"; |
30 |
} |
31 |
|
32 |
our @black_img = load_img "b01-100x102.png"; |
33 |
our @white_img = map +(load_img "w0$_-100x102_retouched.png"), 1,2,3; |
34 |
our $board_img = load_img "woodgrain-01.jpg"; |
35 |
|
36 |
{ |
37 |
use Storable (); |
38 |
use Scalar::Util (); |
39 |
|
40 |
my $staterc = "$ENV{HOME}/.kgsueme"; |
41 |
|
42 |
my $state = -r $staterc ? Storable::retrieve($staterc) : {}; |
43 |
my @widgets; |
44 |
|
45 |
$config = $state->{config} ||= {}; |
46 |
|
47 |
# grr... more gtk+ brokenness |
48 |
my %get = ( |
49 |
window_size => sub { [ @{$_[0]->allocation}[2,3] ] }, |
50 |
#window_pos => sub { die PApp::Util::dumpval [ $_[0]->get_root_origin ] }, |
51 |
clist_column_widths => sub { |
52 |
$_[0]{column_widths}; |
53 |
}, |
54 |
); |
55 |
|
56 |
my %set = ( |
57 |
hpane_position => sub { $_[0]->set_position($_[1]) }, |
58 |
vpane_position => sub { $_[0]->set_position($_[1]) }, |
59 |
window_size => sub { $_[0]->set_default_size(@{$_[1]}) }, |
60 |
#window_pos => sub { $_[0]->set_uposition(@{$_[1]}) if @{$_[1]} }, |
61 |
clist_column_widths => sub { |
62 |
my ($w, $v) = @_; |
63 |
$v->[$_] && $w->set_column_width($_, $v->[$_]) for 0..$#$v; |
64 |
$w->{column_widths} = $v; |
65 |
$w->signal_connect(resize_column => sub { $v->[$_[1]] = $_[2]; }); |
66 |
}, |
67 |
); |
68 |
|
69 |
sub state { |
70 |
my ($widget, $class, $instance, %attr) = @_; |
71 |
|
72 |
while (my ($k, $v) = each %attr) { |
73 |
my ($set, $get) = $k =~ /=/ ? split /=/, $k : ($k, $k); |
74 |
$v = $state->{$class}{"*"}{$get} if exists $state->{$class}{"*"}{$get}; |
75 |
$v = $state->{$class}{$instance}{$get} if exists $state->{$class}{$instance}{$get}; |
76 |
$set{$get} ? $set{$get}->($widget, $v) : $widget->set($set => $v); |
77 |
} |
78 |
|
79 |
$widget = [$widget, $class, $instance, \%attr]; |
80 |
Scalar::Util::weaken $widget->[0]; |
81 |
|
82 |
@widgets = (grep $_->[0], @widgets, $widget); |
83 |
} |
84 |
|
85 |
sub save_state { |
86 |
for (@widgets) { |
87 |
if ($_->[0]) { |
88 |
my ($widget, $class, $instance, $attr) = @$_; |
89 |
|
90 |
$widget->realize; |
91 |
|
92 |
while (my ($k, $v) = each %$attr) { |
93 |
my ($set, $get) = $k =~ /=/ ? split /=/, $k : ($k, $k); |
94 |
$v = $get{$get} ? $get{$get}->($widget) : $widget->get($get); |
95 |
|
96 |
$state->{$class}{"*"}{$get} = $v; |
97 |
$state->{$class}{$instance}{$get} = $v; |
98 |
} |
99 |
} |
100 |
::status("save_state", "layout saved"); |
101 |
} |
102 |
|
103 |
Storable::nstore($state, $staterc); |
104 |
} |
105 |
} |
106 |
|
107 |
# make a clist unselectable |
108 |
sub clist_autosort { |
109 |
my $w = shift; |
110 |
my ($c, $o) = (-1); |
111 |
for (0..$w->columns-1) { |
112 |
$w->signal_connect(click_column => sub { |
113 |
if ($_[1] != $c) { |
114 |
$c = $_[1]; |
115 |
$o = 0; |
116 |
} else { |
117 |
$o = !$o; |
118 |
} |
119 |
$w->set_sort_column($c); |
120 |
$w->set_sort_type($o ? "descending" : "ascending"); |
121 |
$w->sort; |
122 |
}); |
123 |
} |
124 |
|
125 |
} |
126 |
|
127 |
{ |
128 |
my $main = new kgsueme; |
129 |
|
130 |
my %context_id; |
131 |
|
132 |
sub status { |
133 |
my ($type, $text) = @_; |
134 |
|
135 |
$main->{status}->pop($context_id{$type}) if $context_id{$type}; |
136 |
$main->{status}->push($context_id{$type} ||= $main->{status}->get_context_id($type), $text) if $text; |
137 |
} |
138 |
} |
139 |
|
140 |
main Gtk; |
141 |
|
142 |
############################################################################# |
143 |
|
144 |
package kgsueme; |
145 |
|
146 |
use base KGS::Listener; |
147 |
|
148 |
sub new { |
149 |
my $self = shift; |
150 |
$self = $self->SUPER::new(@_); |
151 |
|
152 |
$self->{conn} = new KGS::Protocol; |
153 |
|
154 |
KGS::Listener::Debug->new->listen($self->{conn}); #d# debug only :) |
155 |
|
156 |
$self->listen($self->{conn}); |
157 |
|
158 |
$self->{roomlist} = new roomlist conn => $self->{conn}; |
159 |
|
160 |
$self->{window} = new Gtk::Window 'toplevel'; |
161 |
$self->{window}->set_title('kgsueme'); |
162 |
::state $self->{window}, "main::window", undef, window_size => [400, 100]; |
163 |
$self->{window}->signal_connect(delete_event => sub { main_quit Gtk; 1 }); |
164 |
|
165 |
$self->{window}->add(my $vbox = new Gtk::VBox); |
166 |
|
167 |
$vbox->pack_start(($buttonbox = new Gtk::HButtonBox), 0, 1, 0); |
168 |
$buttonbox->set_spacing(0); |
169 |
|
170 |
my $button = sub { |
171 |
$buttonbox->add(my $button = new Gtk::Button $_[0]); |
172 |
signal_connect $button clicked => $_[1]; |
173 |
}; |
174 |
|
175 |
$button->("Login", sub { $self->login; }); |
176 |
$button->("Roomlist", sub { $self->{roomlist}->show; }); |
177 |
$button->("Save Config & Layout", sub { ::save_state }); |
178 |
$button->("Quit", sub { main_quit Gtk }); |
179 |
|
180 |
$vbox->pack_start((my $hbox = new Gtk::HBox), 0, 1, 0); |
181 |
|
182 |
$hbox->add(new Gtk::Label "Login"); |
183 |
|
184 |
$hbox->add($self->{login} = new_with_max_length Gtk::Entry 12); |
185 |
$self->{login}->set_text($::config->{login}); |
186 |
|
187 |
if ($::HACK) { |
188 |
$self->{login}->signal_connect(activate => sub { |
189 |
$self->{conn}{name} = $self->{login}->get_text; |
190 |
}); |
191 |
} |
192 |
|
193 |
$hbox->add(new Gtk::Label "Password"); |
194 |
$hbox->add($self->{password} = new Gtk::Entry); |
195 |
$self->{password}->set_visibility(0); |
196 |
|
197 |
$vbox->pack_start(($self->{status} = new Gtk::Statusbar), 0, 1, 0); |
198 |
|
199 |
$self->{window}->show_all; |
200 |
|
201 |
$self; |
202 |
} |
203 |
|
204 |
sub login { |
205 |
my ($self) = @_; |
206 |
|
207 |
$self->{conn}->disconnect; |
208 |
|
209 |
# initialize new socket and connection |
210 |
my $sock = new IO::Socket::INET PeerHost => "kgs.kiseido.com", PeerPort => "2379" |
211 |
or die; |
212 |
|
213 |
$sock->blocking(1); |
214 |
$self->{conn}->handshake($sock); |
215 |
$sock->blocking(0); |
216 |
|
217 |
my $input; $input = input_add Gtk::Gdk fileno $sock, "read", sub { |
218 |
# this is dorked |
219 |
my $buf; |
220 |
if (0 >= sysread $sock, $buf, 16384 |
221 |
and !$!{EINTR} and !$!{EAGAIN}) { |
222 |
input_remove Gtk::Gdk $input; |
223 |
$self->event_disconnect; |
224 |
} |
225 |
$self->{conn}->feed_data($buf); |
226 |
}; |
227 |
|
228 |
# now login |
229 |
$self->{conn}->login($self->{login}->get_text, $self->{password}->get_text); |
230 |
} |
231 |
|
232 |
sub inject_login { |
233 |
my ($self, $msg) = @_; |
234 |
|
235 |
::status("login", "logged in as '$self->{conn}{name}' with status '$msg->{result}'"); |
236 |
$::config->{login} = $self->{conn}{name}; |
237 |
|
238 |
if ($msg->{success}) { |
239 |
for (keys %{$::config->{rooms}}) { |
240 |
$self->{roomlist}->join_room($_); |
241 |
} |
242 |
} |
243 |
|
244 |
warn PApp::Util::dumpval($::config); |
245 |
} |
246 |
|
247 |
sub event_disconnect { } |
248 |
|
249 |
############################################################################# |
250 |
|
251 |
package roomlist; |
252 |
|
253 |
use base KGS::Listener::Roomlist; |
254 |
|
255 |
sub new { |
256 |
my $self = shift; |
257 |
$self = $self->SUPER::new(@_); |
258 |
|
259 |
$self->listen($self->{conn}); |
260 |
|
261 |
$self->{window} = new Gtk::Window 'toplevel'; |
262 |
$self->{window}->set_title('KGS Rooms'); |
263 |
::state $self->{window}, "roomlist::window", undef, window_size => [400, 200]; |
264 |
|
265 |
$self->{window}->signal_connect(delete_event => sub { $self->{window}->hide; 1 }); |
266 |
|
267 |
$self->{window}->add(my $vbox = new Gtk::VBox); |
268 |
|
269 |
$vbox->pack_start((my $sw = new Gtk::ScrolledWindow), 1, 1, 0); |
270 |
$sw->set_policy("automatic", "always"); |
271 |
|
272 |
$sw->add($self->{roomlist} = new_with_titles Gtk::CList "Group", "Room Name", "Users", "Games", "Flags", "Channel"); |
273 |
$self->{roomlist}->set_selection_mode('multiple'); |
274 |
::clist_autosort $self->{roomlist}; |
275 |
::state $self->{roomlist}, "roomlist::roomlist", undef, clist_column_widths => [20, 200]; |
276 |
|
277 |
$self->{roomlist}->signal_connect(select_row => sub { |
278 |
my $room = $self->{roomlist}->get_row_data($_[1]) |
279 |
or return; |
280 |
$self->{roomlist}->unselect_all; |
281 |
$self->join_room($room->{channel}); |
282 |
}); |
283 |
|
284 |
$self; |
285 |
} |
286 |
|
287 |
sub join_room { |
288 |
my ($self, $channel) = @_; |
289 |
|
290 |
$self->{room}{$channel} ||= room->new(channel => $channel, conn => $self->{conn}, users => {}); |
291 |
$self->{room}{$channel}->join; |
292 |
} |
293 |
|
294 |
sub show { |
295 |
my ($self, $msg) = @_; |
296 |
|
297 |
$self->msg(list_rooms => group => $_) for 0..5; # fetch all room names (should not!) |
298 |
$self->{window}->show_all; |
299 |
} |
300 |
|
301 |
sub event_update { |
302 |
my ($self) = @_; |
303 |
|
304 |
$self->{event_update} ||= Gtk->timeout_add(200, sub { |
305 |
my $l = $self->{roomlist}; |
306 |
|
307 |
$l->freeze; |
308 |
my $pos = $l->get_vadjustment->get_value; |
309 |
$l->clear; |
310 |
|
311 |
my $row = 0; |
312 |
for (values %{$self->{rooms}}) { |
313 |
$l->append($_->{group}, $_->{name}, $_->{users}, $_->{games}, $_->{flags}, $_->{channel}); |
314 |
$l->set_row_data($row++, $_); |
315 |
} |
316 |
$l->sort; |
317 |
$l->get_vadjustment->set_value($pos); |
318 |
$l->thaw; |
319 |
|
320 |
delete $self->{event_update}; |
321 |
0; |
322 |
}); |
323 |
} |
324 |
|
325 |
############################################################################# |
326 |
|
327 |
package room; |
328 |
|
329 |
use base KGS::Listener::Room; |
330 |
|
331 |
sub new { |
332 |
my $self = shift; |
333 |
$self = $self->SUPER::new(@_); |
334 |
|
335 |
$self->listen($self->{conn}); |
336 |
|
337 |
$self->{window} = new Gtk::Window 'toplevel'; |
338 |
$self->{window}->set_title("KGS Room $self->{name}"); |
339 |
::state $self->{window}, "room::window", $self->{name}, window_size => [600, 400]; |
340 |
|
341 |
$self->{window}->signal_connect(delete_event => sub { $self->part; 1 }); |
342 |
|
343 |
$self->{window}->add(my $hpane = new Gtk::HPaned); |
344 |
::state $hpane, "room::hpane", $self->{name}, hpane_position => 200; |
345 |
|
346 |
$hpane->add(my $vpane = new Gtk::VPaned); |
347 |
::state $vpane, "room::vpane", $self->{name}, vpane_position => 200; |
348 |
|
349 |
$vpane->add(my $sw = new Gtk::ScrolledWindow); |
350 |
$sw->set_policy("automatic", "always"); |
351 |
|
352 |
$sw->add($self->{gamelist} = new_with_titles Gtk::CList "T", "Black", "White", "Rules", "Notes"); |
353 |
::clist_autosort $self->{gamelist}; |
354 |
::state $self->{gamelist}, "room::gamelist", $self->{name}, clist_column_widths => [20, 120, 120, 120]; |
355 |
|
356 |
$self->{gamelist}->signal_connect(select_row => sub { |
357 |
my $game = $self->{gamelist}->get_row_data($_[1]) |
358 |
or return; |
359 |
$self->{game}{$game->{channel}} ||= new game %$game, conn => $self->{conn}; |
360 |
$self->{game}{$game->{channel}}->join; |
361 |
$self->{gamelist}->unselect_all; |
362 |
}); |
363 |
|
364 |
$vpane->add(my $vbox = new Gtk::VBox); |
365 |
|
366 |
$vbox->pack_start((my $sw = new Gtk::ScrolledWindow), 1, 1, 0); |
367 |
$sw->set_policy("automatic", "always"); |
368 |
|
369 |
$sw->add($self->{text} = new Gtk::Text); |
370 |
|
371 |
$vbox->pack_start(($self->{entry} = new Gtk::Entry), 0, 1, 0); |
372 |
$self->{entry}->signal_connect(activate => sub { |
373 |
my $text = $self->{entry}->get_text; |
374 |
$self->say($text) if $text =~ /\S/; |
375 |
$self->{entry}->set_text(""); |
376 |
}); |
377 |
|
378 |
$hpane->add(my $sw = new Gtk::ScrolledWindow); |
379 |
$sw->set_policy("automatic", "always"); |
380 |
|
381 |
$sw->add($self->{userlist} = new_with_titles Gtk::CList "User", "Rank", "Flags"); |
382 |
::clist_autosort $self->{userlist}; |
383 |
::state $self->{userlist}, "room::userlist", $self->{name}, clist_column_widths => [120, 30]; |
384 |
|
385 |
$self; |
386 |
} |
387 |
|
388 |
sub event_update { |
389 |
my ($self) = @_; |
390 |
|
391 |
$self->{event_update} ||= Gtk->timeout_add(200, sub { |
392 |
my $l = $self->{userlist}; |
393 |
|
394 |
$l->freeze; |
395 |
my $pos = $l->get_vadjustment->get_value; |
396 |
$l->clear; |
397 |
|
398 |
my $row = 0; |
399 |
for (values %{$self->{users}}) { |
400 |
$l->append($_->{name}); |
401 |
$l->set_row_data($row++, $_); |
402 |
} |
403 |
$l->sort; |
404 |
$l->get_vadjustment->set_value($pos); |
405 |
$l->thaw; |
406 |
|
407 |
delete $self->{event_update}; |
408 |
0; |
409 |
}); |
410 |
} |
411 |
|
412 |
sub event_update_games { |
413 |
my ($self) = @_; |
414 |
|
415 |
$self->{event_update_games} ||= Gtk->timeout_add(200, sub { |
416 |
my $l = $self->{gamelist}; |
417 |
|
418 |
$l->freeze; |
419 |
my $pos = $l->get_vadjustment->get_value; |
420 |
$l->clear; |
421 |
|
422 |
my $row = 0; |
423 |
for (values %{$self->{games}}) { |
424 |
$l->append($_->type, $_->user0, $_->user1, $_->rules, $_->notes); |
425 |
$l->set_row_data($row++, $_); |
426 |
} |
427 |
$l->sort; |
428 |
$l->get_vadjustment->set_value($pos); |
429 |
$l->thaw; |
430 |
|
431 |
delete $self->{event_update_games}; |
432 |
0; |
433 |
}); |
434 |
} |
435 |
|
436 |
sub join { |
437 |
my ($self) = @_; |
438 |
$self->SUPER::join; |
439 |
|
440 |
$self->{window}->show_all; |
441 |
} |
442 |
|
443 |
sub part { |
444 |
my ($self) = @_; |
445 |
$self->SUPER::part; |
446 |
|
447 |
delete $::config->{rooms}{$self->{channel}}; |
448 |
$self->{window}->hide_all; |
449 |
$self->event_update; |
450 |
$self->event_update_games; |
451 |
} |
452 |
|
453 |
sub event_join { |
454 |
my ($self) = @_; |
455 |
$self->SUPER::event_join; |
456 |
|
457 |
$::config->{rooms}{$self->{channel}} = 1; |
458 |
} |
459 |
|
460 |
sub event_update_roominfo { |
461 |
my ($self) = @_; |
462 |
|
463 |
$self->{text}->insert(undef, undef, undef, "\n$self->{owner}: $self->{description}\n"); |
464 |
} |
465 |
|
466 |
sub inject_msg_room { |
467 |
my ($self, $msg) = @_; |
468 |
return unless $self->{channel} == $msg->{channel}; |
469 |
|
470 |
$self->{text}->insert(undef, undef, undef, "\n$msg->{name}: $msg->{message}"); |
471 |
} |
472 |
|
473 |
############################################################################# |
474 |
|
475 |
package game; |
476 |
|
477 |
use KGS::Constants; |
478 |
|
479 |
use base KGS::Listener::Game; |
480 |
use base KGS::Game; |
481 |
|
482 |
sub new { |
483 |
my $self = shift; |
484 |
$self = $self->SUPER::new(@_); |
485 |
|
486 |
$self->listen($self->{conn}); |
487 |
|
488 |
$self->{window} = new Gtk::Window 'toplevel'; |
489 |
$self->{window}->set_title("KGS Game ".$self->user0." ".$self->user1); |
490 |
::state $self->{window}, "game::window", undef, window_size => [600, 500]; |
491 |
|
492 |
$self->{window}->signal_connect(delete_event => sub { $self->part; 1 }); |
493 |
|
494 |
$self->{window}->add(my $hpane = new Gtk::HPaned); |
495 |
::state $hpane, "game::hpane", undef, hpane_position => 500; |
496 |
|
497 |
$::config{aa}++; |
498 |
$self->{canvas} = $::config{aa} ? new_aa Gnome::Canvas : new Gnome::Canvas; |
499 |
$hpane->add($self->{canvas}); |
500 |
|
501 |
$self->{canvas}->signal_connect(size_allocate => sub { |
502 |
warn "@_ @{$_[1]}"; |
503 |
#$self->{canvas}->set_pixels_per_unit( |
504 |
}); |
505 |
|
506 |
my $line_colour = $::config{line_colour} || "darkbrown"; |
507 |
my $border = 0.1; |
508 |
|
509 |
$self->{canvas}->set_pixels_per_unit(300); |
510 |
$self->{canvas}->set_scroll_region(-$border,-$border,1+$border,1+$border); |
511 |
|
512 |
{ |
513 |
my $croot = $self->{canvas}->root; |
514 |
my $cgroup = $croot; #->new($croot, "Gnome::CanvasGroup"); |
515 |
|
516 |
if ($::board_img) { |
517 |
$cgroup->new($cgroup, "Gnome::CanvasImage", |
518 |
x => -$border, y => -$border, |
519 |
width => 1+$border*2, height => 1+$border*2, |
520 |
image => $::board_img, |
521 |
anchor => "nw" |
522 |
); |
523 |
} else { |
524 |
die; |
525 |
$cgroup->new($cgroup, "Gnome::CanvasRect", |
526 |
x1 => -$border, x2 => 1+$border, |
527 |
y1 => -$border, y2 => 1+$border, |
528 |
outline_color => "black", |
529 |
fill_color => "brown", |
530 |
width_pixels => 2, |
531 |
); |
532 |
} |
533 |
|
534 |
my $x1 = $pad * 2; # == my $y1 = 30; |
535 |
my $x2 = $w - ($pad * 2); # == my $y2 = 270; |
536 |
my $w = $x2 - $x1; |
537 |
my $s = $self->{size} - 1; |
538 |
my $ofs = 0.5 / $self->{size}; |
539 |
|
540 |
my $a = "A"; |
541 |
for my $i (1 .. $self->{size}) { # one more iteration for the last lines |
542 |
my $k = $i / $self->{size} - $ofs; |
543 |
|
544 |
$cgroup->new($cgroup, "Gnome::CanvasLine", |
545 |
points => [ $k,$ofs, $k,1-$ofs ], |
546 |
fill_color => $line_colour, |
547 |
width_pixels => 1); |
548 |
|
549 |
$cgroup->new($cgroup, "Gnome::CanvasLine", |
550 |
points => [ $ofs,$k, 1-$ofs,$k ], |
551 |
fill_color => $line_colour, |
552 |
width_pixels => 1); |
553 |
|
554 |
for ( [$k,-$border*0.5,$a], [$k,1+$border*0.5,$a], [-$border*0.5,$k,$i], [1+$border*0.5,$k,$i] ) { |
555 |
my ($x, $y, $text) = @$_; |
556 |
|
557 |
my $text = $cgroup->new($cgroup, "Gnome::CanvasText", |
558 |
x => 0, y => 0, text => $text, |
559 |
justification => "center", |
560 |
anchor => "center", |
561 |
font => "-*-helvetica-medium-r-*--34-*", |
562 |
fill_color => $line_colour); |
563 |
|
564 |
$text->affine_relative($border*0.015,0, 0,$border*0.015, $x,$y); |
565 |
} |
566 |
|
567 |
$a++; |
568 |
} |
569 |
|
570 |
my $stones = [[]]; |
571 |
|
572 |
for my $x (1 .. $self->{size}) { |
573 |
my $xk = $x / $self->{size} - $ofs; |
574 |
for my $y (1 .. $self->{size}) { |
575 |
my $yk = $y / $self->{size} - $ofs; |
576 |
|
577 |
my $col = $stones->[$x-1][$y-1] = []; |
578 |
|
579 |
if (1) { |
580 |
$col->[0] = |
581 |
$cgroup->new($cgroup, "Gnome::CanvasImage", |
582 |
x => $xk, y => $yk, |
583 |
width => $ofs*2, height => $ofs*2, |
584 |
image => $::black_img[int rand @::black_img], |
585 |
anchor => "center"); |
586 |
$col->[1] = |
587 |
$cgroup->new($cgroup, "Gnome::CanvasImage", |
588 |
x => $xk, y => $yk, |
589 |
width => $ofs*2, height => $ofs*2, |
590 |
image => $::white_img[int rand @::white_img], |
591 |
anchor => "center"); |
592 |
} else { |
593 |
# too large, scale has NO effect |
594 |
$col->[0] = |
595 |
$cgroup->new($cgroup, "Gnome::CanvasEllipse", |
596 |
x1 => $xk, x2 => $xk + 0.001, |
597 |
y1 => $yk, y2 => $yk + 0.001, |
598 |
outline_color => "black", width_pixels => 2, |
599 |
fill_color => "black"); |
600 |
$col->[1] = |
601 |
$cgroup->new($cgroup, "Gnome::CanvasEllipse", |
602 |
x1 => $xk, x2 => $xk + 0.001, |
603 |
y1 => $yk, y2 => $yk + 0.001, |
604 |
outline_color => "black", width_pixels => 2, |
605 |
fill_color => "white"); |
606 |
} |
607 |
} |
608 |
} |
609 |
|
610 |
$self->{board_gfx}->{stones} = $stones; |
611 |
} |
612 |
|
613 |
$hpane->add(my $vpane = new Gtk::VPaned); |
614 |
::state $vpane, "game", $self->{name}, vpane_position => 80; |
615 |
|
616 |
$vpane->add(my $sw = new Gtk::ScrolledWindow); |
617 |
$sw->set_policy("automatic", "always"); |
618 |
|
619 |
$sw->add($self->{userlist} = new_with_titles Gtk::CList "User", "Rank", "Flags"); |
620 |
::clist_autosort $self->{userlist}; |
621 |
::state $self->{userlist}, "room::userlist", $self->{name}, clist_column_widths => [120, 30]; |
622 |
|
623 |
$vpane->add(my $vbox = new Gtk::VBox); |
624 |
|
625 |
$vbox->pack_start((my $sw = new Gtk::ScrolledWindow), 1, 1, 0); |
626 |
$sw->set_policy("automatic", "always"); |
627 |
|
628 |
$sw->add($self->{text} = new Gtk::Text); |
629 |
|
630 |
$vbox->pack_start(($self->{entry} = new Gtk::Entry), 0, 1, 0); |
631 |
$self->{entry}->signal_connect(activate => sub { |
632 |
my $text = $self->{entry}->get_text; |
633 |
# add message |
634 |
$self->{entry}->set_text(""); |
635 |
}); |
636 |
|
637 |
$self; |
638 |
} |
639 |
|
640 |
sub event_update { |
641 |
my ($self) = @_; |
642 |
|
643 |
$self->{event_update} ||= Gtk->timeout_add(200, sub { |
644 |
my $l = $self->{userlist}; |
645 |
|
646 |
$l->freeze; |
647 |
my $pos = $l->get_vadjustment->get_value; |
648 |
$l->clear; |
649 |
|
650 |
my $row = 0; |
651 |
for (values %{$self->{users}}) { |
652 |
$l->append($_->{name}); |
653 |
$l->set_row_data($row++, $_); |
654 |
} |
655 |
$l->sort; |
656 |
$l->get_vadjustment->set_value($pos); |
657 |
$l->thaw; |
658 |
|
659 |
delete $self->{event_update}; |
660 |
0; |
661 |
}); |
662 |
} |
663 |
|
664 |
sub join { |
665 |
my ($self) = @_; |
666 |
$self->SUPER::join; |
667 |
|
668 |
$self->{window}->show_all; |
669 |
} |
670 |
|
671 |
sub part { |
672 |
my ($self) = @_; |
673 |
$self->SUPER::part; |
674 |
|
675 |
$self->{window}->hide_all; |
676 |
$self->event_update; |
677 |
} |
678 |
|
679 |
sub event_update_tree { |
680 |
my ($self) = @_; |
681 |
|
682 |
my $board = new KGS::Game::Board $self->{size}; |
683 |
$board->interpret_path ($self->get_path); |
684 |
|
685 |
# if (not defined $self->{board_pm}) { |
686 |
# $self->{board_pm} = new Gtk::Gdk::Pixmap ($self->{board}->window, 100, 100, -1); |
687 |
# $self->{board_pm}->draw_rectangle($self->{board}->style->white_gc, 1, 0, 0, 100, 100); |
688 |
# } |
689 |
# my $red = $self->{board}->window->get_colormap->color_alloc( { red => 65000, green => 0, blue => 0 } ); |
690 |
# my $red_gc = new Gtk::Gdk::GC ( $self->{board}->window ); |
691 |
# $red_gc->set_foreground( $red ); |
692 |
# |
693 |
# my $px = $self->{board_pm}; |
694 |
|
695 |
for my $x (0 .. $self->{size} - 1) { |
696 |
for my $y (0 .. $self->{size} - 1) { |
697 |
my $v = $board->{board}[$x][$y]; |
698 |
|
699 |
$v & MARK_B |
700 |
? $self->{board_gfx}->{stones}->[$x][$y][0]->show |
701 |
: $self->{board_gfx}->{stones}->[$x][$y][0]->hide; |
702 |
$v & MARK_W |
703 |
? $self->{board_gfx}->{stones}->[$x][$y][1]->show |
704 |
: $self->{board_gfx}->{stones}->[$x][$y][1]->hide; |
705 |
} |
706 |
} |
707 |
} |
708 |
|
709 |
1; |
710 |
|
711 |
|
712 |
|