ViewVC Help
View File | Revision Log | Show Annotations | Download File
/cvs/deliantra/Deliantra-Client/DC/UI.pm
(Generate patch)

Comparing deliantra/Deliantra-Client/DC/UI.pm (file contents):
Revision 1.249 by root, Sun May 28 01:40:02 2006 UTC vs.
Revision 1.271 by root, Fri Jun 2 22:13:47 2006 UTC

17our $BUTTON_STATE; 17our $BUTTON_STATE;
18 18
19our %WIDGET; # all widgets, weak-referenced 19our %WIDGET; # all widgets, weak-referenced
20 20
21sub get_layout { 21sub get_layout {
22 my $layout;
23
22 for (grep { $_->{name} } values %WIDGET) { 24 for (grep { $_->{name} } values %WIDGET) {
23 $LAYOUT->{$_->{name}} = { 25 my $win = $layout->{$_->{name}} = { };
24 x => $_->{x} / $::WIDTH,
25 y => $_->{y} / $::HEIGHT,
26 w => $_->{w} / $::WIDTH,
27 h => $_->{h} / $::HEIGHT
28 }; 26
29 } 27 $win->{x} = ($_->{x} + $_->{w} * 0.5) / $::WIDTH if $_->{x} =~ /^[0-9.]+$/;
28 $win->{y} = ($_->{y} + $_->{h} * 0.5) / $::HEIGHT if $_->{y} =~ /^[0-9.]+$/;
29 $win->{w} = $_->{w} / $::WIDTH if defined $_->{w};
30 $win->{h} = $_->{h} / $::HEIGHT if defined $_->{h};
30 31
31 return $LAYOUT; 32 $win->{show} = $_->{visible} && $_->{is_toplevel};
33 }
34
35 $layout
32} 36}
33 37
34sub set_layout { 38sub set_layout {
35 my ($layout) = @_; 39 my ($layout) = @_;
40
36 $LAYOUT = $layout; 41 $LAYOUT = $layout;
37} 42}
38 43
39sub check_tooltip { 44sub check_tooltip {
45 return if $ENV{CFPLUS_DEBUG} & 8;
46
40 if (!$GRAB) { 47 if (!$GRAB) {
41 for (my $widget = $HOVER; $widget; $widget = $widget->{parent}) { 48 for (my $widget = $HOVER; $widget; $widget = $widget->{parent}) {
42 if (length $widget->{tooltip}) { 49 if (length $widget->{tooltip}) {
43
44 if ($TOOLTIP->{owner} != $widget) { 50 if ($TOOLTIP->{owner} != $widget) {
51 $TOOLTIP->hide;
52
45 $TOOLTIP->{owner} = $widget; 53 $TOOLTIP->{owner} = $widget;
46 54
47 my $tip = $widget->{tooltip}; 55 my $tip = $widget->{tooltip};
48 56
49 $tip = $tip->($widget) if CODE:: eq ref $tip; 57 $tip = $tip->($widget) if CODE:: eq ref $tip;
50 58
51 $TOOLTIP->set_tooltip_from ($widget); 59 $TOOLTIP->set_tooltip_from ($widget);
52 $TOOLTIP->show; 60 $TOOLTIP->show;
53
54 my ($x, $y) = $widget->coord2global ($widget->{w}, 0);
55
56 ($x, $y) = $widget->coord2global (-$TOOLTIP->{w}, 0)
57 if $x + $TOOLTIP->{w} > $::WIDTH;
58
59 $TOOLTIP->move ($x, $y);
60 $TOOLTIP->check_size;
61 $TOOLTIP->update;
62 } 61 }
63 62
64 return; 63 return;
65 } 64 }
66 } 65 }
172sub rescale_widgets { 171sub rescale_widgets {
173 my ($sx, $sy) = @_; 172 my ($sx, $sy) = @_;
174 173
175 for my $widget (values %WIDGET) { 174 for my $widget (values %WIDGET) {
176 if ($widget->{is_toplevel}) { 175 if ($widget->{is_toplevel}) {
176 $widget->{x} += int $widget->{w} * 0.5 if $widget->{x} =~ /^[0-9.]+$/;
177 $widget->{y} += int $widget->{h} * 0.5 if $widget->{y} =~ /^[0-9.]+$/;
178
177 $widget->{x} = int 0.5 + $widget->{x} * $sx if exists $widget->{x}; 179 $widget->{x} = int 0.5 + $widget->{x} * $sx if $widget->{x} =~ /^[0-9.]+$/;
178 $widget->{w} = int 0.5 + $widget->{w} * $sx if exists $widget->{w}; 180 $widget->{w} = int 0.5 + $widget->{w} * $sx if exists $widget->{w};
179 $widget->{req_w} = int 0.5 + $widget->{req_w} * $sx if exists $widget->{req_w}; 181 $widget->{force_w} = int 0.5 + $widget->{force_w} * $sx if exists $widget->{force_w};
180 $widget->{y} = int 0.5 + $widget->{y} * $sy if exists $widget->{y}; 182 $widget->{y} = int 0.5 + $widget->{y} * $sy if $widget->{y} =~ /^[0-9.]+$/;
181 $widget->{h} = int 0.5 + $widget->{h} * $sy if exists $widget->{h}; 183 $widget->{h} = int 0.5 + $widget->{h} * $sy if exists $widget->{h};
182 $widget->{req_h} = int 0.5 + $widget->{req_h} * $sy if exists $widget->{req_h}; 184 $widget->{force_h} = int 0.5 + $widget->{force_h} * $sy if exists $widget->{force_h};
185
186 $widget->{x} -= int $widget->{w} * 0.5 if $widget->{x} =~ /^[0-9.]+$/;
187 $widget->{y} -= int $widget->{h} * 0.5 if $widget->{y} =~ /^[0-9.]+$/;
188
183 } 189 }
184 } 190 }
185 191
186 reconfigure_widgets; 192 reconfigure_widgets;
187} 193}
196 202
197sub new { 203sub new {
198 my $class = shift; 204 my $class = shift;
199 205
200 my $self = bless { 206 my $self = bless {
201 x => 0, 207 x => "center",
202 y => 0, 208 y => "center",
203 z => 0, 209 z => 0,
210 w => undef,
211 h => undef,
204 can_events => 1, 212 can_events => 1,
205 @_ 213 @_
206 }, $class; 214 }, $class;
215
216 Scalar::Util::weaken ($CFClient::UI::WIDGET{$self+0} = $self);
207 217
208 for (keys %$self) { 218 for (keys %$self) {
209 if (/^on_(.*)$/) { 219 if (/^on_(.*)$/) {
210 $self->connect ($1 => delete $self->{$_}); 220 $self->connect ($1 => delete $self->{$_});
211 } 221 }
212 } 222 }
213 223
214 Scalar::Util::weaken ($CFClient::UI::WIDGET{$self+0} = $self);
215
216 if (my $layout = $CFClient::UI::LAYOUT->{$self->{name}}) { 224 if (my $layout = $CFClient::UI::LAYOUT->{$self->{name}}) {
217 $self->{req_x} = $layout->{x} * $::WIDTH; 225 $self->{x} = $layout->{x} * $CFClient::UI::ROOT->{alloc_w} if exists $layout->{x};
218 $self->{req_y} = $layout->{y} * $::HEIGHT; 226 $self->{y} = $layout->{y} * $CFClient::UI::ROOT->{alloc_h} if exists $layout->{y};
219 $self->{def_w} = ($layout->{w} != 0 ? $layout->{w} : 1) * $::WIDTH; 227 $self->{force_w} = $layout->{w} * $CFClient::UI::ROOT->{alloc_w} if exists $layout->{w};
220 $self->{def_h} = ($layout->{h} != 0 ? $layout->{h} : 1) * $::HEIGHT; 228 $self->{force_h} = $layout->{h} * $CFClient::UI::ROOT->{alloc_h} if exists $layout->{h};
229
230 $self->{x} -= $self->{force_w} * 0.5 if exists $layout->{x};
231 $self->{y} -= $self->{force_h} * 0.5 if exists $layout->{y};
232
233 $self->show if $layout->{show};
221 } 234 }
222 235
223 $self 236 $self
224}
225
226sub toggle_visibility {
227 my ($self) = @_;
228
229 if ($self->{visible}) {
230 $self->hide;
231 } else {
232 $self->show;
233 }
234} 237}
235 238
236sub destroy { 239sub destroy {
237 my ($self) = @_; 240 my ($self) = @_;
238 241
246 return if $self->{parent}; 249 return if $self->{parent};
247 250
248 $CFClient::UI::ROOT->add ($self); 251 $CFClient::UI::ROOT->add ($self);
249} 252}
250 253
251sub center {
252 my ($self) = @_;
253
254 $CFClient::UI::ROOT->on_post_alloc (
255 "center_$self" => sub {
256 $self->move (($self->{parent}{w} - $self->{w}) * 0.5, ($self->{parent}{h} - $self->{h}) * 0.5);
257 },
258 );
259
260 $self->update;
261}
262
263sub set_visible { 254sub set_visible {
264 my ($self) = @_; 255 my ($self) = @_;
265 256
266 return if $self->{visible}; 257 return if $self->{visible};
267 258
268 $self->{root} = $self->{parent}{root}; 259 $self->{root} = $self->{parent}{root};
269 $self->{visible} = $self->{parent}{visible} + 1; 260 $self->{visible} = $self->{parent}{visible} + 1;
270 261
271 $self->emit (visibility_change => 1); 262 $self->emit (visibility_change => 1);
263
264 $self->realloc if !exists $self->{req_w};
265
266 $_->set_visible for $self->children;
272} 267}
273 268
274sub set_invisible { 269sub set_invisible {
275 my ($self) = @_; 270 my ($self) = @_;
276 271
277 return unless $self->{visible}; 272 return unless $self->{visible};
278 273
279 # broken show/hide model 274 $_->set_invisible for $self->children;
280 275
281 delete $self->{root}; 276 delete $self->{root};
282 delete $self->{visible}; 277 delete $self->{visible};
283 278
284 undef $GRAB if $GRAB == $self; 279 undef $GRAB if $GRAB == $self;
285 undef $HOVER if $HOVER == $self; 280 undef $HOVER if $HOVER == $self;
286 281
287 CFClient::UI::check_tooltip 282 CFClient::UI::check_tooltip
288 if $CFClient::UI::TOOLTIP->{owner} == $self; 283 if $TOOLTIP->{owner} == $self;
289 284
290 $self->focus_out; 285 $self->focus_out;
291 286
292 $self->emit (visibility_change => 0); 287 $self->emit (visibility_change => 0);
288}
289
290sub set_visibility {
291 my ($self, $visible) = @_;
292
293 return if $self->{visible} == $visible;
294
295 $visible ? $self->hide
296 : $self->show;
297}
298
299sub toggle_visibility {
300 my ($self) = @_;
301
302 $self->{visible}
303 ? $self->hide
304 : $self->show;
293} 305}
294 306
295sub hide { 307sub hide {
296 my ($self) = @_; 308 my ($self) = @_;
297 309
299 311
300 $self->{parent}->remove ($self) 312 $self->{parent}->remove ($self)
301 if $self->{parent}; 313 if $self->{parent};
302} 314}
303 315
304sub move { 316sub move_abs {
305 my ($self, $x, $y, $z) = @_; 317 my ($self, $x, $y, $z) = @_;
306 318
307 $self->{x} = int $x; 319 $self->{x} = List::Util::max 0, int $x;
308 $self->{y} = int $y; 320 $self->{y} = List::Util::max 0, int $y;
309 $self->{z} = $z if defined $z; 321 $self->{z} = $z if defined $z;
310 322
311 $self->update; 323 $self->update;
312} 324}
313 325
314sub set_size { 326sub set_size {
315 my ($self, $w, $h) = @_; 327 my ($self, $w, $h) = @_;
316 328
317 $self->{def_w} = $w; 329 $self->{force_w} = $w;
318 $self->{def_h} = $h; 330 $self->{force_h} = $h;
319 331
320 $self->check_size; 332 $self->realloc;
321} 333}
322 334
323sub size_request { 335sub size_request {
324 require Carp; 336 require Carp;
325 Carp::confess "size_request is abstract"; 337 Carp::confess "size_request is abstract";
327 339
328sub configure { 340sub configure {
329 my ($self, $x, $y, $w, $h) = @_; 341 my ($self, $x, $y, $w, $h) = @_;
330 342
331 if ($self->{aspect}) { 343 if ($self->{aspect}) {
344 my ($ow, $oh) = ($w, $h);
345
332 my $w2 = List::Util::min $w, int $h * $self->{aspect}; 346 $w = List::Util::min $w, int $h * $self->{aspect};
333 my $h2 = List::Util::min $h, int $w / $self->{aspect}; 347 $h = List::Util::min $h, int $w / $self->{aspect};
334 348
335 # use alignment to adjust x, y 349 # use alignment to adjust x, y
336 350
337 $x += int +($w - $w2) * 0.5; 351 $x += int 0.5 * ($ow - $w);
338 $y += int +($h - $h2) * 0.5; 352 $y += int 0.5 * ($oh - $h);
339
340 ($w, $h) = ($w2, $h2);
341 } 353 }
342 354
343 if ($self->{x} != $x || $self->{y} != $y) { 355 if ($self->{x} ne $x || $self->{y} ne $y) {
344 $self->{x} = $x; 356 $self->{x} = $x;
345 $self->{y} = $y; 357 $self->{y} = $y;
346 $self->update; 358 $self->update;
347 } 359 }
348 360
349 if ($self->{w} != $w || $self->{h} != $h) { 361 if ($self->{alloc_w} != $w || $self->{alloc_h} != $h) {
350 $CFClient::UI::ROOT->{size_alloc}{$self} = [$self, $w, $h]; 362 return unless $self->{visible};
363
364 $self->{alloc_w} = $w;
365 $self->{alloc_h} = $h;
366
367 $self->{root}{size_alloc}{$self+0} = $self;
351 } 368 }
352} 369}
353 370
354sub size_allocate { 371sub size_allocate {
355 # nothing to be done 372 # nothing to be done
356}
357
358sub reconfigure {
359 my ($self) = @_;
360
361 $self->check_size (1);
362 $self->update;
363} 373}
364 374
365sub children { 375sub children {
366} 376}
367 377
429 439
430 $::MAPWIDGET->focus_in #d# focus mapwidget if no other widget has focus 440 $::MAPWIDGET->focus_in #d# focus mapwidget if no other widget has focus
431 unless $FOCUS; 441 unless $FOCUS;
432} 442}
433 443
434sub mouse_motion { } 444sub mouse_motion { 0 }
435sub button_up { } 445sub button_up { 0 }
436sub key_down { } 446sub key_down { 0 }
437sub key_up { } 447sub key_up { 0 }
438 448
439sub button_down { 449sub button_down {
440 my ($self, $ev, $x, $y) = @_; 450 my ($self, $ev, $x, $y) = @_;
441 451
442 $self->focus_in; 452 $self->focus_in;
443}
444 453
445sub w { $_[0]{w} = $_[1] if @_ > 1; $_[0]{w} } 454 0
446sub h { $_[0]{h} = $_[1] if @_ > 1; $_[0]{h} } 455}
447sub x { $_[0]{x} = $_[1] if @_ > 1; $_[0]{x} } 456
448sub y { $_[0]{y} = $_[1] if @_ > 1; $_[0]{y} } 457sub find_widget {
449sub z { $_[0]{z} = $_[1] if @_ > 1; $_[0]{z} } 458 my ($self, $x, $y) = @_;
459
460 return () unless $self->{can_events};
461
462 return $self
463 if $x >= $self->{x} && $x < $self->{x} + $self->{w}
464 && $y >= $self->{y} && $y < $self->{y} + $self->{h};
465
466 ()
467}
468
469sub set_parent {
470 my ($self, $parent) = @_;
471
472 Scalar::Util::weaken ($self->{parent} = $parent);
473 $self->set_visible if $parent->{visible};
474}
475
476sub connect {
477 my ($self, $signal, $cb) = @_;
478
479 push @{ $self->{signal_cb}{$signal} }, $cb;
480}
481
482sub _emit {
483 my ($self, $signal, @args) = @_;
484
485 List::Util::sum map $_->($self, @args), @{$self->{signal_cb}{$signal} || []}
486}
487
488sub emit {
489 my ($self, $signal, @args) = @_;
490
491 $self->_emit ($signal, @args)
492 || $self->$signal (@args);
493}
494
495sub visibility_change {
496 #my ($self, $visible) = @_;
497}
498
499sub realloc {
500 my ($self) = @_;
501
502 if ($self->{visible}) {
503 return if $self->{root}{realloc}{$self+0};
504
505 $self->{root}{realloc}{$self+0} = $self;
506 $self->{root}->update;
507 } else {
508 delete $self->{req_w};
509 delete $self->{req_h};
510 }
511}
512
513sub update {
514 my ($self) = @_;
515
516 $self->{parent}->update
517 if $self->{parent};
518}
519
520sub reconfigure {
521 my ($self) = @_;
522
523 $self->realloc;
524 $self->update;
525}
526
527# using global variables seems a bit hacky, but passing through all drawing
528# functions seems pointless.
529our ($draw_x, $draw_y, $draw_w, $draw_h); # screen rectangle being drawn
450 530
451sub draw { 531sub draw {
452 my ($self) = @_; 532 my ($self) = @_;
453 533
454 return unless $self->{h} && $self->{w}; 534 return unless $self->{h} && $self->{w};
535
536 # update screen rectangle
537 local $draw_x = $draw_x + $self->{x};
538 local $draw_y = $draw_y + $self->{y};
539 local $draw_w = $draw_x + $self->{w};
540 local $draw_h = $draw_y + $self->{h};
541
542 # skip widgets that are entirely outside the drawing area
543 return if ($draw_x + $self->{w} < 0) || ($draw_x >= $draw_w)
544 || ($draw_y + $self->{h} < 0) || ($draw_y >= $draw_h);
455 545
456 glPushMatrix; 546 glPushMatrix;
457 glTranslate $self->{x}, $self->{y}, 0; 547 glTranslate $self->{x}, $self->{y}, 0;
458 $self->_draw; 548 $self->_draw;
459 glPopMatrix; 549 glPopMatrix;
471 glVertex $x , $y + $self->{h}; 561 glVertex $x , $y + $self->{h};
472 glEnd; 562 glEnd;
473 glDisable GL_BLEND; 563 glDisable GL_BLEND;
474 } 564 }
475 565
476 if ($ENV{PCLIENT_DEBUG}) { 566 if ($ENV{CFPLUS_DEBUG} & 1) {
477 glPushMatrix; 567 glPushMatrix;
478 glColor 1, 1, 0, 1; 568 glColor 1, 1, 0, 1;
479 glTranslate $self->{x} + 0.375, $self->{y} + 0.375; 569 glTranslate $self->{x} + 0.375, $self->{y} + 0.375;
480 glBegin GL_LINE_LOOP; 570 glBegin GL_LINE_LOOP;
481 glVertex 0 , 0; 571 glVertex 0 , 0;
492 my ($self) = @_; 582 my ($self) = @_;
493 583
494 warn "no draw defined for $self\n"; 584 warn "no draw defined for $self\n";
495} 585}
496 586
497sub find_widget {
498 my ($self, $x, $y) = @_;
499
500 return () unless $self->{can_events};
501
502 return $self
503 if $x >= $self->{x} && $x < $self->{x} + $self->{w}
504 && $y >= $self->{y} && $y < $self->{y} + $self->{h};
505
506 ()
507}
508
509sub set_parent {
510 my ($self, $parent) = @_;
511
512 Scalar::Util::weaken ($self->{parent} = $parent);
513
514 $self->set_visible; #TODO why breakssssss borked damn if $parent->{visible};
515
516 $self->check_size;
517}
518
519sub check_size {
520 my ($self, $forced) = @_;
521
522 $self->{force_alloc} = 1 if $forced;
523 $CFClient::UI::ROOT->{check_size}{$self} = $self;
524}
525
526sub update {
527 my ($self) = @_;
528
529 $self->{parent}->update
530 if $self->{parent};
531}
532
533sub connect {
534 my ($self, $signal, $cb) = @_;
535
536 push @{ $self->{signal_cb}{$signal} }, $cb;
537}
538
539sub _emit {
540 my ($self, $signal, @args) = @_;
541
542 List::Util::sum map $_->($self, @args), @{$self->{signal_cb}{$signal} || []}
543}
544
545sub emit {
546 my ($self, $signal, @args) = @_;
547
548 $self->_emit ($signal, @args)
549 || $self->$signal (@args);
550}
551
552sub visibility_change {
553 #my ($self, $visible) = @_;
554}
555
556sub DESTROY { 587sub DESTROY {
557 my ($self) = @_; 588 my ($self) = @_;
558 589
559 delete $WIDGET{$self+0}; 590 delete $WIDGET{$self+0};
560 #$self->deactivate; 591 #$self->deactivate;
616 my ($class, %arg) = @_; 647 my ($class, %arg) = @_;
617 $class->SUPER::new (can_events => 0, %arg); 648 $class->SUPER::new (can_events => 0, %arg);
618} 649}
619 650
620sub size_request { 651sub size_request {
621 (0, 0) 652 my ($self) = @_;
653
654 ($self->{w} + 0, $self->{h} + 0)
622} 655}
623 656
624sub draw { } 657sub draw { }
625 658
626############################################################################# 659#############################################################################
655 $self->{children} = [ 688 $self->{children} = [
656 sort { $a->{z} <=> $b->{z} } 689 sort { $a->{z} <=> $b->{z} }
657 @{$self->{children}}, @widgets 690 @{$self->{children}}, @widgets
658 ]; 691 ];
659 692
660 $self->check_size (1); 693 $self->realloc;
661 $self->update;
662} 694}
663 695
664sub children { 696sub children {
665 @{ $_[0]{children} } 697 @{ $_[0]{children} }
666} 698}
671 delete $child->{parent}; 703 delete $child->{parent};
672 $child->hide; 704 $child->hide;
673 705
674 $self->{children} = [ grep $_ != $child, @{ $self->{children} } ]; 706 $self->{children} = [ grep $_ != $child, @{ $self->{children} } ];
675 707
676 $self->check_size (1); 708 $self->realloc;
677 $self->update;
678} 709}
679 710
680sub clear { 711sub clear {
681 my ($self) = @_; 712 my ($self) = @_;
682 713
686 for (@$children) { 717 for (@$children) {
687 delete $_->{parent}; 718 delete $_->{parent};
688 $_->hide; 719 $_->hide;
689 } 720 }
690 721
691 $self->check_size; 722 $self->realloc;
692 $self->update;
693} 723}
694 724
695sub find_widget { 725sub find_widget {
696 my ($self, $x, $y) = @_; 726 my ($self, $x, $y) = @_;
697 727
784 $self->SUPER::size_allocate ($w, $h); 814 $self->SUPER::size_allocate ($w, $h);
785 $self->update; 815 $self->update;
786} 816}
787 817
788sub _render { 818sub _render {
819 my ($self) = @_;
820
789 $_[0]{children}[0]->draw; 821 $self->{children}[0]->draw;
790} 822}
791 823
792sub render_child { 824sub render_child {
793 my ($self) = @_; 825 my ($self) = @_;
794 826
795 $self->{texture} = new_from_opengl CFClient::Texture $self->{w}, $self->{h}, sub { 827 $self->{texture} = new_from_opengl CFClient::Texture $self->{w}, $self->{h}, sub {
796 glClearColor 0, 0, 0, 0; 828 glClearColor 0, 0, 0, 0;
797 glClear GL_COLOR_BUFFER_BIT; 829 glClear GL_COLOR_BUFFER_BIT;
798 830
831 {
832 package CFClient::UI::Base;
833
834 ($draw_x, $draw_y, $draw_w, $draw_h) =
835 (0, 0, $self->{w}, $self->{h});
836 }
837
799 $self->_render; 838 $self->_render;
800 }; 839 };
801} 840}
802 841
803sub _draw { 842sub _draw {
804 my ($self) = @_; 843 my ($self) = @_;
805 844
806 my ($w, $h) = ($self->w, $self->h); 845 my ($w, $h) = @$self{qw(w h)};
807 846
808 my $tex = $self->{texture} 847 my $tex = $self->{texture}
809 or return; 848 or return;
810 849
811 glEnable GL_TEXTURE_2D; 850 glEnable GL_TEXTURE_2D;
834} 873}
835 874
836sub size_request { 875sub size_request {
837 my ($self) = @_; 876 my ($self) = @_;
838 877
839 my ($w, $h) = @$self{qw(child_w child_h)} = @{$self->child}{qw(req_w req_h)}; 878 my ($w, $h) = @{$self->child}{qw(req_w req_h)};
840 879
841 $w = 10 if $self->{scroll_x}; 880 $w = 10 if $self->{scroll_x};
842 $h = 10 if $self->{scroll_y}; 881 $h = 10 if $self->{scroll_y};
843 882
844 ($w, $h) 883 ($w, $h)
845} 884}
846 885
847sub size_allocate { 886sub size_allocate {
848 my ($self, $w, $h) = @_; 887 my ($self, $w, $h) = @_;
849 888
889 my $child = $self->child;
890
850 $w = $self->{child_w} if $self->{scroll_x} && $self->{child_w}; 891 $w = $child->{req_w} if $self->{scroll_x} && $child->{req_w};
851 $h = $self->{child_h} if $self->{scroll_y} && $self->{child_h}; 892 $h = $child->{req_h} if $self->{scroll_y} && $child->{req_h};
852 893
853 $self->child->configure (0, 0, $w, $h); 894 $self->child->configure (0, 0, $w, $h);
854 $self->update; 895 $self->update;
855} 896}
856 897
891 } 932 }
892} 933}
893 934
894sub _render { 935sub _render {
895 my ($self) = @_; 936 my ($self) = @_;
937
938 local $CFClient::UI::Base::draw_x = $CFClient::UI::Base::draw_x - $self->{view_x};
939 local $CFClient::UI::Base::draw_y = $CFClient::UI::Base::draw_y - $self->{view_y};
896 940
897 CFClient::OpenGL::glTranslate -$self->{view_x}, -$self->{view_y}; 941 CFClient::OpenGL::glTranslate -$self->{view_x}, -$self->{view_y};
898 942
899 $self->SUPER::_render; 943 $self->SUPER::_render;
900} 944}
999 1043
1000our @ISA = CFClient::UI::Bin::; 1044our @ISA = CFClient::UI::Bin::;
1001 1045
1002use CFClient::OpenGL; 1046use CFClient::OpenGL;
1003 1047
1004my @tex = 1048my $bg =
1049 new_from_file CFClient::Texture CFClient::find_rcfile "d1_bg.png",
1050 mipmap => 1, wrap => 1;
1051
1052my @border =
1005 map { new_from_file CFClient::Texture CFClient::find_rcfile $_, mipmap => 1 } 1053 map { new_from_file CFClient::Texture CFClient::find_rcfile $_, mipmap => 1 }
1006 qw(d1_bg.png d1_border_top.png d1_border_right.png d1_border_left.png d1_border_bottom.png); 1054 qw(d1_border_top.png d1_border_right.png d1_border_left.png d1_border_bottom.png);
1007 1055
1008sub new { 1056sub new {
1009 my $class = shift; 1057 my ($class, %arg) = @_;
1058
1059 my $title = delete $arg{title};
1010 1060
1011 my $self = $class->SUPER::new ( 1061 my $self = $class->SUPER::new (
1012 bg => [1, 1, 1, 1], 1062 bg => [1, 1, 1, 1],
1013 border_bg => [1, 1, 1, 1], 1063 border_bg => [1, 1, 1, 1],
1014 border => 0.6, 1064 border => 0.6,
1015 is_toplevel => 1,
1016 can_events => 1, 1065 can_events => 1,
1017 @_ 1066 min_w => 16,
1067 min_h => 16,
1068 %arg,
1018 ); 1069 );
1019 1070
1020 $self->{title} &&= new CFClient::UI::Label 1071 $self->{title} = new CFClient::UI::Label
1021 align => 0, 1072 align => 0,
1022 valign => 1, 1073 valign => 1,
1023 text => $self->{title}, 1074 text => $title,
1024 fontsize => $self->{border}; 1075 fontsize => $self->{border}
1076 if defined $title;
1025 1077
1026 $self 1078 $self
1079}
1080
1081sub add {
1082 my ($self, @widgets) = @_;
1083
1084 $self->SUPER::add (@widgets);
1085 $self->CFClient::UI::Container::add ($self->{title}) if $self->{title};
1027} 1086}
1028 1087
1029sub border { 1088sub border {
1030 int $_[0]{border} * $::FONTSIZE 1089 int $_[0]{border} * $::FONTSIZE
1031} 1090}
1032 1091
1033sub size_request { 1092sub size_request {
1034 my ($self) = @_; 1093 my ($self) = @_;
1094
1095 $self->{title}->size_request
1096 if $self->{title};
1035 1097
1036 my ($w, $h) = $self->SUPER::size_request; 1098 my ($w, $h) = $self->SUPER::size_request;
1037 1099
1038 ( 1100 (
1039 $w + $self->border * 2, 1101 $w + $self->border * 2,
1042} 1104}
1043 1105
1044sub size_allocate { 1106sub size_allocate {
1045 my ($self, $w, $h) = @_; 1107 my ($self, $w, $h) = @_;
1046 1108
1109 if ($self->{title}) {
1110 $self->{title}{w} = $w;
1111 $self->{title}{h} = $h;
1112 $self->{title}->size_allocate ($w, $h);
1113 }
1114
1115 my $border = $self->border;
1116
1047 $h -= List::Util::max 0, $self->border * 2; 1117 $h -= List::Util::max 0, $border * 2;
1048 $w -= List::Util::max 0, $self->border * 2; 1118 $w -= List::Util::max 0, $border * 2;
1049 1119
1050 $self->{title}->configure ($self->border, int $self->border - $::FONTSIZE * 2, $w, int $::FONTSIZE * 2)
1051 if $self->{title};
1052
1053 $self->child->configure ($self->border, $self->border, $w, $h); 1120 $self->child->configure ($border, $border, $w, $h);
1054} 1121}
1055 1122
1056sub button_down { 1123sub button_down {
1057 my ($self, $ev, $x, $y) = @_; 1124 my ($self, $ev, $x, $y) = @_;
1058 1125
1074 my ($ev, $x, $y) = @_; 1141 my ($ev, $x, $y) = @_;
1075 1142
1076 my $dx = $ev->{x} - $ox; 1143 my $dx = $ev->{x} - $ox;
1077 my $dy = $ev->{y} - $oy; 1144 my $dy = $ev->{y} - $oy;
1078 1145
1079 $self->{user_x} = $wx + $dx * $mx;
1080 $self->{user_y} = $wy + $dy * $my;
1081 $self->{def_w} = $bw + $dx * ($mx ? -1 : 1); 1146 $self->{force_w} = $bw + $dx * ($mx ? -1 : 1);
1082 $self->{def_h} = $bh + $dy * ($my ? -1 : 1); 1147 $self->{force_h} = $bh + $dy * ($my ? -1 : 1);
1083 $self->move ($self->{user_x}, $self->{user_y}); 1148
1084 $self->check_size; 1149 $self->realloc;
1150 $self->move_abs ($wx + $dx * $mx, $wy + $dy * $my);
1085 }; 1151 };
1086 1152
1087 } elsif ($lr ^ $td) { 1153 } elsif ($lr ^ $td) {
1088 my ($ox, $oy) = ($ev->{x}, $ev->{y}); 1154 my ($ox, $oy) = ($ev->{x}, $ev->{y});
1089 my ($bx, $by) = ($self->{x}, $self->{y}); 1155 my ($bx, $by) = ($self->{x}, $self->{y});
1091 $self->{motion} = sub { 1157 $self->{motion} = sub {
1092 my ($ev, $x, $y) = @_; 1158 my ($ev, $x, $y) = @_;
1093 1159
1094 ($x, $y) = ($ev->{x}, $ev->{y}); 1160 ($x, $y) = ($ev->{x}, $ev->{y});
1095 1161
1096 $self->{user_x} = $bx + $x - $ox; 1162 $self->move_abs ($bx + $x - $ox, $by + $y - $oy);
1097 $self->{user_y} = $by + $y - $oy;
1098
1099 $self->move ($self->{user_x}, $self->{user_y});
1100 $self->update;
1101 }; 1163 };
1164 } else {
1165 return 0;
1166 }
1167
1102 } 1168 1
1103} 1169}
1104 1170
1105sub button_up { 1171sub button_up {
1106 my ($self, $ev, $x, $y) = @_; 1172 my ($self, $ev, $x, $y) = @_;
1107 1173
1108 delete $self->{motion}; 1174 !!delete $self->{motion}
1109} 1175}
1110 1176
1111sub mouse_motion { 1177sub mouse_motion {
1112 my ($self, $ev, $x, $y) = @_; 1178 my ($self, $ev, $x, $y) = @_;
1113 1179
1114 $self->{motion}->($ev, $x, $y) if $self->{motion}; 1180 $self->{motion}->($ev, $x, $y) if $self->{motion};
1181
1182 !!$self->{motion}
1115} 1183}
1116 1184
1117sub _draw { 1185sub _draw {
1118 my ($self) = @_; 1186 my ($self) = @_;
1119 1187
1188 my $child = $self->{children}[0];
1189
1120 my ($w, $h ) = ($self->{w}, $self->{h}); 1190 my ($w, $h ) = ($self->{w}, $self->{h});
1121 my ($cw, $ch) = ($self->child->{w}, $self->child->{h}); 1191 my ($cw, $ch) = ($child->{w}, $child->{h});
1122 1192
1123 glEnable GL_TEXTURE_2D; 1193 glEnable GL_TEXTURE_2D;
1124 glTexEnv GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_MODULATE; 1194 glTexEnv GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_MODULATE;
1125 1195
1126 my $border = $self->border; 1196 my $border = $self->border;
1127 1197
1128 glColor @{ $self->{border_bg} }; 1198 glColor @{ $self->{border_bg} };
1129 $tex[1]->draw_quad_alpha (0, 0, $w, $border); 1199 $border[0]->draw_quad_alpha (0, 0, $w, $border);
1130 $tex[3]->draw_quad_alpha (0, $border, $border, $ch); 1200 $border[1]->draw_quad_alpha (0, $border, $border, $ch);
1131 $tex[2]->draw_quad_alpha ($w - $border, $border, $border, $ch); 1201 $border[2]->draw_quad_alpha ($w - $border, $border, $border, $ch);
1132 $tex[4]->draw_quad_alpha (0, $h - $border, $w, $border); 1202 $border[3]->draw_quad_alpha (0, $h - $border, $w, $border);
1133 1203
1134 if (@{$self->{bg}} < 4 || $self->{bg}[3]) { 1204 if (@{$self->{bg}} < 4 || $self->{bg}[3]) {
1135 my $bg = $tex[0]; 1205 glColor @{ $self->{bg} };
1136 1206
1137 # TODO: repeat texture not scale 1207 # TODO: repeat texture not scale
1208 # solve this better(?)
1138 my $rep_x = $cw / $bg->{w}; 1209 $bg->{s} = $cw / $bg->{w};
1139 my $rep_y = $ch / $bg->{h}; 1210 $bg->{t} = $ch / $bg->{h};
1140
1141 glColor @{ $self->{bg} };
1142
1143 $bg->{s} = $rep_x;
1144 $bg->{t} = $rep_y;
1145 $bg->{wrap_mode} = 1;
1146 $bg->draw_quad_alpha ($border, $border, $cw, $ch); 1211 $bg->draw_quad_alpha ($border, $border, $cw, $ch);
1147 } 1212 }
1148 1213
1149 glDisable GL_TEXTURE_2D; 1214 glDisable GL_TEXTURE_2D;
1150 1215
1151 $self->{title}->draw if $self->{title};
1152
1153 $self->child->draw; 1216 $child->draw;
1217
1218 if ($self->{title}) {
1219 glTranslate 0, $border - $self->{h};
1220 $self->{title}->_draw;
1221 }
1154} 1222}
1155 1223
1156############################################################################# 1224#############################################################################
1157 1225
1158package CFClient::UI::Table; 1226package CFClient::UI::Table;
1180 my ($self, $x, $y, $child) = @_; 1248 my ($self, $x, $y, $child) = @_;
1181 1249
1182 $child->set_parent ($self); 1250 $child->set_parent ($self);
1183 $self->{children}[$y][$x] = $child; 1251 $self->{children}[$y][$x] = $child;
1184 1252
1185 $self->check_size (1); 1253 $self->realloc;
1186} 1254}
1187 1255
1188# TODO: move to container class maybe? send children a signal on removal? 1256# TODO: move to container class maybe? send children a signal on removal?
1189sub clear { 1257sub clear {
1190 my ($self) = @_; 1258 my ($self) = @_;
1195 for (@children) { 1263 for (@children) {
1196 delete $_->{parent}; 1264 delete $_->{parent};
1197 $_->hide; 1265 $_->hide;
1198 } 1266 }
1199 1267
1200 $self->check_size (1); 1268 $self->realloc;
1201 $self->update;
1202} 1269}
1203 1270
1204sub get_wh { 1271sub get_wh {
1205 my ($self) = @_; 1272 my ($self) = @_;
1206 1273
1414 ellipsise => 3, # end 1481 ellipsise => 3, # end
1415 layout => (new CFClient::Layout), 1482 layout => (new CFClient::Layout),
1416 fontsize => 1, 1483 fontsize => 1,
1417 align => -1, 1484 align => -1,
1418 valign => -1, 1485 valign => -1,
1419 padding => 2, 1486 padding_x => 2,
1487 padding_y => 2,
1420 can_events => 0, 1488 can_events => 0,
1421 %arg 1489 %arg
1422 ); 1490 );
1423 1491
1424 if (exists $self->{template}) { 1492 if (exists $self->{template}) {
1460 $self->{text} = "T$text"; 1528 $self->{text} = "T$text";
1461 1529
1462 $self->{layout} = new CFClient::Layout if $self->{layout}->is_rgba; 1530 $self->{layout} = new CFClient::Layout if $self->{layout}->is_rgba;
1463 $self->{layout}->set_text ($text); 1531 $self->{layout}->set_text ($text);
1464 1532
1533 $self->realloc;
1465 $self->update; 1534 $self->update;
1466 $self->check_size;
1467} 1535}
1468 1536
1469sub set_markup { 1537sub set_markup {
1470 my ($self, $markup) = @_; 1538 my ($self, $markup) = @_;
1471 1539
1475 my $rgba = $markup =~ /span.*(?:foreground|background)/; 1543 my $rgba = $markup =~ /span.*(?:foreground|background)/;
1476 1544
1477 $self->{layout} = new CFClient::Layout $rgba if $self->{layout}->is_rgba != $rgba; 1545 $self->{layout} = new CFClient::Layout $rgba if $self->{layout}->is_rgba != $rgba;
1478 $self->{layout}->set_markup ($markup); 1546 $self->{layout}->set_markup ($markup);
1479 1547
1548 $self->realloc;
1480 $self->update; 1549 $self->update;
1481 $self->check_size;
1482} 1550}
1483 1551
1484sub size_request { 1552sub size_request {
1485 my ($self) = @_; 1553 my ($self) = @_;
1486 1554
1500 1568
1501 $w = List::Util::max $w, $w2; 1569 $w = List::Util::max $w, $w2;
1502 $h = List::Util::max $h, $h2; 1570 $h = List::Util::max $h, $h2;
1503 } 1571 }
1504 1572
1505 ( 1573 ($w, $h)
1506 $w + $self->{padding} * 2,
1507 $h + $self->{padding} * 2,
1508 )
1509} 1574}
1510 1575
1511sub size_allocate { 1576sub size_allocate {
1512 my ($self, $w, $h) = @_; 1577 my ($self, $w, $h) = @_;
1513 1578
1579 delete $self->{ox};
1580
1514 delete $self->{texture}; 1581 delete $self->{texture}
1582 unless $w >= $self->{req_w} && $self->{old_w} >= $self->{req_w};
1515} 1583}
1516 1584
1517sub set_fontsize { 1585sub set_fontsize {
1518 my ($self, $fontsize) = @_; 1586 my ($self, $fontsize) = @_;
1519 1587
1520 $self->{fontsize} = $fontsize; 1588 $self->{fontsize} = $fontsize;
1521 delete $self->{texture}; 1589 delete $self->{texture};
1522 1590
1523 $self->update; 1591 $self->realloc;
1524 $self->check_size;
1525} 1592}
1526 1593
1527sub _draw { 1594sub _draw {
1528 my ($self) = @_; 1595 my ($self) = @_;
1529 1596
1535 $self->{layout}->set_width ($self->{w}); 1602 $self->{layout}->set_width ($self->{w});
1536 $self->{layout}->set_ellipsise ($self->{ellipsise}); 1603 $self->{layout}->set_ellipsise ($self->{ellipsise});
1537 $self->{layout}->set_single_paragraph_mode ($self->{ellipsise}); 1604 $self->{layout}->set_single_paragraph_mode ($self->{ellipsise});
1538 $self->{layout}->set_height ($self->{fontsize} * $::FONTSIZE); 1605 $self->{layout}->set_height ($self->{fontsize} * $::FONTSIZE);
1539 1606
1540 my $tex = new_from_layout CFClient::Texture $self->{layout}; 1607 new_from_layout CFClient::Texture $self->{layout}
1608 };
1541 1609
1610 unless (exists $self->{ox}) {
1542 $self->{ox} = int ($self->{align} < 0 ? $self->{padding} 1611 $self->{ox} = int ($self->{align} < 0 ? $self->{padding_x}
1543 : $self->{align} > 0 ? $self->{w} - $tex->{w} - $self->{padding} 1612 : $self->{align} > 0 ? $self->{w} - $tex->{w} - $self->{padding_x}
1544 : ($self->{w} - $tex->{w}) * 0.5); 1613 : ($self->{w} - $tex->{w}) * 0.5);
1545 1614
1546 $self->{oy} = int ($self->{valign} < 0 ? $self->{padding} 1615 $self->{oy} = int ($self->{valign} < 0 ? $self->{padding_y}
1547 : $self->{valign} > 0 ? $self->{h} - $tex->{h} - $self->{padding} 1616 : $self->{valign} > 0 ? $self->{h} - $tex->{h} - $self->{padding_y}
1548 : ($self->{h} - $tex->{h}) * 0.5); 1617 : ($self->{h} - $tex->{h}) * 0.5);
1549
1550 $tex
1551 }; 1618 };
1552 1619
1553 glEnable GL_TEXTURE_2D; 1620 glEnable GL_TEXTURE_2D;
1554 glTexEnv GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_REPLACE; 1621 glTexEnv GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_REPLACE;
1555 1622
1609sub set_text { 1676sub set_text {
1610 my ($self, $text) = @_; 1677 my ($self, $text) = @_;
1611 1678
1612 $self->{cursor} = length $text; 1679 $self->{cursor} = length $text;
1613 $self->_set_text ($text); 1680 $self->_set_text ($text);
1614 $self->update; 1681
1615 $self->check_size; 1682 $self->realloc;
1616} 1683}
1617 1684
1618sub get_text { 1685sub get_text {
1619 $_[0]{text} 1686 $_[0]{text}
1620} 1687}
1650 $self->{cursor} = length $text; 1717 $self->{cursor} = length $text;
1651 } elsif ($uni == 27) { 1718 } elsif ($uni == 27) {
1652 $self->_emit ('escape'); 1719 $self->_emit ('escape');
1653 } elsif ($uni) { 1720 } elsif ($uni) {
1654 substr $text, $self->{cursor}++, 0, chr $uni; 1721 substr $text, $self->{cursor}++, 0, chr $uni;
1722 } else {
1723 return 0;
1655 } 1724 }
1656 1725
1657 $self->_set_text ($text); 1726 $self->_set_text ($text);
1658 $self->update; 1727
1659 $self->check_size; 1728 $self->realloc;
1729
1730 1
1660} 1731}
1661 1732
1662sub focus_in { 1733sub focus_in {
1663 my ($self) = @_; 1734 my ($self) = @_;
1664 1735
1679 utf8::encode $text; 1750 utf8::encode $text;
1680 $self->{cursor} = length substr $text, 0, $idx; 1751 $self->{cursor} = length substr $text, 0, $idx;
1681 1752
1682 $self->_set_text ($self->{text}); 1753 $self->_set_text ($self->{text});
1683 $self->update; 1754 $self->update;
1755
1756 1
1684} 1757}
1685 1758
1686sub mouse_motion { 1759sub mouse_motion {
1687 my ($self, $ev, $x, $y) = @_; 1760 my ($self, $ev, $x, $y) = @_;
1688# printf "M %d,%d %d,%d\n", $ev->motion_x, $ev->motion_y, $x, $y;#d# 1761# printf "M %d,%d %d,%d\n", $ev->motion_x, $ev->motion_y, $x, $y;#d#
1762
1763 0
1689} 1764}
1690 1765
1691sub _draw { 1766sub _draw {
1692 my ($self) = @_; 1767 my ($self) = @_;
1693 1768
1770 } else { 1845 } else {
1771 $self->set_text ($self->{history_saveback}); 1846 $self->set_text ($self->{history_saveback});
1772 } 1847 }
1773 1848
1774 } else { 1849 } else {
1775 $self->SUPER::key_down ($ev); 1850 return $self->SUPER::key_down ($ev)
1851 }
1852
1776 } 1853 1
1777
1778} 1854}
1779 1855
1780############################################################################# 1856#############################################################################
1781 1857
1782package CFClient::UI::Button; 1858package CFClient::UI::Button;
1791 1867
1792sub new { 1868sub new {
1793 my $class = shift; 1869 my $class = shift;
1794 1870
1795 $class->SUPER::new ( 1871 $class->SUPER::new (
1796 padding => 4, 1872 padding_x => 4,
1873 padding_y => 4,
1797 fg => [1, 1, 1], 1874 fg => [1, 1, 1],
1798 active_fg => [0, 0, 1], 1875 active_fg => [0, 0, 1],
1799 can_hover => 1, 1876 can_hover => 1,
1800 align => 0, 1877 align => 0,
1801 valign => 0, 1878 valign => 0,
1810 my ($self, $ev, $x, $y) = @_; 1887 my ($self, $ev, $x, $y) = @_;
1811 1888
1812 $self->emit ("activate") 1889 $self->emit ("activate")
1813 if $x >= 0 && $x < $self->{w} 1890 if $x >= 0 && $x < $self->{w}
1814 && $y >= 0 && $y < $self->{h}; 1891 && $y >= 0 && $y < $self->{h};
1892
1893 1
1815} 1894}
1816 1895
1817sub _draw { 1896sub _draw {
1818 my ($self) = @_; 1897 my ($self) = @_;
1819 1898
1848 1927
1849sub new { 1928sub new {
1850 my $class = shift; 1929 my $class = shift;
1851 1930
1852 $class->SUPER::new ( 1931 $class->SUPER::new (
1853 padding => 2, 1932 padding_x => 2,
1933 padding_y => 2,
1854 fg => [1, 1, 1], 1934 fg => [1, 1, 1],
1855 active_fg => [1, 1, 0], 1935 active_fg => [1, 1, 0],
1856 bg => [0, 0, 0, 0.2], 1936 bg => [0, 0, 0, 0.2],
1857 active_bg => [1, 1, 1, 0.5], 1937 active_bg => [1, 1, 1, 0.5],
1858 state => 0, 1938 state => 0,
1862} 1942}
1863 1943
1864sub size_request { 1944sub size_request {
1865 my ($self) = @_; 1945 my ($self) = @_;
1866 1946
1867 ($self->{padding} * 2 + 6) x 2 1947 (6) x 2
1868} 1948}
1869 1949
1870sub button_down { 1950sub button_down {
1871 my ($self, $ev, $x, $y) = @_; 1951 my ($self, $ev, $x, $y) = @_;
1872 1952
1873 if ($x >= $self->{padding} && $x < $self->{w} - $self->{padding} 1953 if ($x >= $self->{padding_x} && $x < $self->{w} - $self->{padding_x}
1874 && $y >= $self->{padding} && $y < $self->{h} - $self->{padding}) { 1954 && $y >= $self->{padding_y} && $y < $self->{h} - $self->{padding_y}) {
1875 $self->{state} = !$self->{state}; 1955 $self->{state} = !$self->{state};
1876 $self->_emit (changed => $self->{state}); 1956 $self->_emit (changed => $self->{state});
1957 } else {
1958 return 0
1959 }
1960
1877 } 1961 1
1878} 1962}
1879 1963
1880sub _draw { 1964sub _draw {
1881 my ($self) = @_; 1965 my ($self) = @_;
1882 1966
1883 $self->SUPER::_draw; 1967 $self->SUPER::_draw;
1884 1968
1885 glTranslate $self->{padding} + 0.375, $self->{padding} + 0.375, 0; 1969 glTranslate $self->{padding_x} + 0.375, $self->{padding_y} + 0.375, 0;
1886 1970
1887 my $s = (List::Util::min @$self{qw(w h)}) - $self->{padding} * 2; 1971 my ($w, $h) = @$self{qw(w h)};
1972
1973 my $s = List::Util::min $w - $self->{padding_x} * 2, $h - $self->{padding_y} * 2;
1888 1974
1889 glColor @{ $FOCUS == $self ? $self->{active_fg} : $self->{fg} }; 1975 glColor @{ $FOCUS == $self ? $self->{active_fg} : $self->{fg} };
1890 1976
1891 my $tex = $self->{state} ? $tex[1] : $tex[0]; 1977 my $tex = $self->{state} ? $tex[1] : $tex[0];
1892 1978
2157 fg => [1, 1, 1], 2243 fg => [1, 1, 1],
2158 active_fg => [0, 0, 0], 2244 active_fg => [0, 0, 0],
2159 bg => [0, 0, 0, 0.2], 2245 bg => [0, 0, 0, 0.2],
2160 active_bg => [1, 1, 1, 0.5], 2246 active_bg => [1, 1, 1, 0.5],
2161 range => [0, 0, 100, 10, 0], 2247 range => [0, 0, 100, 10, 0],
2162 req_w => $::WIDTH / 80, 2248 min_w => $::WIDTH / 80,
2163 req_h => $::WIDTH / 80, 2249 min_h => $::WIDTH / 80,
2164 vertical => 0, 2250 vertical => 0,
2165 can_hover => 1, 2251 can_hover => 1,
2166 inner_pad => 0.02, 2252 inner_pad => 0.02,
2167 @_ 2253 @_
2168 ); 2254 );
2171 $self->update; 2257 $self->update;
2172 2258
2173 $self 2259 $self
2174} 2260}
2175 2261
2262sub changed { }
2263
2176sub set_range { 2264sub set_range {
2177 my ($self, $range) = @_; 2265 my ($self, $range) = @_;
2178 2266
2179 ($range, $self->{range}) = ($self->{range}, $range); 2267 ($range, $self->{range}) = ($self->{range}, $range);
2180 2268
2206} 2294}
2207 2295
2208sub size_request { 2296sub size_request {
2209 my ($self) = @_; 2297 my ($self) = @_;
2210 2298
2211 my $w = $self->{req_w}; 2299 ($self->{req_w}, $self->{req_h})
2212 my $h = $self->{req_h};
2213
2214 $self->{vertical} ? ($h, $w) : ($w, $h)
2215} 2300}
2216 2301
2217sub button_down { 2302sub button_down {
2218 my ($self, $ev, $x, $y) = @_; 2303 my ($self, $ev, $x, $y) = @_;
2219 2304
2220 $self->SUPER::button_down ($ev, $x, $y); 2305 $self->SUPER::button_down ($ev, $x, $y);
2221 2306
2222 $self->{click} = [$self->{range}[0], $self->{vertical} ? $y : $x]; 2307 $self->{click} = [$self->{range}[0], $self->{vertical} ? $y : $x];
2223 2308
2224 $self->mouse_motion ($ev, $x, $y); 2309 $self->mouse_motion ($ev, $x, $y)
2225} 2310}
2226 2311
2227sub mouse_motion { 2312sub mouse_motion {
2228 my ($self, $ev, $x, $y) = @_; 2313 my ($self, $ev, $x, $y) = @_;
2229 2314
2233 my (undef, $lo, $hi, $page) = @{$self->{range}}; 2318 my (undef, $lo, $hi, $page) = @{$self->{range}};
2234 2319
2235 $x = ($x - $self->{click}[1]) / ($w * $self->{scale}); 2320 $x = ($x - $self->{click}[1]) / ($w * $self->{scale});
2236 2321
2237 $self->set_value ($self->{click}[0] + $x * ($hi - $page - $lo)); 2322 $self->set_value ($self->{click}[0] + $x * ($hi - $page - $lo));
2323 } else {
2324 return 0;
2325 }
2326
2238 } 2327 1
2239} 2328}
2240 2329
2241sub update { 2330sub update {
2242 my ($self) = @_; 2331 my ($self) = @_;
2243 2332
2604} 2693}
2605 2694
2606sub set_tooltip_from { 2695sub set_tooltip_from {
2607 my ($self, $widget) = @_; 2696 my ($self, $widget) = @_;
2608 2697
2698 my $tooltip = $widget->{tooltip};
2699
2700 if ($ENV{CFPLUS_DEBUG} & 2) {
2701 $tooltip .= "\n\n" . (ref $widget) . "\n"
2702 . "$widget->{x} $widget->{y} $widget->{w} $widget->{h}\n"
2703 . "req $widget->{req_w} $widget->{req_h}\n"
2704 . "visible $widget->{visible}";
2705 }
2706
2609 $self->add (new CFClient::UI::Label 2707 $self->add (new CFClient::UI::Label
2610 markup => $widget->{tooltip}, 2708 markup => $tooltip,
2611 max_w => ($widget->{tooltip_width} || 0.25) * $::WIDTH, 2709 max_w => ($widget->{tooltip_width} || 0.25) * $::WIDTH,
2612 fontsize => 0.8, 2710 fontsize => 0.8,
2613 fg => [0, 0, 0, 1], 2711 fg => [0, 0, 0, 1],
2614 ellipsise => 0, 2712 ellipsise => 0,
2615 font => ($widget->{tooltip_font} || $::FONT_PROP), 2713 font => ($widget->{tooltip_font} || $::FONT_PROP),
2626 2724
2627sub size_allocate { 2725sub size_allocate {
2628 my ($self, $w, $h) = @_; 2726 my ($self, $w, $h) = @_;
2629 2727
2630 $self->SUPER::size_allocate ($w - 4, $h - 4); 2728 $self->SUPER::size_allocate ($w - 4, $h - 4);
2729}
2730
2731sub visibility_change {
2732 my ($self, $visible) = @_;
2733
2734 return unless $visible;
2735
2736 $self->{root}->on_post_alloc ("move_$self" => sub {
2737 my $widget = $self->{owner}
2738 or return;
2739
2740 my ($x, $y) = $widget->coord2global ($widget->{w}, 0);
2741
2742 ($x, $y) = $widget->coord2global (-$self->{w}, 0)
2743 if $x + $self->{w} > $::WIDTH;
2744
2745 $self->move_abs ($x, $y);
2746 });
2631} 2747}
2632 2748
2633sub _draw { 2749sub _draw {
2634 my ($self) = @_; 2750 my ($self) = @_;
2635 2751
2652 glVertex $w, $h; 2768 glVertex $w, $h;
2653 glVertex $w, 0; 2769 glVertex $w, 0;
2654 glEnd; 2770 glEnd;
2655 2771
2656 glTranslate 2 - 0.375, 2 - 0.375; 2772 glTranslate 2 - 0.375, 2 - 0.375;
2773
2657 $self->SUPER::_draw; 2774 $self->SUPER::_draw;
2658} 2775}
2659 2776
2660############################################################################# 2777#############################################################################
2661 2778
2737 $self->SUPER::DESTROY; 2854 $self->SUPER::DESTROY;
2738} 2855}
2739 2856
2740############################################################################# 2857#############################################################################
2741 2858
2742package CFClient::UI::Inventory;
2743
2744our @ISA = CFClient::UI::ScrolledWindow::;
2745
2746sub new {
2747 my $class = shift;
2748
2749 my $self = $class->SUPER::new (
2750 scrolled => (new CFClient::UI::Table col_expand => [0, 1, 0]),
2751 @_,
2752 );
2753
2754 $self
2755}
2756
2757sub set_items {
2758 my ($self, $items) = @_;
2759
2760 $self->{scrolled}->clear;
2761 return unless $items;
2762
2763 my @items = sort {
2764 ($a->{type} <=> $b->{type})
2765 or ($a->{name} cmp $b->{name})
2766 } @$items;
2767
2768 $self->{real_items} = \@items;
2769
2770 my $row = 0;
2771 for my $item (@items) {
2772 CFClient::Item::update_widgets $item;
2773
2774 $self->{scrolled}->add (0, $row, $item->{face_widget});
2775 $self->{scrolled}->add (1, $row, $item->{desc_widget});
2776 $self->{scrolled}->add (2, $row, $item->{weight_widget});
2777
2778 $row++;
2779 }
2780}
2781
2782#############################################################################
2783
2784package CFClient::UI::Menu; 2859package CFClient::UI::Menu;
2785 2860
2786our @ISA = CFClient::UI::FancyFrame::; 2861our @ISA = CFClient::UI::FancyFrame::;
2787 2862
2788use CFClient::OpenGL; 2863use CFClient::OpenGL;
2826 # maybe save $GRAB? must be careful about events... 2901 # maybe save $GRAB? must be careful about events...
2827 $GRAB = $self; 2902 $GRAB = $self;
2828 $self->{button} = $ev->{button}; 2903 $self->{button} = $ev->{button};
2829 2904
2830 $self->show; 2905 $self->show;
2831 $self->move ($ev->{x} - $self->{w} * 0.5, $ev->{y} - $self->{h} * 0.5); 2906 $self->move_abs ($ev->{x} - $self->{w} * 0.5, $ev->{y} - $self->{h} * 0.5);
2832} 2907}
2833 2908
2834sub mouse_motion { 2909sub mouse_motion {
2835 my ($self, $ev, $x, $y) = @_; 2910 my ($self, $ev, $x, $y) = @_;
2836 2911
2837 # TODO: should use vbox->find_widget or so 2912 # TODO: should use vbox->find_widget or so
2838 $HOVER = $ROOT->find_widget ($ev->{x}, $ev->{y}); 2913 $HOVER = $ROOT->find_widget ($ev->{x}, $ev->{y});
2839 $self->{hover} = $self->{item}{$HOVER}; 2914 $self->{hover} = $self->{item}{$HOVER};
2915
2916 0
2840} 2917}
2841 2918
2842sub button_up { 2919sub button_up {
2843 my ($self, $ev, $x, $y) = @_; 2920 my ($self, $ev, $x, $y) = @_;
2844 2921
2846 undef $GRAB; 2923 undef $GRAB;
2847 $self->hide; 2924 $self->hide;
2848 2925
2849 $self->_emit ("popdown"); 2926 $self->_emit ("popdown");
2850 $self->{hover}[1]->() if $self->{hover}; 2927 $self->{hover}[1]->() if $self->{hover};
2928 } else {
2929 return 0
2930 }
2931
2851 } 2932 1
2852} 2933}
2853 2934
2854############################################################################# 2935#############################################################################
2855 2936
2856package CFClient::UI::Statusbox; 2937package CFClient::UI::Statusbox;
2961 $self->SUPER::reconfigure; 3042 $self->SUPER::reconfigure;
2962} 3043}
2963 3044
2964############################################################################# 3045#############################################################################
2965 3046
2966package CFClient::UI::Root; 3047package CFClient::UI::Inventory;
2967 3048
2968our @ISA = CFClient::UI::Container::; 3049our @ISA = CFClient::UI::ScrolledWindow::;
2969
2970use CFClient::OpenGL;
2971 3050
2972sub new { 3051sub new {
2973 my $class = shift; 3052 my $class = shift;
2974 3053
2975 $class->SUPER::new ( 3054 my $self = $class->SUPER::new (
3055 scrolled => (new CFClient::UI::Table col_expand => [0, 1, 0]),
3056 @_,
3057 );
3058
3059 $self
3060}
3061
3062sub set_items {
3063 my ($self, $items) = @_;
3064
3065 $self->{scrolled}->clear;
3066 return unless $items;
3067
3068 my @items = sort {
3069 ($a->{type} <=> $b->{type})
3070 or ($a->{name} cmp $b->{name})
3071 } @$items;
3072
3073 $self->{real_items} = \@items;
3074
3075 my $row = 0;
3076 for my $item (@items) {
3077 CFClient::Item::update_widgets $item;
3078
3079 $self->{scrolled}->add (0, $row, $item->{face_widget});
3080 $self->{scrolled}->add (1, $row, $item->{desc_widget});
3081 $self->{scrolled}->add (2, $row, $item->{weight_widget});
3082
3083 $row++;
3084 }
3085}
3086
3087#############################################################################
3088
3089package CFClient::UI::BindEditor;
3090
3091our @ISA = CFClient::UI::FancyFrame::;
3092
3093sub new {
3094 my $class = shift;
3095
3096 my $self = $class->SUPER::new (binding => [], commands => [], @_);
3097
3098 $self->add (my $vb = new CFClient::UI::VBox);
3099
3100
3101 $vb->add ($self->{rec_btn} = new CFClient::UI::Button
3102 text => "start recording",
3103 tooltip => "Start/Stops recording of actions."
3104 ."All subsequent actions after the recording started will be captured."
3105 ."The actions are displayed after the record was stopped."
3106 ."To bind the action you have to click on the 'Bind' button",
3107 on_activate => sub {
3108 unless ($self->{recording}) {
3109 $self->start;
3110 } else {
3111 $self->stop;
3112 }
3113 });
3114
3115 $vb->add (new CFClient::UI::Label text => "Actions:");
3116 $vb->add ($self->{cmdbox} = new CFClient::UI::VBox);
3117
3118 $vb->add (new CFClient::UI::Label text => "Bound to: ");
3119 $vb->add (my $hb = new CFClient::UI::HBox);
3120 $hb->add ($self->{keylbl} = new CFClient::UI::Label expand => 1);
3121 $hb->add (new CFClient::UI::Button
3122 text => "bind",
3123 tooltip => "This opens a query where you have to press the key combination to bind the recorded actions",
3124 on_activate => sub {
3125 $self->ask_for_bind;
3126 });
3127
3128 $vb->add (my $hb = new CFClient::UI::HBox);
3129 $hb->add (new CFClient::UI::Button
3130 text => "ok",
3131 expand => 1,
3132 tooltip => "This closes the binding editor and saves the binding",
3133 on_activate => sub {
3134 $self->hide;
3135 $self->commit;
3136 });
3137
3138 $hb->add (new CFClient::UI::Button
3139 text => "cancel",
3140 expand => 1,
3141 tooltip => "This closes the binding editor without saving",
3142 on_activate => sub {
3143 $self->hide;
3144 $self->{binding_cancel}->()
3145 if $self->{binding_cancel};
3146 });
3147
3148 $self->update_binding_widgets;
3149
3150 $self
3151}
3152
3153sub commit {
3154 my ($self) = @_;
3155 my ($mod, $sym, $cmds) = $self->get_binding;
3156 if ($sym != 0 && @$cmds > 0) {
3157 $::STATUSBOX->add ("Bound actions to '".CFClient::Binder::keycombo_to_name ($mod, $sym)
3158 ."'. Don't forget 'Save Config'!");
3159 $self->{binding_change}->($mod, $sym, $cmds)
3160 if $self->{binding_change};
3161 } else {
3162 $::STATUSBOX->add ("No action bound, no key or action specified!");
3163 $self->{binding_cancel}->()
3164 if $self->{binding_cancel};
3165 }
3166}
3167
3168sub start {
3169 my ($self) = @_;
3170
3171 $self->{rec_btn}->set_text ("stop recording");
3172 $self->{recording} = 1;
3173 $self->clear_command_list;
3174 $::CONN->start_record if $::CONN;
3175}
3176
3177sub stop {
3178 my ($self) = @_;
3179
3180 $self->{rec_btn}->set_text ("start recording");
3181 $self->{recording} = 0;
3182
3183 my $rec;
3184 $rec = $::CONN->stop_record if $::CONN;
3185 return unless ref $rec eq 'ARRAY';
3186 $self->set_command_list ($rec);
3187}
3188
3189
3190sub ask_for_bind_and_commit {
3191 my ($self) = @_;
3192 $self->ask_for_bind (1);
3193}
3194
3195sub ask_for_bind {
3196 my ($self, $commit) = @_;
3197
3198 CFClient::Binder::open_binding_dialog (sub {
3199 my ($mod, $sym) = @_;
3200 $self->{binding} = [$mod, $sym]; # XXX: how to stop that memleak?
3201 $self->update_binding_widgets;
3202 $self->commit if $commit;
3203 });
3204}
3205
3206# $mod and $sym are the modifiers and key symbol
3207# $cmds is a array ref of strings (the commands)
3208# $cb is the callback that is executed on OK
3209# $ccb is the callback that is executed on CANCEL and
3210# when the binding was unsuccessful on OK
3211sub set_binding {
3212 my ($self, $mod, $sym, $cmds, $cb, $ccb) = @_;
3213
3214 $self->clear_command_list;
3215 $self->{recording} = 0;
3216 $self->{rec_btn}->set_text ("start recording");
3217
3218 $self->{binding} = [$mod, $sym];
3219 $self->{commands} = $cmds;
3220
3221 $self->{binding_change} = $cb;
3222 $self->{binding_cancel} = $ccb;
3223
3224 $self->update_binding_widgets;
3225}
3226
3227# this is a shortcut method that asks for a binding
3228# and then just binds it.
3229sub do_quick_binding {
3230 my ($self, $cmds) = @_;
3231 $self->set_binding (undef, undef, $cmds, sub {
3232 $::CFG->{bindings}->{$_[0]}->{$_[1]} = $_[2];
3233 });
3234 $self->ask_for_bind (1);
3235}
3236
3237sub update_binding_widgets {
3238 my ($self) = @_;
3239 my ($mod, $sym, $cmds) = $self->get_binding;
3240 $self->{keylbl}->set_text (CFClient::Binder::keycombo_to_name ($mod, $sym));
3241 $self->set_command_list ($cmds);
3242}
3243
3244sub get_binding {
3245 my ($self) = @_;
3246 return (
3247 $self->{binding}->[0],
3248 $self->{binding}->[1],
3249 [ grep { defined $_ } @{$self->{commands}} ]
3250 );
3251}
3252
3253sub clear_command_list {
3254 my ($self) = @_;
3255 $self->{cmdbox}->clear ();
3256}
3257
3258sub set_command_list {
3259 my ($self, $cmds) = @_;
3260
3261 $self->{cmdbox}->clear ();
3262 $self->{commands} = $cmds;
3263
3264 my $idx = 0;
3265
3266 for (@$cmds) {
3267 $self->{cmdbox}->add (my $hb = new CFClient::UI::HBox);
3268
3269 my $i = $idx;
3270 $hb->add (new CFClient::UI::Label text => $_);
3271 $hb->add (new CFClient::UI::Button
3272 text => "delete",
3273 tooltip => "Deletes the action from the record",
3274 on_activate => sub {
3275 $self->{cmdbox}->remove ($hb);
3276 $cmds->[$i] = undef;
3277 });
3278
3279
3280 $idx++
3281 }
3282}
3283
3284#############################################################################
3285
3286package CFClient::UI::SpellList;
3287
3288our @ISA = CFClient::UI::FancyFrame::;
3289
3290sub new {
3291 my $class = shift;
3292
3293 my $self = $class->SUPER::new (binding => [], commands => [], @_);
3294
3295 $self->add (new CFClient::UI::ScrolledWindow
3296 scrolled => $self->{spellbox} = new CFClient::UI::Table);
3297
3298 $self;
3299}
3300
3301# XXX: Do sorting? Argl...
3302sub add_spell {
3303 my ($self, $spell) = @_;
3304 $self->{spells}->{$spell->{name}} = $spell;
3305
3306 $self->{spellbox}->add (0, $self->{tbl_idx}, new CFClient::UI::Face
3307 face => $spell->{face},
3308 can_hover => 1,
3309 can_events => 1,
3310 tooltip => $spell->{message});
3311
3312 $self->{spellbox}->add (1, $self->{tbl_idx}, new CFClient::UI::Label
3313 text => $spell->{name},
3314 can_hover => 1,
3315 can_events => 1,
3316 tooltip => $spell->{message},
3317 expand => 1);
3318
3319 $self->{spellbox}->add (2, $self->{tbl_idx}, new CFClient::UI::Label
3320 text => (sprintf "lvl: %2d sp: %2d dmg: %2d",
3321 $spell->{level}, ($spell->{mana} || $spell->{grace}), $spell->{damage}),
3322 expand => 1);
3323
3324 $self->{spellbox}->add (3, $self->{tbl_idx}++, new CFClient::UI::Button
3325 text => "bind to key",
3326 on_activate => sub { $::BIND_EDITOR->do_quick_binding (["cast $spell->{name}"]) });
3327}
3328
3329sub rebuild_spell_list {
3330 my ($self) = @_;
3331 $self->{tbl_idx} = 0;
3332 $self->add_spell ($_) for values %{$self->{spells}};
3333}
3334
3335sub remove_spell {
3336 my ($self, $spell) = @_;
3337 delete $self->{spells}->{$spell->{name}};
3338 $self->rebuild_spell_list;
3339}
3340
3341#############################################################################
3342
3343package CFClient::UI::Root;
3344
3345our @ISA = CFClient::UI::Container::;
3346
3347use CFClient::OpenGL;
3348
3349sub new {
3350 my $class = shift;
3351
3352 my $self = $class->SUPER::new (
2976 visible => 1, 3353 visible => 1,
2977 @_, 3354 @_,
2978 ) 3355 );
2979}
2980 3356
2981sub configure { 3357 Scalar::Util::weaken ($self->{root} = $self);
2982 my ($self, $x, $y, $w, $h) = @_;
2983 3358
2984 $self->{w} = $w; 3359 $self
2985 $self->{h} = $h;
2986}
2987
2988sub check_size {
2989 my ($self) = @_;
2990
2991 $self->size_allocate ($self->{w}, $self->{h})
2992 if $self->{w};
2993} 3360}
2994 3361
2995sub size_request { 3362sub size_request {
2996 my ($self) = @_; 3363 my ($self) = @_;
2997 3364
2998 ($self->{w}, $self->{h}) 3365 ($self->{w}, $self->{h})
3366}
3367
3368sub _to_pixel {
3369 my ($coord, $size, $max) = @_;
3370
3371 $coord =
3372 $coord eq "center" ? ($max - $size) * 0.5
3373 : $coord eq "max" ? $max
3374 : $coord;
3375
3376 $coord = 0 if $coord < 0;
3377 $coord = $max - $size if $coord > $max - $size;
3378
3379 int $coord + 0.5
2999} 3380}
3000 3381
3001sub size_allocate { 3382sub size_allocate {
3002 my ($self, $w, $h) = @_; 3383 my ($self, $w, $h) = @_;
3003 3384
3004 for my $child ($self->children) { 3385 for my $child ($self->children) {
3005 my ($X, $Y, $W, $H) = @$child{qw(x y req_w req_h)}; 3386 my ($X, $Y, $W, $H) = @$child{qw(x y req_w req_h)};
3006 3387
3007 $X = $child->{req_x} > 0 ? $child->{req_x} : $w - $W - $child->{req_x} + 1 3388 $X = $child->{force_x} if exists $child->{force_x};
3008 if exists $child->{req_x}; 3389 $Y = $child->{force_y} if exists $child->{force_y};
3009 3390
3010 $Y = $child->{req_y} > 0 ? $child->{req_y} : $h - $H - $child->{req_y} + 1 3391 $X = _to_pixel $X, $W, $self->{w};
3011 if exists $child->{req_y}; 3392 $Y = _to_pixel $Y, $H, $self->{h};
3012
3013 delete @$child{qw(req_x req_y)};
3014
3015 $X = List::Util::max 0, List::Util::min $w - $W, int $X + 0.5;
3016 $Y = List::Util::max 0, List::Util::min $h - $H, int $Y + 0.5;
3017 3393
3018 $child->configure ($X, $Y, $W, $H); 3394 $child->configure ($X, $Y, $W, $H);
3019 } 3395 }
3020} 3396}
3021 3397
3032} 3408}
3033 3409
3034sub update { 3410sub update {
3035 my ($self) = @_; 3411 my ($self) = @_;
3036 3412
3037 $self->check_size;
3038 $::WANT_REFRESH++; 3413 $::WANT_REFRESH++;
3039} 3414}
3040 3415
3041sub add { 3416sub add {
3042 my ($self, @children) = @_; 3417 my ($self, @children) = @_;
3043 3418
3044 for my $child (@children) {
3045 $child->{is_toplevel} = 1; 3419 $_->{is_toplevel} = 1
3046 3420 for @children;
3047 # integerise window positions
3048 $child->{x} = int $child->{x};
3049 $child->{y} = int $child->{y};
3050 }
3051 3421
3052 $self->SUPER::add (@children); 3422 $self->SUPER::add (@children);
3053
3054 for (my @widgets = @children; my $w = pop @widgets; ) {
3055 push @widgets, $w->children;
3056 $w->set_visible;
3057 }
3058
3059} 3423}
3060 3424
3061sub remove { 3425sub remove {
3062 my ($self, @children) = @_; 3426 my ($self, @children) = @_;
3063 3427
3064 $self->SUPER::remove (@children); 3428 $self->SUPER::remove (@children);
3429
3430 delete $self->{is_toplevel}
3431 for @children;
3065 3432
3066 while (@children) { 3433 while (@children) {
3067 my $w = pop @children; 3434 my $w = pop @children;
3068 push @children, $w->children; 3435 push @children, $w->children;
3069 $w->set_invisible; 3436 $w->set_invisible;
3088 while ($self->{refresh_hook}) { 3455 while ($self->{refresh_hook}) {
3089 $_->() 3456 $_->()
3090 for values %{delete $self->{refresh_hook}}; 3457 for values %{delete $self->{refresh_hook}};
3091 } 3458 }
3092 3459
3093 if ($self->{check_size}) { 3460 if ($self->{realloc}) {
3461 my %queue;
3094 my @queue; 3462 my @queue;
3463 my $widget;
3095 3464
3096 for (;;) { 3465 outer:
3097 if ($self->{check_size}) { 3466 while () {
3098 #TODO use array-of-depth approach 3467 if (my $realloc = delete $self->{realloc}) {
3468 for $widget (values %$realloc) {
3469 $widget->{visible} or next; # do not resize invisible widgets
3099 3470
3100 @queue = sort { $a->{visible} <=> $b->{visible} } 3471 $queue{$widget+0}++ and next; # duplicates are common
3101 @queue, values %{delete $self->{check_size}}; 3472
3473 push @{ $queue[$widget->{visible}] }, $widget;
3474 }
3102 } 3475 }
3103 3476
3477 while () {
3478 @queue or last outer;
3479
3104 my $widget = pop @queue || last; 3480 $widget = pop @{ $queue[-1] || [] }
3105 3481 and last;
3106 defined $widget->{visible} or last; # do not resize invisible widgets
3107
3108 my ($w, $h) = $widget->{def_w} && $widget->{def_h}
3109 ? @$widget{qw(def_w def_h)}
3110 : $widget->size_request;
3111
3112 if (delete $widget->{force_alloc}
3113 or $w != $widget->{req_w} or $h != $widget->{req_h}) {
3114 Carp::confess "$widget: size_request is negative" if $w < 0 || $h < 0;#d#
3115 3482
3483 pop @queue;
3484 }
3485
3486 delete $queue{$widget+0};
3487
3488 my ($w, $h) = $widget->size_request;
3489
3490 $w = List::Util::max $widget->{min_w}, $w + $widget->{padding_x} * 2;
3491 $h = List::Util::max $widget->{min_h}, $h + $widget->{padding_y} * 2;
3492
3493 $w = $widget->{force_w} if exists $widget->{force_w};
3494 $h = $widget->{force_h} if exists $widget->{force_h};
3495
3496 if ($widget->{req_w} != $w || $widget->{req_h} != $h
3497 || delete $widget->{force_realloc}) {
3116 $widget->{req_w} = $w; 3498 $widget->{req_w} = $w;
3117 $widget->{req_h} = $h; 3499 $widget->{req_h} = $h;
3118 3500
3119 $self->{size_alloc}{$widget} = [$widget, $widget->{w} || $w, $widget->{h} || $h]; 3501 $self->{size_alloc}{$widget+0} = $widget;
3120 3502
3121 $widget->{parent}->check_size
3122 if $widget->{parent}; 3503 if (my $parent = $widget->{parent}) {
3504 $self->{realloc}{$parent+0} = $parent
3505 unless $queue{$parent+0};
3506
3507 $parent->{force_size_alloc} = 1;
3508 $self->{size_alloc}{$parent+0} = $parent;
3509 }
3123 } 3510 }
3511
3512 delete $self->{realloc}{$widget+0};
3124 } 3513 }
3125 } 3514 }
3126 3515
3127 while ($self->{size_alloc}) { 3516 while (my $size_alloc = delete $self->{size_alloc}) {
3128 for (
3129 sort { $a->[0]{visible} <=> $b->[0]{visible} } 3517 my @queue = sort { $b->{visible} <=> $a->{visible} }
3130 values %{delete $self->{size_alloc}} 3518 values %$size_alloc;
3131 ) { 3519
3132 my ($widget, $w, $h) = @$_; 3520 while () {
3521 my $widget = pop @queue || last;
3522
3523 my ($w, $h) = @$widget{qw(alloc_w alloc_h)};
3133 3524
3134 $w = 0 if $w < 0; 3525 $w = 0 if $w < 0;
3135 $h = 0 if $h < 0; 3526 $h = 0 if $h < 0;
3136 3527
3528 $w = int $w + 0.5;
3529 $h = int $h + 0.5;
3530
3531 if ($widget->{w} != $w || $widget->{h} != $h || delete $widget->{force_size_alloc}) {
3532 $widget->{old_w} = $widget->{w};
3533 $widget->{old_h} = $widget->{h};
3534
3137 $widget->{w} = $w; 3535 $widget->{w} = $w;
3138 $widget->{h} = $h; 3536 $widget->{h} = $h;
3537
3139 $widget->emit (size_allocate => $w, $h); 3538 $widget->emit (size_allocate => $w, $h);
3539 }
3140 } 3540 }
3141 } 3541 }
3142 3542
3143 while ($self->{post_alloc_hook}) { 3543 while ($self->{post_alloc_hook}) {
3144 $_->() 3544 $_->()
3145 for values %{delete $self->{post_alloc_hook}}; 3545 for values %{delete $self->{post_alloc_hook}};
3146 } 3546 }
3547
3147 3548
3148 glViewport 0, 0, $::WIDTH, $::HEIGHT; 3549 glViewport 0, 0, $::WIDTH, $::HEIGHT;
3149 glClearColor +($::CFG->{fow_intensity}) x 3, 1; 3550 glClearColor +($::CFG->{fow_intensity}) x 3, 1;
3150 glClear GL_COLOR_BUFFER_BIT; 3551 glClear GL_COLOR_BUFFER_BIT;
3151 3552
3153 glLoadIdentity; 3554 glLoadIdentity;
3154 glOrtho 0, $::WIDTH, $::HEIGHT, 0, -10000, 10000; 3555 glOrtho 0, $::WIDTH, $::HEIGHT, 0, -10000, 10000;
3155 glMatrixMode GL_MODELVIEW; 3556 glMatrixMode GL_MODELVIEW;
3156 glLoadIdentity; 3557 glLoadIdentity;
3157 3558
3559 {
3560 package CFClient::UI::Base;
3561
3562 ($draw_x, $draw_y, $draw_w, $draw_h) =
3563 (0, 0, $self->{w}, $self->{h});
3564 }
3565
3158 $self->_draw; 3566 $self->_draw;
3159} 3567}
3160 3568
3161############################################################################# 3569#############################################################################
3162 3570

Diff Legend

Removed lines
+ Added lines
< Changed lines
> Changed lines