… | |
… | |
31 | $TOOLTIP->set_tooltip_from ($widget); |
31 | $TOOLTIP->set_tooltip_from ($widget); |
32 | $TOOLTIP->show; |
32 | $TOOLTIP->show; |
33 | |
33 | |
34 | my ($x, $y) = $widget->coord2global ($widget->{w}, 0); |
34 | my ($x, $y) = $widget->coord2global ($widget->{w}, 0); |
35 | |
35 | |
36 | if ($x + $TOOLTIP->{w} > $::WIDTH) { |
|
|
37 | ($x, $y) = $widget->coord2global (-$TOOLTIP->{w}, 0); |
36 | ($x, $y) = $widget->coord2global (-$TOOLTIP->{w}, 0) |
38 | } |
37 | if $x + $TOOLTIP->{w} > $::WIDTH; |
39 | |
38 | |
40 | $TOOLTIP->move ($x, $y); |
39 | $TOOLTIP->move ($x, $y); |
41 | $TOOLTIP->check_size; |
40 | $TOOLTIP->check_size; |
42 | $TOOLTIP->update; |
41 | $TOOLTIP->update; |
43 | } |
42 | } |
… | |
… | |
141 | |
140 | |
142 | # call when resolution changes etc. |
141 | # call when resolution changes etc. |
143 | sub rescale_widgets { |
142 | sub rescale_widgets { |
144 | my ($sx, $sy) = @_; |
143 | my ($sx, $sy) = @_; |
145 | |
144 | |
|
|
145 | # make a copy, otherwise for complains about freed values. |
146 | for my $widget (values %WIDGET) { |
146 | my @widgets = values %WIDGET; |
|
|
147 | |
|
|
148 | for my $widget (@widgets) { |
147 | if ($widget->{toplevel}) { |
149 | if ($widget->{toplevel}) { |
148 | $widget->{x} = int 0.5 + $widget->{x} * $sx if exists $widget->{x}; |
150 | $widget->{x} = int 0.5 + $widget->{x} * $sx if exists $widget->{x}; |
149 | $widget->{w} = int 0.5 + $widget->{w} * $sx if exists $widget->{w}; |
151 | $widget->{w} = int 0.5 + $widget->{w} * $sx if exists $widget->{w}; |
150 | $widget->{req_w} = int 0.5 + $widget->{req_w} * $sx if exists $widget->{req_w}; |
152 | $widget->{req_w} = int 0.5 + $widget->{req_w} * $sx if exists $widget->{req_w}; |
151 | $widget->{user_w} = int 0.5 + $widget->{user_w} * $sx if exists $widget->{user_w}; |
153 | $widget->{user_w} = int 0.5 + $widget->{user_w} * $sx if exists $widget->{user_w}; |
… | |
… | |
281 | sub set_max_size { |
283 | sub set_max_size { |
282 | my ($self, $w, $h) = @_; |
284 | my ($self, $w, $h) = @_; |
283 | |
285 | |
284 | delete $self->{max_w}; $self->{max_w} = $w if $w; |
286 | delete $self->{max_w}; $self->{max_w} = $w if $w; |
285 | delete $self->{max_h}; $self->{max_h} = $h if $h; |
287 | delete $self->{max_h}; $self->{max_h} = $h if $h; |
|
|
288 | } |
|
|
289 | |
|
|
290 | sub set_tooltip { |
|
|
291 | my ($self, $tooltip) = @_; |
|
|
292 | |
|
|
293 | $self->{tooltip} = $tooltip; |
|
|
294 | |
|
|
295 | if ($CFClient::UI::TOOLTIP->{owner} == $self) { |
|
|
296 | delete $CFClient::UI::TOOLTIP->{owner}; |
|
|
297 | CFClient::UI::check_tooltip; |
|
|
298 | } |
286 | } |
299 | } |
287 | |
300 | |
288 | # translate global coordinates to local coordinate system |
301 | # translate global coordinates to local coordinate system |
289 | sub coord2local { |
302 | sub coord2local { |
290 | my ($self, $x, $y) = @_; |
303 | my ($self, $x, $y) = @_; |
… | |
… | |
376 | glVertex $self->{w}, 0; |
389 | glVertex $self->{w}, 0; |
377 | glVertex $self->{w}, $self->{h}; |
390 | glVertex $self->{w}, $self->{h}; |
378 | glVertex 0 , $self->{h}; |
391 | glVertex 0 , $self->{h}; |
379 | glEnd; |
392 | glEnd; |
380 | glPopMatrix; |
393 | glPopMatrix; |
381 | CFClient::UI::Label->new (w => $self->{w}, h => $self->{h}, text => $self, fontsize => 0)->_draw; |
394 | #CFClient::UI::Label->new (w => $self->{w}, h => $self->{h}, text => $self, fontsize => 0)->_draw; |
382 | } |
395 | } |
383 | } |
396 | } |
384 | |
397 | |
385 | sub _draw { |
398 | sub _draw { |
386 | my ($self) = @_; |
399 | my ($self) = @_; |
… | |
… | |
456 | my $class = shift; |
469 | my $class = shift; |
457 | |
470 | |
458 | # range [value, low, high, page] |
471 | # range [value, low, high, page] |
459 | |
472 | |
460 | $class->SUPER::new ( |
473 | $class->SUPER::new ( |
461 | bg => [0, 0, 0, 0.2], |
474 | #bg => [0, 0, 0, 0.2], |
462 | active_bg => [1, 1, 1, 0.5], |
475 | #active_bg => [1, 1, 1, 0.5], |
463 | @_ |
476 | @_ |
464 | ) |
477 | ) |
465 | } |
478 | } |
466 | |
479 | |
467 | sub _draw { |
480 | sub _draw { |
468 | my ($self) = @_; |
481 | my ($self) = @_; |
469 | |
482 | |
|
|
483 | my $color = $FOCUS == $self && $self->{active_bg} |
|
|
484 | ? $self->{active_bg} |
|
|
485 | : $self->{bg}; |
|
|
486 | |
|
|
487 | if ($color && (@$color < 4 || $color->[3])) { |
470 | my ($w, $h) = @$self{qw(w h)}; |
488 | my ($w, $h) = @$self{qw(w h)}; |
471 | |
489 | |
472 | glEnable GL_BLEND; |
490 | glEnable GL_BLEND; |
473 | glBlendFunc GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA; |
491 | glBlendFunc GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA; |
474 | glColor @{ $FOCUS == $self ? $self->{active_bg} : $self->{bg} }; |
492 | glColor @$color; |
475 | |
493 | |
476 | glBegin GL_QUADS; |
494 | glBegin GL_QUADS; |
477 | glVertex 0 , 0; |
495 | glVertex 0 , 0; |
478 | glVertex 0 , $h; |
496 | glVertex 0 , $h; |
479 | glVertex $w, $h; |
497 | glVertex $w, $h; |
480 | glVertex $w, 0; |
498 | glVertex $w, 0; |
481 | glEnd; |
499 | glEnd; |
482 | |
500 | |
483 | glDisable GL_BLEND; |
501 | glDisable GL_BLEND; |
|
|
502 | } |
484 | } |
503 | } |
485 | |
504 | |
486 | ############################################################################# |
505 | ############################################################################# |
487 | |
506 | |
488 | package CFClient::UI::Empty; |
507 | package CFClient::UI::Empty; |
… | |
… | |
672 | $self->{texture} = new_from_opengl CFClient::Texture $self->{w}, $self->{h}, sub { |
691 | $self->{texture} = new_from_opengl CFClient::Texture $self->{w}, $self->{h}, sub { |
673 | glClearColor 0, 0, 0, 0; |
692 | glClearColor 0, 0, 0, 0; |
674 | glClear GL_COLOR_BUFFER_BIT; |
693 | glClear GL_COLOR_BUFFER_BIT; |
675 | |
694 | |
676 | $self->_render; |
695 | $self->_render; |
677 | # glColorMask 1, 1, 1, 0; |
|
|
678 | # glEnable GL_BLEND; |
|
|
679 | # glBlendFunc GL_SRC_ALPHA, GL_ZERO; |
|
|
680 | # glRasterPos 0, 0; |
|
|
681 | # glCopyPixels 0, 0, $self->{w}, $self->{h}; |
|
|
682 | # glDisable GL_BLEND; |
|
|
683 | # glColorMask 1, 1, 1, 1; |
|
|
684 | }; |
696 | }; |
685 | } |
697 | } |
686 | |
698 | |
687 | sub _draw { |
699 | sub _draw { |
688 | my ($self) = @_; |
700 | my ($self) = @_; |
… | |
… | |
692 | my $tex = $self->{texture} |
704 | my $tex = $self->{texture} |
693 | or return; |
705 | or return; |
694 | |
706 | |
695 | glEnable GL_TEXTURE_2D; |
707 | glEnable GL_TEXTURE_2D; |
696 | glTexEnv GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_REPLACE; |
708 | glTexEnv GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_REPLACE; |
697 | glColor 0, 0, 0, 1; |
709 | glColor 1, 1, 1, 1; |
698 | |
710 | |
699 | $tex->draw_quad_alpha_premultiplied (0, 0, $w, $h); |
711 | $tex->draw_quad_alpha_premultiplied (0, 0, $w, $h); |
700 | |
712 | |
701 | glDisable GL_TEXTURE_2D; |
713 | glDisable GL_TEXTURE_2D; |
702 | } |
714 | } |
… | |
… | |
896 | my ($self, $w, $h) = @_; |
908 | my ($self, $w, $h) = @_; |
897 | |
909 | |
898 | $h -= List::Util::max 0, $self->border * 2; |
910 | $h -= List::Util::max 0, $self->border * 2; |
899 | $w -= List::Util::max 0, $self->border * 2; |
911 | $w -= List::Util::max 0, $self->border * 2; |
900 | |
912 | |
901 | $self->{title}->configure ($self->border, $self->border - $::FONTSIZE * 2, $w, $::FONTSIZE * 2) |
913 | $self->{title}->configure ($self->border, int $self->border - $::FONTSIZE * 2, $w, int $::FONTSIZE * 2) |
902 | if $self->{title}; |
914 | if $self->{title}; |
903 | |
915 | |
904 | $self->child->configure ($self->border, $self->border, $w, $h); |
916 | $self->child->configure ($self->border, $self->border, $w, $h); |
905 | } |
917 | } |
906 | |
918 | |
… | |
… | |
1267 | |
1279 | |
1268 | ############################################################################# |
1280 | ############################################################################# |
1269 | |
1281 | |
1270 | package CFClient::UI::Label; |
1282 | package CFClient::UI::Label; |
1271 | |
1283 | |
1272 | our @ISA = CFClient::UI::Base::; |
1284 | our @ISA = CFClient::UI::DrawBG::; |
1273 | |
1285 | |
1274 | use CFClient::OpenGL; |
1286 | use CFClient::OpenGL; |
1275 | |
1287 | |
1276 | sub new { |
1288 | sub new { |
1277 | my ($class, %arg) = @_; |
1289 | my ($class, %arg) = @_; |
1278 | |
1290 | |
1279 | my $self = $class->SUPER::new ( |
1291 | my $self = $class->SUPER::new ( |
1280 | fg => [1, 1, 1], |
1292 | fg => [1, 1, 1], |
|
|
1293 | #bg => none |
|
|
1294 | #active_bg => none |
1281 | #font => default_font |
1295 | #font => default_font |
1282 | #text => initial text |
1296 | #text => initial text |
1283 | #markup => initial narkup |
1297 | #markup => initial narkup |
|
|
1298 | #max_w => maximum pixel width |
|
|
1299 | ellipsise => 3, # end |
1284 | layout => (new CFClient::Layout), |
1300 | layout => (new CFClient::Layout), |
1285 | fontsize => 1, |
1301 | fontsize => 1, |
1286 | align => -1, |
1302 | align => -1, |
1287 | valign => -1, |
1303 | valign => -1, |
1288 | padding => 2, |
1304 | padding => 2, |
… | |
… | |
1303 | } |
1319 | } |
1304 | |
1320 | |
1305 | $self |
1321 | $self |
1306 | } |
1322 | } |
1307 | |
1323 | |
1308 | sub escape { |
1324 | sub escape($) { |
1309 | local $_ = $_[1]; |
1325 | local $_ = $_[0]; |
1310 | |
1326 | |
1311 | s/&/&/g; |
1327 | s/&/&/g; |
1312 | s/>/>/g; |
1328 | s/>/>/g; |
1313 | s/</</g; |
1329 | s/</</g; |
1314 | |
1330 | |
1315 | $_[1] |
1331 | $_ |
1316 | } |
1332 | } |
1317 | |
1333 | |
1318 | sub update { |
1334 | sub update { |
1319 | my ($self) = @_; |
1335 | my ($self) = @_; |
1320 | |
1336 | |
… | |
… | |
1353 | sub size_request { |
1369 | sub size_request { |
1354 | my ($self) = @_; |
1370 | my ($self) = @_; |
1355 | |
1371 | |
1356 | $self->{layout}->set_font ($self->{font}) if $self->{font}; |
1372 | $self->{layout}->set_font ($self->{font}) if $self->{font}; |
1357 | $self->{layout}->set_width ($self->{max_w} || -1); |
1373 | $self->{layout}->set_width ($self->{max_w} || -1); |
|
|
1374 | $self->{layout}->set_ellipsise ($self->{ellipsise}); |
|
|
1375 | $self->{layout}->set_single_paragraph_mode ($self->{ellipsise}); |
1358 | $self->{layout}->set_height ($self->{fontsize} * $::FONTSIZE); |
1376 | $self->{layout}->set_height ($self->{fontsize} * $::FONTSIZE); |
1359 | |
1377 | |
1360 | my ($w, $h) = $self->{layout}->size; |
1378 | my ($w, $h) = $self->{layout}->size; |
1361 | |
1379 | |
1362 | if (exists $self->{template}) { |
1380 | if (exists $self->{template}) { |
… | |
… | |
1392 | } |
1410 | } |
1393 | |
1411 | |
1394 | sub _draw { |
1412 | sub _draw { |
1395 | my ($self) = @_; |
1413 | my ($self) = @_; |
1396 | |
1414 | |
|
|
1415 | $self->SUPER::_draw; # draw background, if applicable |
|
|
1416 | |
1397 | my $tex = $self->{texture} ||= do { |
1417 | my $tex = $self->{texture} ||= do { |
1398 | $self->{layout}->set_foreground (@{$self->{fg}}); |
1418 | $self->{layout}->set_foreground (@{$self->{fg}}); |
1399 | $self->{layout}->set_font ($self->{font}) if $self->{font}; |
1419 | $self->{layout}->set_font ($self->{font}) if $self->{font}; |
1400 | $self->{layout}->set_width ($self->{w}); |
1420 | $self->{layout}->set_width ($self->{w}); |
|
|
1421 | $self->{layout}->set_ellipsise ($self->{ellipsise}); |
|
|
1422 | $self->{layout}->set_single_paragraph_mode ($self->{ellipsise}); |
1401 | $self->{layout}->set_height (List::Util::min $self->{h}, $self->{fontsize} * $::FONTSIZE); |
1423 | $self->{layout}->set_height ($self->{fontsize} * $::FONTSIZE); |
1402 | |
1424 | |
1403 | my $tex = new_from_layout CFClient::Texture $self->{layout}; |
1425 | my $tex = new_from_layout CFClient::Texture $self->{layout}; |
1404 | |
1426 | |
1405 | $self->{ox} = int $self->{align} < 0 ? $self->{padding} |
1427 | $self->{ox} = int ($self->{align} < 0 ? $self->{padding} |
1406 | : $self->{align} > 0 ? $self->{w} - $tex->{w} - $self->{padding} |
1428 | : $self->{align} > 0 ? $self->{w} - $tex->{w} - $self->{padding} |
1407 | : ($self->{w} - $tex->{w}) * 0.5; |
1429 | : ($self->{w} - $tex->{w}) * 0.5); |
1408 | |
1430 | |
1409 | $self->{oy} = int $self->{valign} < 0 ? $self->{padding} |
1431 | $self->{oy} = int ($self->{valign} < 0 ? $self->{padding} |
1410 | : $self->{valign} > 0 ? $self->{h} - $tex->{h} - $self->{padding} |
1432 | : $self->{valign} > 0 ? $self->{h} - $tex->{h} - $self->{padding} |
1411 | : ($self->{h} - $tex->{h}) * 0.5; |
1433 | : ($self->{h} - $tex->{h}) * 0.5); |
1412 | |
1434 | |
1413 | $tex |
1435 | $tex |
1414 | }; |
1436 | }; |
1415 | |
1437 | |
1416 | glEnable GL_TEXTURE_2D; |
1438 | glEnable GL_TEXTURE_2D; |
… | |
… | |
1471 | sub set_text { |
1493 | sub set_text { |
1472 | my ($self, $text) = @_; |
1494 | my ($self, $text) = @_; |
1473 | |
1495 | |
1474 | $self->{cursor} = length $text; |
1496 | $self->{cursor} = length $text; |
1475 | $self->_set_text ($text); |
1497 | $self->_set_text ($text); |
|
|
1498 | $self->update; |
1476 | $self->check_size; |
1499 | $self->check_size; |
1477 | $self->update; |
|
|
1478 | } |
1500 | } |
1479 | |
1501 | |
1480 | sub get_text { |
1502 | sub get_text { |
1481 | $_[0]{text} |
1503 | $_[0]{text} |
1482 | } |
1504 | } |
… | |
… | |
1522 | substr $text, $self->{cursor}++, 0, chr $uni; |
1544 | substr $text, $self->{cursor}++, 0, chr $uni; |
1523 | } |
1545 | } |
1524 | |
1546 | |
1525 | $self->_set_text ($text); |
1547 | $self->_set_text ($text); |
1526 | $self->update; |
1548 | $self->update; |
|
|
1549 | $self->check_size; |
1527 | } |
1550 | } |
1528 | |
1551 | |
1529 | sub focus_in { |
1552 | sub focus_in { |
1530 | my ($self) = @_; |
1553 | my ($self) = @_; |
1531 | |
1554 | |
… | |
… | |
1660 | my $class = shift; |
1683 | my $class = shift; |
1661 | |
1684 | |
1662 | $class->SUPER::new ( |
1685 | $class->SUPER::new ( |
1663 | padding => 4, |
1686 | padding => 4, |
1664 | fg => [1, 1, 1], |
1687 | fg => [1, 1, 1], |
1665 | bg => [1, 1, 1, 0.2], |
|
|
1666 | active_fg => [0, 0, 1], |
1688 | active_fg => [0, 0, 1], |
1667 | can_hover => 1, |
1689 | can_hover => 1, |
1668 | align => 0, |
1690 | align => 0, |
1669 | valign => 0, |
1691 | valign => 0, |
1670 | can_events => 1, |
1692 | can_events => 1, |
… | |
… | |
1718 | |
1740 | |
1719 | $class->SUPER::new ( |
1741 | $class->SUPER::new ( |
1720 | padding => 2, |
1742 | padding => 2, |
1721 | fg => [1, 1, 1], |
1743 | fg => [1, 1, 1], |
1722 | active_fg => [1, 1, 0], |
1744 | active_fg => [1, 1, 0], |
|
|
1745 | bg => [0, 0, 0, 0.2], |
|
|
1746 | active_bg => [1, 1, 1, 0.5], |
1723 | state => 0, |
1747 | state => 0, |
1724 | can_hover => 1, |
1748 | can_hover => 1, |
1725 | @_ |
1749 | @_ |
1726 | ) |
1750 | ) |
1727 | } |
1751 | } |
… | |
… | |
2011 | qw(s1_slider.png s1_slider_bg.png); |
2035 | qw(s1_slider.png s1_slider_bg.png); |
2012 | |
2036 | |
2013 | sub new { |
2037 | sub new { |
2014 | my $class = shift; |
2038 | my $class = shift; |
2015 | |
2039 | |
2016 | # range [value, low, high, page] |
2040 | # range [value, low, high, page, unit] |
2017 | |
2041 | |
2018 | # TODO: 0-width page |
2042 | # TODO: 0-width page |
2019 | # TODO: req_w/h are wrong with vertical |
2043 | # TODO: req_w/h are wrong with vertical |
2020 | # TODO: calculations are off |
2044 | # TODO: calculations are off |
2021 | my $self = $class->SUPER::new ( |
2045 | my $self = $class->SUPER::new ( |
2022 | fg => [1, 1, 1], |
2046 | fg => [1, 1, 1], |
2023 | active_fg => [0, 0, 0], |
2047 | active_fg => [0, 0, 0], |
|
|
2048 | bg => [0, 0, 0, 0.2], |
|
|
2049 | active_bg => [1, 1, 1, 0.5], |
2024 | range => [0, 0, 100, 10], |
2050 | range => [0, 0, 100, 10], |
2025 | req_w => $::WIDTH / 80, |
2051 | req_w => $::WIDTH / 80, |
2026 | req_h => $::WIDTH / 80, |
2052 | req_h => $::WIDTH / 80, |
2027 | vertical => 0, |
2053 | vertical => 0, |
2028 | can_hover => 1, |
2054 | can_hover => 1, |
2029 | inner_pad => 5, |
2055 | inner_pad => .05, |
2030 | @_ |
2056 | @_ |
2031 | ); |
2057 | ); |
2032 | |
2058 | |
|
|
2059 | $self->set_value ($self->{range}[0]); |
|
|
2060 | $self->update; |
|
|
2061 | |
2033 | $self |
2062 | $self |
|
|
2063 | } |
|
|
2064 | |
|
|
2065 | sub set_value { |
|
|
2066 | my ($self, $value) = @_; |
|
|
2067 | |
|
|
2068 | my ($old_value, $lo, $hi, $page, $unit) = @{$self->{range}}; |
|
|
2069 | |
|
|
2070 | $hi = $lo + 1 if $hi <= $lo; |
|
|
2071 | |
|
|
2072 | $value = $lo if $value < $lo; |
|
|
2073 | $value = $hi if $value > $hi; |
|
|
2074 | |
|
|
2075 | $value = $lo + $unit * int +($value - $lo + $unit * 0.5) / $unit |
|
|
2076 | if $unit; |
|
|
2077 | |
|
|
2078 | $page = $hi - $lo if $page > $hi - $lo; |
|
|
2079 | |
|
|
2080 | @{$self->{range}} = ($value, $lo, $hi, $page, $unit); |
|
|
2081 | |
|
|
2082 | if ($value != $old_value) { |
|
|
2083 | $self->emit (changed => $value); |
|
|
2084 | $self->update; |
|
|
2085 | } |
2034 | } |
2086 | } |
2035 | |
2087 | |
2036 | sub size_request { |
2088 | sub size_request { |
2037 | my ($self) = @_; |
2089 | my ($self) = @_; |
2038 | |
2090 | |
… | |
… | |
2051 | |
2103 | |
2052 | sub mouse_motion { |
2104 | sub mouse_motion { |
2053 | my ($self, $ev, $x, $y) = @_; |
2105 | my ($self, $ev, $x, $y) = @_; |
2054 | |
2106 | |
2055 | if ($GRAB == $self) { |
2107 | if ($GRAB == $self) { |
|
|
2108 | my ($x, $w) = $self->{vertical} ? ($y, $self->{h}) : ($x, $self->{w}); |
|
|
2109 | |
|
|
2110 | my (undef, $lo, $hi, $page) = @{$self->{range}}; |
|
|
2111 | |
|
|
2112 | $x = $x / ($w * (1 - 2 * $self->{inner_pad})) - $self->{inner_pad}; |
|
|
2113 | |
|
|
2114 | $self->set_value ($x * ($hi - $lo) + $lo); |
|
|
2115 | } |
|
|
2116 | } |
|
|
2117 | |
|
|
2118 | sub update { |
|
|
2119 | my ($self) = @_; |
|
|
2120 | |
|
|
2121 | $CFClient::UI::ROOT->on_post_alloc ($self => sub { |
|
|
2122 | $self->set_value ($self->{range}[0]); |
|
|
2123 | |
2056 | my ($value, $lo, $hi, $page) = @{$self->{range}}; |
2124 | my ($value, $lo, $hi, $page) = @{$self->{range}}; |
2057 | |
2125 | |
2058 | my ($x, $w) = $self->{vertical} ? ($y, $self->{h}) : ($x, $self->{w}); |
2126 | my $inner_w = 1 - 2 * $self->{inner_pad}; |
2059 | |
2127 | |
2060 | my $inner_pad_px = $self->_calc_inner_pad_px ($w); |
2128 | $self->{scale} = ($inner_w / ($hi - $lo)) || 1; |
2061 | my $inner_w = $w - $inner_pad_px * 2; # * 2 for left & right |
|
|
2062 | |
2129 | |
2063 | $x -= $inner_pad_px; # substract the padding |
2130 | $page = $self->{scale} * $page || 10 / ($self->{w} || 1); |
2064 | $x = $x * ($hi - $lo) / $inner_w + $lo; |
2131 | $value = $self->{scale} * ($value - $lo); |
2065 | $x = $lo if $x < $lo; |
|
|
2066 | $x = $hi - $page if $x > $hi - $page; |
|
|
2067 | $self->{range}[0] = $x; |
|
|
2068 | |
2132 | |
2069 | $self->emit (changed => $x); |
2133 | $value = $self->{inner_pad} + ($value - $page * 0.5); |
2070 | $self->update; |
2134 | |
|
|
2135 | $value = 0 if $value < 0; |
|
|
2136 | $page = 1 - $value if $value + $page > 1; |
|
|
2137 | |
|
|
2138 | $self->{knob_x} = $value; |
|
|
2139 | $self->{knob_w} = $page; |
2071 | } |
2140 | }); |
2072 | } |
|
|
2073 | |
2141 | |
2074 | # the inner_* stuff is for generating a padding for the slider handle, |
2142 | $self->SUPER::update; |
2075 | # so that the handle doesn't leave the texture. This calculation isn't 100% |
|
|
2076 | # correct propably, but it does the job for now |
|
|
2077 | sub _calc_inner_pad_px { |
|
|
2078 | my ($self, $w) = @_; |
|
|
2079 | ($w / 100) * $self->{inner_pad} # % to pixels |
|
|
2080 | } |
2143 | } |
2081 | |
2144 | |
2082 | sub _draw { |
2145 | sub _draw { |
2083 | my ($self) = @_; |
2146 | my ($self) = @_; |
2084 | |
2147 | |
2085 | $self->SUPER::_draw (); |
2148 | $self->SUPER::_draw (); |
2086 | |
2149 | |
2087 | my ($w, $h) = @$self{qw(w h)}; |
2150 | glScale $self->{w}, $self->{h}; |
2088 | |
2151 | |
2089 | if ($self->{vertical}) { |
2152 | if ($self->{vertical}) { |
2090 | # draw a vertical slider like a rotated horizontal slider |
2153 | # draw a vertical slider like a rotated horizontal slider |
2091 | |
2154 | |
|
|
2155 | glTranslate 1, 0, 0; |
2092 | glRotate 90, 0, 0, 1; |
2156 | glRotate 90, 0, 0, 1; |
2093 | glTranslate 0, -$self->{w}, 0; |
|
|
2094 | |
|
|
2095 | ($w, $h) = ($h, $w); |
|
|
2096 | } |
2157 | } |
2097 | |
2158 | |
2098 | my $fg = $FOCUS == $self ? $self->{active_fg} : $self->{fg}; |
2159 | my $fg = $FOCUS == $self ? $self->{active_fg} : $self->{fg}; |
2099 | my $bg = $FOCUS == $self ? $self->{active_bg} : $self->{bg}; |
2160 | my $bg = $FOCUS == $self ? $self->{active_bg} : $self->{bg}; |
2100 | |
2161 | |
2101 | my ($value, $lo, $hi, $page) = @{$self->{range}}; |
|
|
2102 | |
|
|
2103 | $hi = $value + 1 if $lo == $hi; |
|
|
2104 | |
|
|
2105 | my $inner_pad_px = $self->_calc_inner_pad_px ($w); |
|
|
2106 | my $inner_w = $w - $inner_pad_px * 2; # * 2 for left & right |
|
|
2107 | |
|
|
2108 | $page = int $page * $inner_w / ($hi - $lo); |
|
|
2109 | $value = int +($value - $lo) * $inner_w / ($hi - $lo); |
|
|
2110 | |
|
|
2111 | $w -= $page; |
|
|
2112 | $page &= ~1; |
|
|
2113 | glTranslate $page * 0.5, 0, 0; |
|
|
2114 | $page ||= 2; |
|
|
2115 | |
|
|
2116 | my $knob_a = $inner_pad_px + ($value - $page * 0.5); |
|
|
2117 | my $knob_b = $inner_pad_px + ($value + $page * 0.5); |
|
|
2118 | |
|
|
2119 | glEnable GL_TEXTURE_2D; |
2162 | glEnable GL_TEXTURE_2D; |
2120 | glTexEnv GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_REPLACE; |
2163 | glTexEnv GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_REPLACE; |
2121 | |
2164 | |
2122 | # draw background |
2165 | # draw background |
2123 | $tex[1]->draw_quad_alpha (0, 0, $w, $h); |
2166 | $tex[1]->draw_quad_alpha (0, 0, 1, 1); |
2124 | |
2167 | |
2125 | # draw handle |
2168 | # draw handle |
2126 | $tex[0]->draw_quad_alpha ($knob_a, 0, $knob_b - $knob_a, $h); |
2169 | $tex[0]->draw_quad_alpha ($self->{knob_x}, 0, $self->{knob_w}, 1); |
2127 | |
2170 | |
2128 | glDisable GL_TEXTURE_2D; |
2171 | glDisable GL_TEXTURE_2D; |
2129 | } |
2172 | } |
2130 | |
2173 | |
2131 | ############################################################################# |
2174 | ############################################################################# |
… | |
… | |
2234 | |
2277 | |
2235 | delete $self->{texture}; |
2278 | delete $self->{texture}; |
2236 | } |
2279 | } |
2237 | |
2280 | |
2238 | $self->{texture} ||= new_from_opengl CFClient::Texture $self->{children}[0]{w}, $self->{children}[0]{h}, sub { |
2281 | $self->{texture} ||= new_from_opengl CFClient::Texture $self->{children}[0]{w}, $self->{children}[0]{h}, sub { |
2239 | glClearColor 0, 0, 0, 0; |
2282 | glClearColor 0.5, 0.5, 0.5, 0; |
2240 | glClear GL_COLOR_BUFFER_BIT; |
2283 | glClear GL_COLOR_BUFFER_BIT; |
2241 | |
2284 | |
2242 | glEnable GL_TEXTURE_2D; |
2285 | glEnable GL_TEXTURE_2D; |
2243 | glTexEnv GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_REPLACE; |
2286 | glTexEnv GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_REPLACE; |
2244 | |
2287 | |
… | |
… | |
2276 | my ($self) = @_; |
2319 | my ($self) = @_; |
2277 | |
2320 | |
2278 | glEnable GL_TEXTURE_2D; |
2321 | glEnable GL_TEXTURE_2D; |
2279 | glTexEnv GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_REPLACE; |
2322 | glTexEnv GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_REPLACE; |
2280 | glColor 1, 1, 1, 1; |
2323 | glColor 1, 1, 1, 1; |
2281 | $self->{texture}->draw_quad_alpha_premultiplied (0, 0, $self->{children}[0]{w}, $self->{children}[0]{h}); |
2324 | $self->{texture}->draw_quad_alpha (0, 0, $self->{children}[0]{w}, $self->{children}[0]{h}); |
2282 | glDisable GL_TEXTURE_2D; |
2325 | glDisable GL_TEXTURE_2D; |
2283 | |
2326 | |
2284 | $self->{children}[1]->draw; |
2327 | $self->{children}[1]->draw; |
2285 | |
2328 | |
2286 | } |
2329 | } |
… | |
… | |
2385 | |
2428 | |
2386 | sub set_tooltip_from { |
2429 | sub set_tooltip_from { |
2387 | my ($self, $widget) = @_; |
2430 | my ($self, $widget) = @_; |
2388 | |
2431 | |
2389 | $self->add (new CFClient::UI::Label |
2432 | $self->add (new CFClient::UI::Label |
2390 | markup => $widget->{tooltip}, |
2433 | markup => $widget->{tooltip}, |
2391 | max_w => ($widget->{tooltip_width} || 0.25) * $::WIDTH, |
2434 | max_w => ($widget->{tooltip_width} || 0.25) * $::WIDTH, |
2392 | fontsize => 0.8, |
2435 | fontsize => 0.8, |
2393 | fg => [0, 0, 0, 1], |
2436 | fg => [0, 0, 0, 1], |
|
|
2437 | ellipsise => 0, |
2394 | font => ($widget->{tooltip_font} || $::FONT_PROP), |
2438 | font => ($widget->{tooltip_font} || $::FONT_PROP), |
2395 | ); |
2439 | ); |
2396 | } |
2440 | } |
2397 | |
2441 | |
2398 | sub size_request { |
2442 | sub size_request { |
2399 | my ($self) = @_; |
2443 | my ($self) = @_; |
… | |
… | |
2477 | |
2521 | |
2478 | package CFClient::UI::InventoryItem; |
2522 | package CFClient::UI::InventoryItem; |
2479 | |
2523 | |
2480 | our @ISA = CFClient::UI::HBox::; |
2524 | our @ISA = CFClient::UI::HBox::; |
2481 | |
2525 | |
|
|
2526 | sub _item_to_desc { |
|
|
2527 | my ($item) = @_; |
|
|
2528 | |
|
|
2529 | my $desc = |
|
|
2530 | $item->{nrof} < 2 |
|
|
2531 | ? $item->{name} |
|
|
2532 | : "$item->{nrof} $item->{name_pl}"; |
|
|
2533 | |
|
|
2534 | $item->{flags} & Crossfire::Protocol::F_OPEN |
|
|
2535 | and $desc .= " (open)"; |
|
|
2536 | $item->{flags} & Crossfire::Protocol::F_APPLIED |
|
|
2537 | and $desc .= " (applied)"; |
|
|
2538 | $item->{flags} & Crossfire::Protocol::F_UNPAID |
|
|
2539 | and $desc .= " (unpaid)"; |
|
|
2540 | $item->{flags} & Crossfire::Protocol::F_MAGIC |
|
|
2541 | and $desc .= " (magic)"; |
|
|
2542 | $item->{flags} & Crossfire::Protocol::F_CURSED |
|
|
2543 | and $desc .= " (cursed)"; |
|
|
2544 | $item->{flags} & Crossfire::Protocol::F_DAMNED |
|
|
2545 | and $desc .= " (damned)"; |
|
|
2546 | $item->{flags} & Crossfire::Protocol::F_LOCKED |
|
|
2547 | and $desc .= " *"; |
|
|
2548 | |
|
|
2549 | $desc |
|
|
2550 | } |
|
|
2551 | |
2482 | sub new { |
2552 | sub new { |
2483 | my $class = shift; |
2553 | my $class = shift; |
2484 | |
2554 | |
2485 | my %args = @_; |
2555 | my %args = @_; |
2486 | |
2556 | |
2487 | my $item = delete $args{item}; |
2557 | my $item = delete $args{item}; |
2488 | |
2558 | |
2489 | my $desc = $item->{nrof} < 2 |
2559 | my $desc = _item_to_desc ($item); |
2490 | ? $item->{name} |
|
|
2491 | : "$item->{nrof} $item->{name_pl}"; |
|
|
2492 | |
|
|
2493 | |
2560 | |
2494 | my $self = $class->SUPER::new ( |
2561 | my $self = $class->SUPER::new ( |
2495 | can_hover => 1, |
2562 | can_hover => 1, |
2496 | can_events => 1, |
2563 | can_events => 1, |
2497 | tooltip => (CFClient::UI::Label->escape ($desc) |
2564 | tooltip => ((CFClient::UI::Label::escape $desc) |
2498 | . "\n<small>leftclick - pick up\nmiddle click - apply\nrightclick - menu</small>"), |
2565 | . "\n<small>leftclick - pick up\nmiddle click - apply\nrightclick - menu</small>"), |
2499 | connect_button_down => sub { |
2566 | connect_button_down => sub { |
2500 | my ($self, $ev, $x, $y) = @_; |
2567 | my ($self, $ev, $x, $y) = @_; |
2501 | |
2568 | |
2502 | # todo: maybe put examine on 1? but should just be a tooltip :( |
2569 | # todo: maybe put examine on 1? but should just be a tooltip :( |
2503 | if ($ev->{button} == 1) { |
2570 | if ($ev->{button} == 1) { |
|
|
2571 | my $targ = $::CONN->{player}{tag}; |
|
|
2572 | |
|
|
2573 | if ($item->{container} == $::CONN->{player}{tag}) { |
|
|
2574 | $targ = $main::OPENCONT; |
|
|
2575 | } |
|
|
2576 | |
2504 | $::CONN->send ("move $::CONN->{player}{tag} $item->{tag} 0"); |
2577 | $::CONN->send ("move $targ $item->{tag} 0"); |
2505 | } elsif ($ev->{button} == 2) { |
2578 | } elsif ($ev->{button} == 2) { |
2506 | $::CONN->send ("apply $item->{tag}"); |
2579 | $::CONN->send ("apply $item->{tag}"); |
2507 | } elsif ($ev->{button} == 3) { |
2580 | } elsif ($ev->{button} == 3) { |
2508 | CFClient::UI::Menu->new ( |
2581 | my @menu_items = ( |
2509 | items => [ |
|
|
2510 | ["examine", sub { $::CONN->send ("examine $item->{tag}") }], |
2582 | ["examine", sub { $::CONN->send ("examine $item->{tag}") }], |
|
|
2583 | ["mark", sub { $::CONN->send ("mark $item->{tag}") }], |
|
|
2584 | ["apply", sub { $::CONN->send ("apply $item->{tag}") }], |
|
|
2585 | ["drop", sub { $::CONN->send ("move $main::OPENCONT $item->{tag} 0") }], |
2511 | [ |
2586 | [ |
2512 | $item->{flags} & Crossfire::Protocol::F_LOCKED ? "lock" : "unlock", |
2587 | $item->{flags} & Crossfire::Protocol::F_LOCKED ? "lock" : "unlock", |
2513 | sub { $::CONN->send ("lock $item->{tag}") }, |
2588 | sub { $::CONN->send ("lock $item->{tag}") }, |
2514 | ], |
|
|
2515 | ["mark", sub { $::CONN->send ("mark $item->{tag}") }], |
|
|
2516 | ["apply", sub { $::CONN->send ("apply $item->{tag}") }], |
|
|
2517 | ["drop", sub { $::CONN->send ("move 0 $item->{tag} 0") }], |
|
|
2518 | ], |
2589 | ], |
2519 | )->popup ($ev); |
2590 | ); |
|
|
2591 | |
|
|
2592 | CFClient::UI::Menu->new (items => \@menu_items)->popup ($ev); |
2520 | } |
2593 | } |
2521 | |
2594 | |
2522 | 1 |
2595 | 1 |
2523 | }, |
2596 | }, |
2524 | %args |
2597 | %args |
2525 | ); |
2598 | ); |
|
|
2599 | |
2526 | |
2600 | |
2527 | $self->add (new CFClient::UI::Face |
2601 | $self->add (new CFClient::UI::Face |
2528 | can_events => 0, |
2602 | can_events => 0, |
2529 | face => $item->{face}, |
2603 | face => $item->{face}, |
2530 | anim => $item->{anim}, |
2604 | anim => $item->{anim}, |
2531 | animspeed => $item->{animspeed}, |
2605 | animspeed => $item->{animspeed}, |
2532 | ); |
2606 | ); |
2533 | |
2607 | |
2534 | $self->add (new CFClient::UI::Label |
2608 | $self->add ($self->{name_lbl} = new CFClient::UI::Label can_events => 0); |
2535 | can_events => 0, |
2609 | |
2536 | text => $desc, |
2610 | $self->{item} = $item; |
2537 | ); |
2611 | |
|
|
2612 | $self->update_item; |
2538 | |
2613 | |
2539 | $self |
2614 | $self |
|
|
2615 | } |
|
|
2616 | |
|
|
2617 | sub update_item { |
|
|
2618 | my ($self) = @_; |
|
|
2619 | |
|
|
2620 | my $desc = _item_to_desc ($self->{item}); |
|
|
2621 | |
|
|
2622 | $self->{name_lbl}->set_text ($desc); |
2540 | } |
2623 | } |
2541 | |
2624 | |
2542 | ############################################################################# |
2625 | ############################################################################# |
2543 | |
2626 | |
2544 | package CFClient::UI::Inventory; |
2627 | package CFClient::UI::Inventory; |
… | |
… | |
2568 | } @$items; |
2651 | } @$items; |
2569 | |
2652 | |
2570 | $self->{real_items} = \@items; |
2653 | $self->{real_items} = \@items; |
2571 | |
2654 | |
2572 | for my $item (@items) { |
2655 | for my $item (@items) { |
2573 | my $desc = $item->{nrof} < 2 |
|
|
2574 | ? $item->{name} |
|
|
2575 | : "$item->{nrof} $item->{name_pl}"; |
|
|
2576 | |
|
|
2577 | $item = $item->{widget} ||= new CFClient::UI::InventoryItem item => $item; |
2656 | $item = $item->{widget} ||= new CFClient::UI::InventoryItem item => $item; |
|
|
2657 | $item->update_item (); |
2578 | } |
2658 | } |
2579 | |
2659 | |
2580 | $self->{scrolled}->add (@items); |
2660 | $self->{scrolled}->add (@items); |
2581 | |
2661 | |
2582 | # $range->{range} = [$self->{pos}, 0, $self->{max_pos}, $page]; |
2662 | # $range->{range} = [$self->{pos}, 0, $self->{max_pos}, $page]; |
… | |
… | |
2663 | |
2743 | |
2664 | package CFClient::UI::Statusbox; |
2744 | package CFClient::UI::Statusbox; |
2665 | |
2745 | |
2666 | our @ISA = CFClient::UI::VBox::; |
2746 | our @ISA = CFClient::UI::VBox::; |
2667 | |
2747 | |
|
|
2748 | sub new { |
|
|
2749 | my $class = shift; |
|
|
2750 | |
|
|
2751 | $class->SUPER::new ( |
|
|
2752 | fontsize => 0.8, |
|
|
2753 | @_, |
|
|
2754 | ) |
|
|
2755 | } |
|
|
2756 | |
2668 | sub reorder { |
2757 | sub reorder { |
2669 | my ($self) = @_; |
2758 | my ($self) = @_; |
2670 | my $NOW = time; |
2759 | my $NOW = time; |
2671 | |
2760 | |
2672 | while (my ($k, $v) = each %{ $self->{item} }) { |
2761 | while (my ($k, $v) = each %{ $self->{item} }) { |
… | |
… | |
2690 | ? "<b>$item->{count} ×</b> $item->{text}" |
2779 | ? "<b>$item->{count} ×</b> $item->{text}" |
2691 | : $item->{text}; |
2780 | : $item->{text}; |
2692 | |
2781 | |
2693 | for ($short) { |
2782 | for ($short) { |
2694 | s/^\s+//; |
2783 | s/^\s+//; |
2695 | s/\012.*/…/s; |
2784 | s/\s+/ /g; |
2696 | my $len = int 40 / $item->{fontsize}; |
|
|
2697 | substr $_, $len, length, "…" if $len < length; |
|
|
2698 | } |
2785 | } |
2699 | |
2786 | |
2700 | new CFClient::UI::Label |
2787 | new CFClient::UI::Label |
2701 | markup => $short, |
2788 | markup => $short, |
2702 | tooltip => $item->{tooltip}, |
2789 | tooltip => $item->{tooltip}, |
2703 | tooltip_font => $::FONT_PROP, |
2790 | tooltip_font => $::FONT_PROP, |
2704 | tooltip_width => 0.67, |
2791 | tooltip_width => 0.67, |
2705 | fontsize => $item->{fontsize}, |
2792 | fontsize => $item->{fontsize} || $self->{fontsize}, |
|
|
2793 | max_w => $::WIDTH * 0.44, |
2706 | color => $item->{color}, |
2794 | fg => $item->{fg}, |
2707 | can_events => 1, |
2795 | can_events => 1, |
2708 | can_hover => 1 |
2796 | can_hover => 1 |
2709 | }; |
2797 | }; |
2710 | } |
2798 | } |
2711 | |
2799 | |
… | |
… | |
2737 | $self->{item}{$group} = { |
2825 | $self->{item}{$group} = { |
2738 | id => ++$self->{id}, |
2826 | id => ++$self->{id}, |
2739 | text => $text, |
2827 | text => $text, |
2740 | timeout => $timeout, |
2828 | timeout => $timeout, |
2741 | tooltip => $text, |
2829 | tooltip => $text, |
2742 | fontsize => 0.8, |
|
|
2743 | color => [0.8, 0.8, 0.8, 0.8], |
2830 | fg => [0.8, 0.8, 0.8, 0.8], |
2744 | pri => 0, |
2831 | pri => 0, |
2745 | count => 1, |
2832 | count => 1, |
2746 | %arg, |
2833 | %arg, |
2747 | }; |
2834 | }; |
2748 | } |
2835 | } |
2749 | |
2836 | |
2750 | $self->reorder; |
2837 | $self->reorder; |
2751 | } |
2838 | } |
2752 | |
2839 | |
|
|
2840 | sub reconfigure { |
|
|
2841 | my ($self) = @_; |
|
|
2842 | |
|
|
2843 | delete $_->{label} |
|
|
2844 | for values %{ $self->{item} || {} }; |
|
|
2845 | |
|
|
2846 | $self->reorder; |
|
|
2847 | $self->SUPER::reconfigure; |
|
|
2848 | } |
|
|
2849 | |
2753 | ############################################################################# |
2850 | ############################################################################# |
2754 | |
2851 | |
2755 | package CFClient::UI::Root; |
2852 | package CFClient::UI::Root; |
2756 | |
2853 | |
2757 | our @ISA = CFClient::UI::Container::; |
2854 | our @ISA = CFClient::UI::Container::; |
… | |
… | |
2796 | if $old_w && $old_h && ($old_w != $w || $old_h != $h); |
2893 | if $old_w && $old_h && ($old_w != $w || $old_h != $h); |
2797 | |
2894 | |
2798 | for my $child ($self->children) { |
2895 | for my $child ($self->children) { |
2799 | my ($X, $Y, $W, $H) = @$child{qw(x y req_w req_h)}; |
2896 | my ($X, $Y, $W, $H) = @$child{qw(x y req_w req_h)}; |
2800 | |
2897 | |
|
|
2898 | $X = $child->{req_x} > 0 ? $child->{req_x} : $w - $W - $child->{req_x} + 1 |
|
|
2899 | if exists $child->{req_x}; |
|
|
2900 | |
|
|
2901 | $Y = $child->{req_y} > 0 ? $child->{req_y} : $h - $H - $child->{req_y} + 1 |
|
|
2902 | if exists $child->{req_y}; |
|
|
2903 | |
2801 | $X = List::Util::max 0, List::Util::min $w - $W, $X; |
2904 | $X = List::Util::max 0, List::Util::min $w - $W, int $X + 0.5; |
2802 | $Y = List::Util::max 0, List::Util::min $h - $H, $Y; |
2905 | $Y = List::Util::max 0, List::Util::min $h - $H, int $Y + 0.5; |
|
|
2906 | |
2803 | $child->configure ($X, $Y, $W, $H); |
2907 | $child->configure ($X, $Y, $W, $H); |
2804 | } |
2908 | } |
2805 | } |
2909 | } |
2806 | |
2910 | |
2807 | sub coord2local { |
2911 | sub coord2local { |
… | |
… | |
2923 | ############################################################################# |
3027 | ############################################################################# |
2924 | |
3028 | |
2925 | package CFClient::UI; |
3029 | package CFClient::UI; |
2926 | |
3030 | |
2927 | $ROOT = new CFClient::UI::Root; |
3031 | $ROOT = new CFClient::UI::Root; |
2928 | $TOOLTIP = new CFClient::UI::Tooltip; |
3032 | $TOOLTIP = new CFClient::UI::Tooltip z => 900; |
2929 | |
3033 | |
2930 | 1 |
3034 | 1 |
2931 | |
3035 | |