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.256 by root, Tue May 30 01:42:16 2006 UTC vs.
Revision 1.268 by root, Thu Jun 1 05:04:41 2006 UTC

26 26
27 $win->{x} = ($_->{x} + $_->{w} * 0.5) / $::WIDTH if $_->{x} =~ /^[0-9.]+$/; 27 $win->{x} = ($_->{x} + $_->{w} * 0.5) / $::WIDTH if $_->{x} =~ /^[0-9.]+$/;
28 $win->{y} = ($_->{y} + $_->{h} * 0.5) / $::HEIGHT if $_->{y} =~ /^[0-9.]+$/; 28 $win->{y} = ($_->{y} + $_->{h} * 0.5) / $::HEIGHT if $_->{y} =~ /^[0-9.]+$/;
29 $win->{w} = $_->{w} / $::WIDTH if defined $_->{w}; 29 $win->{w} = $_->{w} / $::WIDTH if defined $_->{w};
30 $win->{h} = $_->{h} / $::HEIGHT if defined $_->{h}; 30 $win->{h} = $_->{h} / $::HEIGHT if defined $_->{h};
31
32 $win->{show} = $_->{visible} && $_->{is_toplevel};
31 } 33 }
32 34
33 $layout 35 $layout
34} 36}
35 37
36sub set_layout { 38sub set_layout {
37 my ($layout) = @_; 39 my ($layout) = @_;
40
38 $LAYOUT = $layout; 41 $LAYOUT = $layout;
39} 42}
40 43
41sub check_tooltip { 44sub check_tooltip {
45 return if $ENV{CFPLUS_DEBUG} & 8;
46
42 if (!$GRAB) { 47 if (!$GRAB) {
43 for (my $widget = $HOVER; $widget; $widget = $widget->{parent}) { 48 for (my $widget = $HOVER; $widget; $widget = $widget->{parent}) {
44 if (length $widget->{tooltip}) { 49 if (length $widget->{tooltip}) {
45
46 if ($TOOLTIP->{owner} != $widget) { 50 if ($TOOLTIP->{owner} != $widget) {
47 $TOOLTIP->hide; 51 $TOOLTIP->hide;
48 52
49 $TOOLTIP->{owner} = $widget; 53 $TOOLTIP->{owner} = $widget;
50 54
167sub rescale_widgets { 171sub rescale_widgets {
168 my ($sx, $sy) = @_; 172 my ($sx, $sy) = @_;
169 173
170 for my $widget (values %WIDGET) { 174 for my $widget (values %WIDGET) {
171 if ($widget->{is_toplevel}) { 175 if ($widget->{is_toplevel}) {
172 $widget->{x} += $widget->{w} * 0.5 if $widget->{x} =~ /^[0-9.]+$/; 176 $widget->{x} += int $widget->{w} * 0.5 if $widget->{x} =~ /^[0-9.]+$/;
173 $widget->{y} += $widget->{h} * 0.5 if $widget->{y} =~ /^[0-9.]+$/; 177 $widget->{y} += int $widget->{h} * 0.5 if $widget->{y} =~ /^[0-9.]+$/;
174 178
175 $widget->{x} = int 0.5 + $widget->{x} * $sx if $widget->{x} =~ /^[0-9.]+$/; 179 $widget->{x} = int 0.5 + $widget->{x} * $sx if $widget->{x} =~ /^[0-9.]+$/;
176 $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};
177 $widget->{force_w} = int 0.5 + $widget->{force_w} * $sx if exists $widget->{force_w}; 181 $widget->{force_w} = int 0.5 + $widget->{force_w} * $sx if exists $widget->{force_w};
178 $widget->{y} = int 0.5 + $widget->{y} * $sy if $widget->{y} =~ /^[0-9.]+$/; 182 $widget->{y} = int 0.5 + $widget->{y} * $sy if $widget->{y} =~ /^[0-9.]+$/;
179 $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};
180 $widget->{force_h} = int 0.5 + $widget->{force_h} * $sy if exists $widget->{force_h}; 184 $widget->{force_h} = int 0.5 + $widget->{force_h} * $sy if exists $widget->{force_h};
181 185
182 $widget->{x} -= $widget->{w} * 0.5 if $widget->{x} =~ /^[0-9.]+$/; 186 $widget->{x} -= int $widget->{w} * 0.5 if $widget->{x} =~ /^[0-9.]+$/;
183 $widget->{y} -= $widget->{h} * 0.5 if $widget->{y} =~ /^[0-9.]+$/; 187 $widget->{y} -= int $widget->{h} * 0.5 if $widget->{y} =~ /^[0-9.]+$/;
184 188
185 } 189 }
186 } 190 }
187 191
188 reconfigure_widgets; 192 reconfigure_widgets;
207 h => undef, 211 h => undef,
208 can_events => 1, 212 can_events => 1,
209 @_ 213 @_
210 }, $class; 214 }, $class;
211 215
212 if (my $layout = $CFClient::UI::LAYOUT->{$self->{name}}) {
213 $self->{x} = $layout->{x} * $CFClient::UI::ROOT->{w} if exists $layout->{x};
214 $self->{y} = $layout->{y} * $CFClient::UI::ROOT->{h} if exists $layout->{y};
215 $self->{force_w} = $layout->{w} * $CFClient::UI::ROOT->{w} if exists $layout->{w};
216 $self->{force_h} = $layout->{h} * $CFClient::UI::ROOT->{h} if exists $layout->{h};
217
218 $self->{x} -= $self->{force_w} * 0.5 if exists $layout->{x};
219 $self->{y} -= $self->{force_h} * 0.5 if exists $layout->{y};
220 }
221
222 Scalar::Util::weaken ($CFClient::UI::WIDGET{$self+0} = $self); 216 Scalar::Util::weaken ($CFClient::UI::WIDGET{$self+0} = $self);
223 217
224 for (keys %$self) { 218 for (keys %$self) {
225 if (/^on_(.*)$/) { 219 if (/^on_(.*)$/) {
226 $self->connect ($1 => delete $self->{$_}); 220 $self->connect ($1 => delete $self->{$_});
227 } 221 }
228 } 222 }
229 223
224 if (my $layout = $CFClient::UI::LAYOUT->{$self->{name}}) {
225 $self->{x} = $layout->{x} * $CFClient::UI::ROOT->{alloc_w} if exists $layout->{x};
226 $self->{y} = $layout->{y} * $CFClient::UI::ROOT->{alloc_h} if exists $layout->{y};
227 $self->{force_w} = $layout->{w} * $CFClient::UI::ROOT->{alloc_w} if exists $layout->{w};
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};
234 }
235
230 $self 236 $self
231} 237}
232 238
233sub destroy { 239sub destroy {
234 my ($self) = @_; 240 my ($self) = @_;
350 $self->{x} = $x; 356 $self->{x} = $x;
351 $self->{y} = $y; 357 $self->{y} = $y;
352 $self->update; 358 $self->update;
353 } 359 }
354 360
355 if ($self->{w} != $w || $self->{h} != $h) { 361 if ($self->{alloc_w} != $w || $self->{alloc_h} != $h) {
356 return unless $self->{visible}; 362 return unless $self->{visible};
357 363
364 $self->{alloc_w} = $w;
365 $self->{alloc_h} = $h;
366
358 $self->{root}->{size_alloc}{$self+0} = [$self, $w, $h]; 367 $self->{root}{size_alloc}{$self+0} = $self;
359 } 368 }
360} 369}
361 370
362sub size_allocate { 371sub size_allocate {
363 # nothing to be done 372 # nothing to be done
441 my ($self, $ev, $x, $y) = @_; 450 my ($self, $ev, $x, $y) = @_;
442 451
443 $self->focus_in; 452 $self->focus_in;
444} 453}
445 454
446sub w { $_[0]{w} = $_[1] if @_ > 1; $_[0]{w} }
447sub h { $_[0]{h} = $_[1] if @_ > 1; $_[0]{h} }
448sub x { $_[0]{x} = $_[1] if @_ > 1; $_[0]{x} }
449sub y { $_[0]{y} = $_[1] if @_ > 1; $_[0]{y} }
450sub z { $_[0]{z} = $_[1] if @_ > 1; $_[0]{z} }
451
452sub find_widget { 455sub find_widget {
453 my ($self, $x, $y) = @_; 456 my ($self, $x, $y) = @_;
454 457
455 return () unless $self->{can_events}; 458 return () unless $self->{can_events};
456 459
463 466
464sub set_parent { 467sub set_parent {
465 my ($self, $parent) = @_; 468 my ($self, $parent) = @_;
466 469
467 Scalar::Util::weaken ($self->{parent} = $parent); 470 Scalar::Util::weaken ($self->{parent} = $parent);
468
469 $self->set_visible if $parent->{visible}; 471 $self->set_visible if $parent->{visible};
470} 472}
471 473
472sub connect { 474sub connect {
473 my ($self, $signal, $cb) = @_; 475 my ($self, $signal, $cb) = @_;
494 496
495sub realloc { 497sub realloc {
496 my ($self) = @_; 498 my ($self) = @_;
497 499
498 if ($self->{visible}) { 500 if ($self->{visible}) {
499 return if $self->{root}{realloc}{$self}; 501 return if $self->{root}{realloc}{$self+0};
500 502
501 $self->{root}{realloc}{$self} = $self; 503 $self->{root}{realloc}{$self+0} = $self;
502 $self->{root}->update; 504 $self->{root}->update;
503 } else { 505 } else {
504 delete $self->{req_w}; 506 delete $self->{req_w};
507 delete $self->{req_h};
505 } 508 }
506} 509}
507 510
508sub update { 511sub update {
509 my ($self) = @_; 512 my ($self) = @_;
517 520
518 $self->realloc; 521 $self->realloc;
519 $self->update; 522 $self->update;
520} 523}
521 524
525# using global variables seems a bit hacky, but passing through all drawing
526# functions seems pointless.
527our ($draw_x, $draw_y, $draw_w, $draw_h); # screen rectangle being drawn
528
522sub draw { 529sub draw {
523 my ($self) = @_; 530 my ($self) = @_;
524 531
525 return unless $self->{h} && $self->{w}; 532 return unless $self->{h} && $self->{w};
533
534 # update screen rectangle
535 local $draw_x = $draw_x + $self->{x};
536 local $draw_y = $draw_y + $self->{y};
537 local $draw_w = $draw_x + $self->{w};
538 local $draw_h = $draw_y + $self->{h};
539
540 # skip widgets that are entirely outside the drawing area
541 return if ($draw_x + $self->{w} < 0) || ($draw_x >= $draw_w)
542 || ($draw_y + $self->{h} < 0) || ($draw_y >= $draw_h);
526 543
527 glPushMatrix; 544 glPushMatrix;
528 glTranslate $self->{x}, $self->{y}, 0; 545 glTranslate $self->{x}, $self->{y}, 0;
529 $self->_draw; 546 $self->_draw;
530 glPopMatrix; 547 glPopMatrix;
542 glVertex $x , $y + $self->{h}; 559 glVertex $x , $y + $self->{h};
543 glEnd; 560 glEnd;
544 glDisable GL_BLEND; 561 glDisable GL_BLEND;
545 } 562 }
546 563
547 if ($ENV{CFPLUS_DEBUG}) { 564 if ($ENV{CFPLUS_DEBUG} & 1) {
548 glPushMatrix; 565 glPushMatrix;
549 glColor 1, 1, 0, 1; 566 glColor 1, 1, 0, 1;
550 glTranslate $self->{x} + 0.375, $self->{y} + 0.375; 567 glTranslate $self->{x} + 0.375, $self->{y} + 0.375;
551 glBegin GL_LINE_LOOP; 568 glBegin GL_LINE_LOOP;
552 glVertex 0 , 0; 569 glVertex 0 , 0;
761sub size_request { 778sub size_request {
762 $_[0]{children}[0]->size_request 779 $_[0]{children}[0]->size_request
763} 780}
764 781
765sub size_allocate { 782sub size_allocate {
766 my ($self, $w, $h, $changed) = @_; 783 my ($self, $w, $h) = @_;
767 784
768 $self->{children}[0]->configure (0, 0, $w, $h); 785 $self->{children}[0]->configure (0, 0, $w, $h);
769} 786}
770 787
771############################################################################# 788#############################################################################
788 $ROOT->on_post_alloc ($self => sub { $self->render_child }); 805 $ROOT->on_post_alloc ($self => sub { $self->render_child });
789 $self->SUPER::update; 806 $self->SUPER::update;
790} 807}
791 808
792sub size_allocate { 809sub size_allocate {
793 my ($self, $w, $h, $changed) = @_; 810 my ($self, $w, $h) = @_;
794 811
795 $self->SUPER::size_allocate ($w, $h, $changed); 812 $self->SUPER::size_allocate ($w, $h);
796 $self->update 813 $self->update;
797 if $changed;
798} 814}
799 815
800sub _render { 816sub _render {
817 my ($self) = @_;
818
801 $_[0]{children}[0]->draw; 819 $self->{children}[0]->draw;
802} 820}
803 821
804sub render_child { 822sub render_child {
805 my ($self) = @_; 823 my ($self) = @_;
806 824
807 $self->{texture} = new_from_opengl CFClient::Texture $self->{w}, $self->{h}, sub { 825 $self->{texture} = new_from_opengl CFClient::Texture $self->{w}, $self->{h}, sub {
808 glClearColor 0, 0, 0, 0; 826 glClearColor 0, 0, 0, 0;
809 glClear GL_COLOR_BUFFER_BIT; 827 glClear GL_COLOR_BUFFER_BIT;
810 828
829 {
830 package CFClient::UI::Base;
831
832 ($draw_x, $draw_y, $draw_w, $draw_h) =
833 (0, 0, $self->{w}, $self->{h});
834 }
835
811 $self->_render; 836 $self->_render;
812 }; 837 };
813} 838}
814 839
815sub _draw { 840sub _draw {
816 my ($self) = @_; 841 my ($self) = @_;
817 842
818 my ($w, $h) = ($self->w, $self->h); 843 my ($w, $h) = @$self{qw(w h)};
819 844
820 my $tex = $self->{texture} 845 my $tex = $self->{texture}
821 or return; 846 or return;
822 847
823 glEnable GL_TEXTURE_2D; 848 glEnable GL_TEXTURE_2D;
846} 871}
847 872
848sub size_request { 873sub size_request {
849 my ($self) = @_; 874 my ($self) = @_;
850 875
851 my ($w, $h) = @$self{qw(child_w child_h)} = @{$self->child}{qw(req_w req_h)}; 876 my ($w, $h) = @{$self->child}{qw(req_w req_h)};
852 877
853 $w = 10 if $self->{scroll_x}; 878 $w = 10 if $self->{scroll_x};
854 $h = 10 if $self->{scroll_y}; 879 $h = 10 if $self->{scroll_y};
855 880
856 ($w, $h) 881 ($w, $h)
857} 882}
858 883
859sub size_allocate { 884sub size_allocate {
860 my ($self, $w, $h, $changed) = @_; 885 my ($self, $w, $h) = @_;
861 886
887 my $child = $self->child;
888
862 $w = $self->{child_w} if $self->{scroll_x} && $self->{child_w}; 889 $w = $child->{req_w} if $self->{scroll_x} && $child->{req_w};
863 $h = $self->{child_h} if $self->{scroll_y} && $self->{child_h}; 890 $h = $child->{req_h} if $self->{scroll_y} && $child->{req_h};
864 891
865 $self->child->configure (0, 0, $w, $h); 892 $self->child->configure (0, 0, $w, $h);
866 $self->update; 893 $self->update;
867} 894}
868 895
903 } 930 }
904} 931}
905 932
906sub _render { 933sub _render {
907 my ($self) = @_; 934 my ($self) = @_;
935
936 local $CFClient::UI::Base::draw_x = $CFClient::UI::Base::draw_x - $self->{view_x};
937 local $CFClient::UI::Base::draw_y = $CFClient::UI::Base::draw_y - $self->{view_y};
908 938
909 CFClient::OpenGL::glTranslate -$self->{view_x}, -$self->{view_y}; 939 CFClient::OpenGL::glTranslate -$self->{view_x}, -$self->{view_y};
910 940
911 $self->SUPER::_render; 941 $self->SUPER::_render;
912} 942}
952 my $child = $self->{vp}->child; 982 my $child = $self->{vp}->child;
953 $self->{slider}->set_range ([$self->{slider}{range}[0], 0, $child->{h}, $self->{vp}{h}, 1]); 983 $self->{slider}->set_range ([$self->{slider}{range}[0], 0, $child->{h}, $self->{vp}{h}, 1]);
954} 984}
955 985
956sub size_allocate { 986sub size_allocate {
957 my ($self, $w, $h, $changed) = @_; 987 my ($self, $w, $h) = @_;
958 988
959 $self->SUPER::size_allocate ($w, $h, $changed); 989 $self->SUPER::size_allocate ($w, $h);
960 990
961 my $child = $self->{vp}->child; 991 my $child = $self->{vp}->child;
962 $self->{slider}->set_range ([$self->{slider}{range}[0], 0, $child->{h}, $self->{vp}{h}, 1]); 992 $self->{slider}->set_range ([$self->{slider}{range}[0], 0, $child->{h}, $self->{vp}{h}, 1]);
963} 993}
964 994
1026 1056
1027 my $self = $class->SUPER::new ( 1057 my $self = $class->SUPER::new (
1028 bg => [1, 1, 1, 1], 1058 bg => [1, 1, 1, 1],
1029 border_bg => [1, 1, 1, 1], 1059 border_bg => [1, 1, 1, 1],
1030 border => 0.6, 1060 border => 0.6,
1031 is_toplevel => 1,
1032 can_events => 1, 1061 can_events => 1,
1062 min_w => 16,
1063 min_h => 16,
1033 @_ 1064 @_
1034 ); 1065 );
1035 1066
1036 $self->{title} &&= new CFClient::UI::Label 1067 $self->{title} &&= new CFClient::UI::Label
1037 align => 0, 1068 align => 0,
1056 $h + $self->border * 2, 1087 $h + $self->border * 2,
1057 ) 1088 )
1058} 1089}
1059 1090
1060sub size_allocate { 1091sub size_allocate {
1061 my ($self, $w, $h, $changed) = @_; 1092 my ($self, $w, $h) = @_;
1062
1063 return unless $changed;
1064 1093
1065 $h -= List::Util::max 0, $self->border * 2; 1094 $h -= List::Util::max 0, $self->border * 2;
1066 $w -= List::Util::max 0, $self->border * 2; 1095 $w -= List::Util::max 0, $self->border * 2;
1067 1096
1068 $self->{title}->configure ($self->border, int $self->border - $::FONTSIZE * 2, $w, int $::FONTSIZE * 2) 1097 $self->{title}->configure ($self->border, int $self->border - $::FONTSIZE * 2, $w, int $::FONTSIZE * 2)
1240 (sum @$hs), 1269 (sum @$hs),
1241 ) 1270 )
1242} 1271}
1243 1272
1244sub size_allocate { 1273sub size_allocate {
1245 my ($self, $w, $h, $changed) = @_; 1274 my ($self, $w, $h) = @_;
1246 1275
1247 my ($ws, $hs) = $self->get_wh; 1276 my ($ws, $hs) = $self->get_wh;
1248 1277
1249 my $req_w = (sum @$ws) || 1; 1278 my $req_w = (sum @$ws) || 1;
1250 my $req_h = (sum @$hs) || 1; 1279 my $req_h = (sum @$hs) || 1;
1328 (List::Util::max map $_->{req_h}, @{$self->{children}}), 1357 (List::Util::max map $_->{req_h}, @{$self->{children}}),
1329 ) 1358 )
1330} 1359}
1331 1360
1332sub size_allocate { 1361sub size_allocate {
1333 my ($self, $w, $h, $changed) = @_; 1362 my ($self, $w, $h) = @_;
1334 1363
1335 my $space = $self->{vertical} ? $h : $w; 1364 my $space = $self->{vertical} ? $h : $w;
1336 my $children = $self->{children}; 1365 my $children = $self->{children};
1337 1366
1338 my @req; 1367 my @req;
1421 ellipsise => 3, # end 1450 ellipsise => 3, # end
1422 layout => (new CFClient::Layout), 1451 layout => (new CFClient::Layout),
1423 fontsize => 1, 1452 fontsize => 1,
1424 align => -1, 1453 align => -1,
1425 valign => -1, 1454 valign => -1,
1426 padding => 2, 1455 padding_x => 2,
1456 padding_y => 2,
1427 can_events => 0, 1457 can_events => 0,
1428 %arg 1458 %arg
1429 ); 1459 );
1430 1460
1431 if (exists $self->{template}) { 1461 if (exists $self->{template}) {
1507 1537
1508 $w = List::Util::max $w, $w2; 1538 $w = List::Util::max $w, $w2;
1509 $h = List::Util::max $h, $h2; 1539 $h = List::Util::max $h, $h2;
1510 } 1540 }
1511 1541
1512 ( 1542 ($w, $h)
1513 $w + $self->{padding} * 2,
1514 $h + $self->{padding} * 2,
1515 )
1516} 1543}
1517 1544
1518sub size_allocate { 1545sub size_allocate {
1519 my ($self, $w, $h, $changed) = @_; 1546 my ($self, $w, $h) = @_;
1520 1547
1521 delete $self->{texture} 1548 delete $self->{texture}
1522 if $changed; 1549 unless $w >= $self->{req_w} && $self->{old_w} >= $self->{req_w};
1523} 1550}
1524 1551
1525sub set_fontsize { 1552sub set_fontsize {
1526 my ($self, $fontsize) = @_; 1553 my ($self, $fontsize) = @_;
1527 1554
1544 $self->{layout}->set_single_paragraph_mode ($self->{ellipsise}); 1571 $self->{layout}->set_single_paragraph_mode ($self->{ellipsise});
1545 $self->{layout}->set_height ($self->{fontsize} * $::FONTSIZE); 1572 $self->{layout}->set_height ($self->{fontsize} * $::FONTSIZE);
1546 1573
1547 my $tex = new_from_layout CFClient::Texture $self->{layout}; 1574 my $tex = new_from_layout CFClient::Texture $self->{layout};
1548 1575
1549 $self->{ox} = int ($self->{align} < 0 ? $self->{padding} 1576 $self->{ox} = int ($self->{align} < 0 ? $self->{padding_x}
1550 : $self->{align} > 0 ? $self->{w} - $tex->{w} - $self->{padding} 1577 : $self->{align} > 0 ? $self->{w} - $tex->{w} - $self->{padding_x}
1551 : ($self->{w} - $tex->{w}) * 0.5); 1578 : ($self->{w} - $tex->{w}) * 0.5);
1552 1579
1553 $self->{oy} = int ($self->{valign} < 0 ? $self->{padding} 1580 $self->{oy} = int ($self->{valign} < 0 ? $self->{padding_y}
1554 : $self->{valign} > 0 ? $self->{h} - $tex->{h} - $self->{padding} 1581 : $self->{valign} > 0 ? $self->{h} - $tex->{h} - $self->{padding_y}
1555 : ($self->{h} - $tex->{h}) * 0.5); 1582 : ($self->{h} - $tex->{h}) * 0.5);
1556 1583
1557 $tex 1584 $tex
1558 }; 1585 };
1559 1586
1798 1825
1799sub new { 1826sub new {
1800 my $class = shift; 1827 my $class = shift;
1801 1828
1802 $class->SUPER::new ( 1829 $class->SUPER::new (
1803 padding => 4, 1830 padding_x => 4,
1831 padding_y => 4,
1804 fg => [1, 1, 1], 1832 fg => [1, 1, 1],
1805 active_fg => [0, 0, 1], 1833 active_fg => [0, 0, 1],
1806 can_hover => 1, 1834 can_hover => 1,
1807 align => 0, 1835 align => 0,
1808 valign => 0, 1836 valign => 0,
1855 1883
1856sub new { 1884sub new {
1857 my $class = shift; 1885 my $class = shift;
1858 1886
1859 $class->SUPER::new ( 1887 $class->SUPER::new (
1860 padding => 2, 1888 padding_x => 2,
1889 padding_y => 2,
1861 fg => [1, 1, 1], 1890 fg => [1, 1, 1],
1862 active_fg => [1, 1, 0], 1891 active_fg => [1, 1, 0],
1863 bg => [0, 0, 0, 0.2], 1892 bg => [0, 0, 0, 0.2],
1864 active_bg => [1, 1, 1, 0.5], 1893 active_bg => [1, 1, 1, 0.5],
1865 state => 0, 1894 state => 0,
1869} 1898}
1870 1899
1871sub size_request { 1900sub size_request {
1872 my ($self) = @_; 1901 my ($self) = @_;
1873 1902
1874 ($self->{padding} * 2 + 6) x 2 1903 (6) x 2
1875} 1904}
1876 1905
1877sub button_down { 1906sub button_down {
1878 my ($self, $ev, $x, $y) = @_; 1907 my ($self, $ev, $x, $y) = @_;
1879 1908
1880 if ($x >= $self->{padding} && $x < $self->{w} - $self->{padding} 1909 if ($x >= $self->{padding_x} && $x < $self->{w} - $self->{padding_x}
1881 && $y >= $self->{padding} && $y < $self->{h} - $self->{padding}) { 1910 && $y >= $self->{padding_y} && $y < $self->{h} - $self->{padding_y}) {
1882 $self->{state} = !$self->{state}; 1911 $self->{state} = !$self->{state};
1883 $self->_emit (changed => $self->{state}); 1912 $self->_emit (changed => $self->{state});
1884 } 1913 }
1885} 1914}
1886 1915
1887sub _draw { 1916sub _draw {
1888 my ($self) = @_; 1917 my ($self) = @_;
1889 1918
1890 $self->SUPER::_draw; 1919 $self->SUPER::_draw;
1891 1920
1892 glTranslate $self->{padding} + 0.375, $self->{padding} + 0.375, 0; 1921 glTranslate $self->{padding_x} + 0.375, $self->{padding_y} + 0.375, 0;
1893 1922
1894 my $s = (List::Util::min @$self{qw(w h)}) - $self->{padding} * 2; 1923 my ($w, $h) = @$self{qw(w h)};
1924
1925 my $s = List::Util::min $w - $self->{padding_x} * 2, $h - $self->{padding_y} * 2;
1895 1926
1896 glColor @{ $FOCUS == $self ? $self->{active_fg} : $self->{fg} }; 1927 glColor @{ $FOCUS == $self ? $self->{active_fg} : $self->{fg} };
1897 1928
1898 my $tex = $self->{state} ? $tex[1] : $tex[0]; 1929 my $tex = $self->{state} ? $tex[1] : $tex[0];
1899 1930
2164 fg => [1, 1, 1], 2195 fg => [1, 1, 1],
2165 active_fg => [0, 0, 0], 2196 active_fg => [0, 0, 0],
2166 bg => [0, 0, 0, 0.2], 2197 bg => [0, 0, 0, 0.2],
2167 active_bg => [1, 1, 1, 0.5], 2198 active_bg => [1, 1, 1, 0.5],
2168 range => [0, 0, 100, 10, 0], 2199 range => [0, 0, 100, 10, 0],
2169 req_w => $::WIDTH / 80, 2200 min_w => $::WIDTH / 80,
2170 req_h => $::WIDTH / 80, 2201 min_h => $::WIDTH / 80,
2171 vertical => 0, 2202 vertical => 0,
2172 can_hover => 1, 2203 can_hover => 1,
2173 inner_pad => 0.02, 2204 inner_pad => 0.02,
2174 @_ 2205 @_
2175 ); 2206 );
2215} 2246}
2216 2247
2217sub size_request { 2248sub size_request {
2218 my ($self) = @_; 2249 my ($self) = @_;
2219 2250
2220 my $w = $self->{req_w}; 2251 ($self->{req_w}, $self->{req_h})
2221 my $h = $self->{req_h};
2222
2223 $self->{vertical} ? ($h, $w) : ($w, $h)
2224} 2252}
2225 2253
2226sub button_down { 2254sub button_down {
2227 my ($self, $ev, $x, $y) = @_; 2255 my ($self, $ev, $x, $y) = @_;
2228 2256
2378 $self->{fontsize} = $fontsize; 2406 $self->{fontsize} = $fontsize;
2379 $self->reflow; 2407 $self->reflow;
2380} 2408}
2381 2409
2382sub size_allocate { 2410sub size_allocate {
2383 my ($self, $w, $h, $changed) = @_; 2411 my ($self, $w, $h) = @_;
2384 2412
2385 $self->SUPER::size_allocate ($w, $h, $changed); 2413 $self->SUPER::size_allocate ($w, $h);
2386
2387 return unless $changed;
2388 2414
2389 $self->{layout}->set_font ($self->{font}) if $self->{font}; 2415 $self->{layout}->set_font ($self->{font}) if $self->{font};
2390 $self->{layout}->set_height ($self->{fontsize} * $::FONTSIZE); 2416 $self->{layout}->set_height ($self->{fontsize} * $::FONTSIZE);
2391 $self->{layout}->set_width ($self->{children}[0]{w}); 2417 $self->{layout}->set_width ($self->{children}[0]{w});
2392 2418
2615} 2641}
2616 2642
2617sub set_tooltip_from { 2643sub set_tooltip_from {
2618 my ($self, $widget) = @_; 2644 my ($self, $widget) = @_;
2619 2645
2646 my $tooltip = $widget->{tooltip};
2647
2648 if ($ENV{CFPLUS_DEBUG} & 2) {
2649 $tooltip .= "\n\n" . (ref $widget) . "\n"
2650 . "$widget->{x} $widget->{y} $widget->{w} $widget->{h}\n"
2651 . "req $widget->{req_w} $widget->{req_h}\n"
2652 . "visible $widget->{visible}";
2653 }
2654
2620 $self->add (new CFClient::UI::Label 2655 $self->add (new CFClient::UI::Label
2621 markup => $widget->{tooltip}, 2656 markup => $tooltip,
2622 max_w => ($widget->{tooltip_width} || 0.25) * $::WIDTH, 2657 max_w => ($widget->{tooltip_width} || 0.25) * $::WIDTH,
2623 fontsize => 0.8, 2658 fontsize => 0.8,
2624 fg => [0, 0, 0, 1], 2659 fg => [0, 0, 0, 1],
2625 ellipsise => 0, 2660 ellipsise => 0,
2626 font => ($widget->{tooltip_font} || $::FONT_PROP), 2661 font => ($widget->{tooltip_font} || $::FONT_PROP),
2634 2669
2635 ($w + 4, $h + 4) 2670 ($w + 4, $h + 4)
2636} 2671}
2637 2672
2638sub size_allocate { 2673sub size_allocate {
2639 my ($self, $w, $h, $changed) = @_; 2674 my ($self, $w, $h) = @_;
2640 2675
2641 return unless $changed;
2642
2643 $self->SUPER::size_allocate ($w - 4, $h - 4, $changed); 2676 $self->SUPER::size_allocate ($w - 4, $h - 4);
2644} 2677}
2645 2678
2646sub visibility_change { 2679sub visibility_change {
2647 my ($self, $visible) = @_; 2680 my ($self, $visible) = @_;
2648 2681
2769 $self->SUPER::DESTROY; 2802 $self->SUPER::DESTROY;
2770} 2803}
2771 2804
2772############################################################################# 2805#############################################################################
2773 2806
2774package CFClient::UI::Inventory;
2775
2776our @ISA = CFClient::UI::ScrolledWindow::;
2777
2778sub new {
2779 my $class = shift;
2780
2781 my $self = $class->SUPER::new (
2782 scrolled => (new CFClient::UI::Table col_expand => [0, 1, 0]),
2783 @_,
2784 );
2785
2786 $self
2787}
2788
2789sub set_items {
2790 my ($self, $items) = @_;
2791
2792 $self->{scrolled}->clear;
2793 return unless $items;
2794
2795 my @items = sort {
2796 ($a->{type} <=> $b->{type})
2797 or ($a->{name} cmp $b->{name})
2798 } @$items;
2799
2800 $self->{real_items} = \@items;
2801
2802 my $row = 0;
2803 for my $item (@items) {
2804 CFClient::Item::update_widgets $item;
2805
2806 $self->{scrolled}->add (0, $row, $item->{face_widget});
2807 $self->{scrolled}->add (1, $row, $item->{desc_widget});
2808 $self->{scrolled}->add (2, $row, $item->{weight_widget});
2809
2810 $row++;
2811 }
2812}
2813
2814#############################################################################
2815
2816package CFClient::UI::Menu; 2807package CFClient::UI::Menu;
2817 2808
2818our @ISA = CFClient::UI::FancyFrame::; 2809our @ISA = CFClient::UI::FancyFrame::;
2819 2810
2820use CFClient::OpenGL; 2811use CFClient::OpenGL;
2858 # maybe save $GRAB? must be careful about events... 2849 # maybe save $GRAB? must be careful about events...
2859 $GRAB = $self; 2850 $GRAB = $self;
2860 $self->{button} = $ev->{button}; 2851 $self->{button} = $ev->{button};
2861 2852
2862 $self->show; 2853 $self->show;
2863 $self->move ($ev->{x} - $self->{w} * 0.5, $ev->{y} - $self->{h} * 0.5); 2854 $self->move_abs ($ev->{x} - $self->{w} * 0.5, $ev->{y} - $self->{h} * 0.5);
2864} 2855}
2865 2856
2866sub mouse_motion { 2857sub mouse_motion {
2867 my ($self, $ev, $x, $y) = @_; 2858 my ($self, $ev, $x, $y) = @_;
2868 2859
2993 $self->SUPER::reconfigure; 2984 $self->SUPER::reconfigure;
2994} 2985}
2995 2986
2996############################################################################# 2987#############################################################################
2997 2988
2989package CFClient::UI::Inventory;
2990
2991our @ISA = CFClient::UI::ScrolledWindow::;
2992
2993sub new {
2994 my $class = shift;
2995
2996 my $self = $class->SUPER::new (
2997 scrolled => (new CFClient::UI::Table col_expand => [0, 1, 0]),
2998 @_,
2999 );
3000
3001 $self
3002}
3003
3004sub set_items {
3005 my ($self, $items) = @_;
3006
3007 $self->{scrolled}->clear;
3008 return unless $items;
3009
3010 my @items = sort {
3011 ($a->{type} <=> $b->{type})
3012 or ($a->{name} cmp $b->{name})
3013 } @$items;
3014
3015 $self->{real_items} = \@items;
3016
3017 my $row = 0;
3018 for my $item (@items) {
3019 CFClient::Item::update_widgets $item;
3020
3021 $self->{scrolled}->add (0, $row, $item->{face_widget});
3022 $self->{scrolled}->add (1, $row, $item->{desc_widget});
3023 $self->{scrolled}->add (2, $row, $item->{weight_widget});
3024
3025 $row++;
3026 }
3027}
3028
3029#############################################################################
3030
3031package CFClient::UI::BindEditor;
3032
3033our @ISA = CFClient::UI::FancyFrame::;
3034
3035sub new {
3036 my $class = shift;
3037
3038 my $self = $class->SUPER::new (binding => [], commands => [], @_);
3039
3040 $self->add (my $vb = new CFClient::UI::VBox);
3041
3042
3043 $vb->add ($self->{rec_btn} = new CFClient::UI::Button
3044 text => "start recording",
3045 tooltip => "Start/Stops recording of actions."
3046 ."All subsequent actions after the recording started will be captured."
3047 ."The actions are displayed after the record was stopped."
3048 ."To bind the action you have to click on the 'Bind' button",
3049 on_activate => sub {
3050 unless ($self->{recording}) {
3051 $self->start;
3052 } else {
3053 $self->stop;
3054 }
3055 });
3056
3057 $vb->add (new CFClient::UI::Label text => "Actions:");
3058 $vb->add ($self->{cmdbox} = new CFClient::UI::VBox);
3059
3060 $vb->add (new CFClient::UI::Label text => "Bound to: ");
3061 $vb->add (my $hb = new CFClient::UI::HBox);
3062 $hb->add ($self->{keylbl} = new CFClient::UI::Label expand => 1);
3063 $hb->add (new CFClient::UI::Button
3064 text => "bind",
3065 tooltip => "This opens a query where you have to press the key combination to bind the recorded actions",
3066 on_activate => sub {
3067 $self->ask_for_bind;
3068 });
3069
3070 $vb->add (my $hb = new CFClient::UI::HBox);
3071 $hb->add (new CFClient::UI::Button
3072 text => "ok",
3073 expand => 1,
3074 tooltip => "This closes the binding editor and saves the binding",
3075 on_activate => sub {
3076 $self->hide;
3077 $self->commit;
3078 });
3079
3080 $hb->add (new CFClient::UI::Button
3081 text => "cancel",
3082 expand => 1,
3083 tooltip => "This closes the binding editor without saving",
3084 on_activate => sub {
3085 $self->hide;
3086 $self->{binding_cancel}->()
3087 if $self->{binding_cancel};
3088 });
3089
3090 $self->update_binding_widgets;
3091
3092 $self
3093}
3094
3095sub commit {
3096 my ($self) = @_;
3097 my ($mod, $sym, $cmds) = $self->get_binding;
3098 if ($sym != 0 && @$cmds > 0) {
3099 $::STATUSBOX->add ("Bound actions to '".CFClient::Binder::keycombo_to_name ($mod, $sym)
3100 ."'. Don't forget 'Save Config'!");
3101 $self->{binding_change}->($mod, $sym, $cmds)
3102 if $self->{binding_change};
3103 } else {
3104 $::STATUSBOX->add ("No action bound, no key or action specified!");
3105 $self->{binding_cancel}->()
3106 if $self->{binding_cancel};
3107 }
3108}
3109
3110sub start {
3111 my ($self) = @_;
3112
3113 $self->{rec_btn}->set_text ("stop recording");
3114 $self->{recording} = 1;
3115 $self->clear_command_list;
3116 $::CONN->start_record if $::CONN;
3117}
3118
3119sub stop {
3120 my ($self) = @_;
3121
3122 $self->{rec_btn}->set_text ("start recording");
3123 $self->{recording} = 0;
3124
3125 my $rec;
3126 $rec = $::CONN->stop_record if $::CONN;
3127 return unless ref $rec eq 'ARRAY';
3128 $self->set_command_list ($rec);
3129}
3130
3131# if $commit is true, the binding will be set after the user entered a key combo
3132sub ask_for_bind {
3133 my ($self, $commit) = @_;
3134
3135 CFClient::Binder::open_binding_dialog (sub {
3136 my ($mod, $sym) = @_;
3137 $self->{binding} = [$mod, $sym]; # XXX: how to stop that memleak?
3138 $self->update_binding_widgets;
3139 $self->commit if $commit;
3140 });
3141}
3142
3143# $mod and $sym are the modifiers and key symbol
3144# $cmds is a array ref of strings (the commands)
3145# $cb is the callback that is executed on OK
3146# $ccb is the callback that is executed on CANCEL and
3147# when the binding was unsuccessful on OK
3148sub set_binding {
3149 my ($self, $mod, $sym, $cmds, $cb, $ccb) = @_;
3150
3151 $self->clear_command_list;
3152 $self->{recording} = 0;
3153 $self->{rec_btn}->set_text ("start recording");
3154
3155 $self->{binding} = [$mod, $sym];
3156 $self->{commands} = $cmds;
3157
3158 $self->{binding_change} = $cb;
3159 $self->{binding_cancel} = $ccb;
3160
3161 $self->update_binding_widgets;
3162}
3163
3164# this is a shortcut method that asks for a binding
3165# and then just binds it.
3166sub do_quick_binding {
3167 my ($self, $cmds) = @_;
3168 $self->set_binding (undef, undef, $cmds, sub {
3169 $::CFG->{bindings}->{$_[0]}->{$_[1]} = $_[2];
3170 });
3171 $self->ask_for_bind (1);
3172}
3173
3174sub update_binding_widgets {
3175 my ($self) = @_;
3176 my ($mod, $sym, $cmds) = $self->get_binding;
3177 $self->{keylbl}->set_text (CFClient::Binder::keycombo_to_name ($mod, $sym));
3178 $self->set_command_list ($cmds);
3179}
3180
3181sub get_binding {
3182 my ($self) = @_;
3183 return (
3184 $self->{binding}->[0],
3185 $self->{binding}->[1],
3186 [ grep { defined $_ } @{$self->{commands}} ]
3187 );
3188}
3189
3190sub clear_command_list {
3191 my ($self) = @_;
3192 $self->{cmdbox}->clear ();
3193}
3194
3195sub set_command_list {
3196 my ($self, $cmds) = @_;
3197
3198 $self->{cmdbox}->clear ();
3199 $self->{commands} = $cmds;
3200
3201 my $idx = 0;
3202
3203 for (@$cmds) {
3204 $self->{cmdbox}->add (my $hb = new CFClient::UI::HBox);
3205
3206 my $i = $idx;
3207 $hb->add (new CFClient::UI::Label text => $_);
3208 $hb->add (new CFClient::UI::Button
3209 text => "delete",
3210 tooltip => "Deletes the action from the record",
3211 on_activate => sub {
3212 $self->{cmdbox}->remove ($hb);
3213 $cmds->[$i] = undef;
3214 });
3215
3216
3217 $idx++
3218 }
3219}
3220
3221#############################################################################
3222
3223package CFClient::UI::SpellList;
3224
3225our @ISA = CFClient::UI::FancyFrame::;
3226
3227sub new {
3228 my $class = shift;
3229
3230 my $self = $class->SUPER::new (binding => [], commands => [], @_);
3231
3232 $self->add (new CFClient::UI::ScrolledWindow
3233 scrolled => $self->{spellbox} = new CFClient::UI::Table);
3234
3235 $self;
3236}
3237
3238# XXX: Do sorting? Argl...
3239sub add_spell {
3240 my ($self, $spell) = @_;
3241 $self->{spells}->{$spell->{name}} = $spell;
3242
3243 $self->{spellbox}->add (0, $self->{tbl_idx}, new CFClient::UI::Face
3244 face => $spell->{face},
3245 can_hover => 1,
3246 can_events => 1,
3247 tooltip => $spell->{message});
3248
3249 $self->{spellbox}->add (1, $self->{tbl_idx}, new CFClient::UI::Label
3250 text => $spell->{name},
3251 can_hover => 1,
3252 can_events => 1,
3253 tooltip => $spell->{message},
3254 expand => 1);
3255
3256 $self->{spellbox}->add (2, $self->{tbl_idx}, new CFClient::UI::Label
3257 text => (sprintf "lvl: %2d sp: %2d dmg: %2d",
3258 $spell->{level}, ($spell->{mana} || $spell->{grace}), $spell->{damage}),
3259 expand => 1);
3260
3261 $self->{spellbox}->add (3, $self->{tbl_idx}++, new CFClient::UI::Button
3262 text => "bind to key",
3263 on_activate => sub { $::BIND_EDITOR->do_quick_binding (["cast $spell->{name}"]) });
3264}
3265
3266sub rebuild_spell_list {
3267 my ($self) = @_;
3268 $self->{tbl_idx} = 0;
3269 $self->add_spell ($_) for values %{$self->{spells}};
3270}
3271
3272sub remove_spell {
3273 my ($self, $spell) = @_;
3274 delete $self->{spells}->{$spell->{name}};
3275 $self->rebuild_spell_list;
3276}
3277
3278#############################################################################
3279
2998package CFClient::UI::Root; 3280package CFClient::UI::Root;
2999 3281
3000our @ISA = CFClient::UI::Container::; 3282our @ISA = CFClient::UI::Container::;
3001 3283
3002use CFClient::OpenGL; 3284use CFClient::OpenGL;
3012 Scalar::Util::weaken ($self->{root} = $self); 3294 Scalar::Util::weaken ($self->{root} = $self);
3013 3295
3014 $self 3296 $self
3015} 3297}
3016 3298
3017sub configure {
3018 my ($self, $x, $y, $w, $h) = @_;
3019
3020 $self->{w} = $w;
3021 $self->{h} = $h;
3022}
3023
3024sub reconfigure {
3025 my ($self) = @_;
3026
3027 $self->SUPER::reconfigure;
3028
3029 $self->size_allocate ($self->{w}, $self->{h}, 1)
3030 if $self->{w};
3031}
3032
3033sub size_request { 3299sub size_request {
3034 my ($self) = @_; 3300 my ($self) = @_;
3035 3301
3036 ($self->{w}, $self->{h}) 3302 ($self->{w}, $self->{h})
3037} 3303}
3049 3315
3050 int $coord + 0.5 3316 int $coord + 0.5
3051} 3317}
3052 3318
3053sub size_allocate { 3319sub size_allocate {
3054 my ($self, $w, $h, $changed) = @_; 3320 my ($self, $w, $h) = @_;
3055 3321
3056 for my $child ($self->children) { 3322 for my $child ($self->children) {
3057 my ($X, $Y, $W, $H) = @$child{qw(x y req_w req_h)}; 3323 my ($X, $Y, $W, $H) = @$child{qw(x y req_w req_h)};
3058 3324
3059 $X = $child->{force_x} if exists $child->{force_x}; 3325 $X = $child->{force_x} if exists $child->{force_x};
3127 $_->() 3393 $_->()
3128 for values %{delete $self->{refresh_hook}}; 3394 for values %{delete $self->{refresh_hook}};
3129 } 3395 }
3130 3396
3131 if ($self->{realloc}) { 3397 if ($self->{realloc}) {
3398 my %queue;
3132 my @queue; 3399 my @queue;
3400 my $widget;
3133 3401
3402 outer:
3134 while () { 3403 while () {
3135 if ($self->{realloc}) { 3404 if (my $realloc = delete $self->{realloc}) {
3136 #TODO use array-of-depth approach 3405 for $widget (values %$realloc) {
3406 $widget->{visible} or next; # do not resize invisible widgets
3137 3407
3138 @queue = sort { $a->{visible} <=> $b->{visible} } 3408 $queue{$widget+0}++ and next; # duplicates are common
3139 @queue, values %{delete $self->{realloc}}; 3409
3410 push @{ $queue[$widget->{visible}] }, $widget;
3411 }
3140 } 3412 }
3141 3413
3414 while () {
3415 @queue or last outer;
3416
3417 $widget = pop @{ $queue[-1] || [] }
3418 and last;
3419
3420 pop @queue;
3421 }
3422
3423 delete $queue{$widget+0};
3424
3425 my ($w, $h) = $widget->size_request;
3426
3427 $w = List::Util::max $widget->{min_w}, $w + $widget->{padding_x} * 2;
3428 $h = List::Util::max $widget->{min_h}, $h + $widget->{padding_y} * 2;
3429
3430 $w = $widget->{force_w} if exists $widget->{force_w};
3431 $h = $widget->{force_h} if exists $widget->{force_h};
3432
3433 if ($widget->{req_w} != $w || $widget->{req_h} != $h
3434 || delete $widget->{force_realloc}) {
3435 $widget->{req_w} = $w;
3436 $widget->{req_h} = $h;
3437
3438 $self->{size_alloc}{$widget+0} = $widget;
3439
3440 if (my $parent = $widget->{parent}) {
3441 $self->{realloc}{$parent+0} = $parent
3442 unless $queue{$parent+0};
3443
3444 $parent->{force_size_alloc} = 1;
3445 $self->{size_alloc}{$parent+0} = $parent;
3446 }
3447 }
3448
3449 delete $self->{realloc}{$widget+0};
3450 }
3451 }
3452
3453 while (my $size_alloc = delete $self->{size_alloc}) {
3454 my @queue = sort { $b->{visible} <=> $a->{visible} }
3455 values %$size_alloc;
3456
3457 while () {
3142 my $widget = pop @queue || last; 3458 my $widget = pop @queue || last;
3143 3459
3144 $widget->{visible} or last; # do not resize invisible widgets 3460 my ($w, $h) = @$widget{qw(alloc_w alloc_h)};
3145
3146 my ($w, $h) = exists $widget->{force_w} && exists $widget->{force_h}
3147 ? @$widget{qw(force_w force_h)}
3148 : $widget->size_request;
3149
3150 my $min_size = $widget->{is_toplevel} ? 16 : 0;
3151
3152 $w = $min_size if $w < $min_size;
3153 $h = $min_size if $h < $min_size;
3154
3155 $widget->{req_w} = $w;
3156 $widget->{req_h} = $h;
3157
3158 $self->{size_alloc}{$widget} = [$widget, undef, undef];
3159
3160 push @queue, $widget->{parent}
3161 if $widget->{parent};
3162 }
3163 }
3164
3165 while (my $size_alloc = delete $self->{size_alloc}) {
3166 my @queue = sort $b->[0]{visible} <=> $a->[0]{visible},
3167 values %$size_alloc;
3168
3169 while () {
3170 my ($widget, $w, $h) = @{ pop @queue or last };
3171
3172 $w = $widget->{w} || $widget->{req_w} unless defined $w;
3173 $h = $widget->{h} || $widget->{req_h} unless defined $h;
3174 3461
3175 $w = 0 if $w < 0; 3462 $w = 0 if $w < 0;
3176 $h = 0 if $h < 0; 3463 $h = 0 if $h < 0;
3177 3464
3178 $w = int $w + 0.5; 3465 $w = int $w + 0.5;
3179 $h = int $h + 0.5; 3466 $h = int $h + 0.5;
3180 3467
3181 my $changed = $widget->{w} != $w || $widget->{h} != $h; 3468 if ($widget->{w} != $w || $widget->{h} != $h || delete $widget->{force_size_alloc}) {
3469 $widget->{old_w} = $widget->{w};
3470 $widget->{old_h} = $widget->{h};
3182 3471
3183 $widget->{w} = $w; 3472 $widget->{w} = $w;
3184 $widget->{h} = $h; 3473 $widget->{h} = $h;
3185 3474
3186 $widget->emit (size_allocate => $w, $h, $changed); 3475 $widget->emit (size_allocate => $w, $h);
3476 }
3187 } 3477 }
3188 } 3478 }
3189 3479
3190 while ($self->{post_alloc_hook}) { 3480 while ($self->{post_alloc_hook}) {
3191 $_->() 3481 $_->()
3192 for values %{delete $self->{post_alloc_hook}}; 3482 for values %{delete $self->{post_alloc_hook}};
3193 } 3483 }
3484
3194 3485
3195 glViewport 0, 0, $::WIDTH, $::HEIGHT; 3486 glViewport 0, 0, $::WIDTH, $::HEIGHT;
3196 glClearColor +($::CFG->{fow_intensity}) x 3, 1; 3487 glClearColor +($::CFG->{fow_intensity}) x 3, 1;
3197 glClear GL_COLOR_BUFFER_BIT; 3488 glClear GL_COLOR_BUFFER_BIT;
3198 3489
3200 glLoadIdentity; 3491 glLoadIdentity;
3201 glOrtho 0, $::WIDTH, $::HEIGHT, 0, -10000, 10000; 3492 glOrtho 0, $::WIDTH, $::HEIGHT, 0, -10000, 10000;
3202 glMatrixMode GL_MODELVIEW; 3493 glMatrixMode GL_MODELVIEW;
3203 glLoadIdentity; 3494 glLoadIdentity;
3204 3495
3496 {
3497 package CFClient::UI::Base;
3498
3499 ($draw_x, $draw_y, $draw_w, $draw_h) =
3500 (0, 0, $self->{w}, $self->{h});
3501 }
3502
3205 $self->_draw; 3503 $self->_draw;
3206} 3504}
3207 3505
3208############################################################################# 3506#############################################################################
3209 3507

Diff Legend

Removed lines
+ Added lines
< Changed lines
> Changed lines