… | |
… | |
209 | sub show { |
209 | sub show { |
210 | my ($self) = @_; |
210 | my ($self) = @_; |
211 | return if $self->{parent}; |
211 | return if $self->{parent}; |
212 | |
212 | |
213 | $CFClient::UI::ROOT->add ($self); |
213 | $CFClient::UI::ROOT->add ($self); |
|
|
214 | } |
|
|
215 | |
|
|
216 | sub show_centered { |
|
|
217 | my ($self) = @_; |
|
|
218 | return if $self->{parent}; |
|
|
219 | |
|
|
220 | $self->show; |
|
|
221 | |
|
|
222 | $CFClient::UI::ROOT->on_post_alloc ( |
|
|
223 | "centered $self" => sub { |
|
|
224 | $self->move (($::WIDTH - $self->{w}) * 0.5, ($::HEIGHT - $self->{h}) * 0.5); |
|
|
225 | }, |
|
|
226 | ); |
214 | } |
227 | } |
215 | |
228 | |
216 | sub hide { |
229 | sub hide { |
217 | my ($self) = @_; |
230 | my ($self) = @_; |
218 | |
231 | |
… | |
… | |
730 | |
743 | |
731 | sub size_request { |
744 | sub size_request { |
732 | my ($self) = @_; |
745 | my ($self) = @_; |
733 | |
746 | |
734 | @$self{qw(child_w child_h)} = @{$self->child}{qw(req_w req_h)}; |
747 | @$self{qw(child_w child_h)} = @{$self->child}{qw(req_w req_h)}; |
735 | $self->child->configure (0, 0, @$self{qw(child_w child_h)}); |
|
|
736 | |
748 | |
737 | @$self{qw(child_w child_h)} |
749 | @$self{qw(child_w child_h)} |
738 | } |
750 | } |
739 | |
751 | |
740 | sub size_allocate { |
752 | sub size_allocate { |
741 | my ($self, $w, $h) = @_; |
753 | my ($self, $w, $h) = @_; |
742 | |
754 | |
|
|
755 | my ($cw, $ch) = @$self{qw(child_w child_h)}; |
|
|
756 | # $w = $self->{w}; |
|
|
757 | $self->child->configure (0, 0, $cw, $ch); |
743 | $self->update; |
758 | $self->update; |
744 | } |
759 | } |
745 | |
760 | |
746 | sub set_offset { |
761 | sub set_offset { |
747 | my ($self, $x, $y) = @_; |
762 | my ($self, $x, $y) = @_; |
… | |
… | |
1475 | active_fg => [0, 0, 0], |
1490 | active_fg => [0, 0, 0], |
1476 | can_hover => 1, |
1491 | can_hover => 1, |
1477 | can_focus => 1, |
1492 | can_focus => 1, |
1478 | valign => 0, |
1493 | valign => 0, |
1479 | can_events => 1, |
1494 | can_events => 1, |
|
|
1495 | #text => ... |
1480 | @_ |
1496 | @_ |
1481 | ) |
1497 | ) |
1482 | } |
1498 | } |
1483 | |
1499 | |
1484 | sub _set_text { |
1500 | sub _set_text { |
… | |
… | |
2054 | my $self = $class->SUPER::new ( |
2070 | my $self = $class->SUPER::new ( |
2055 | fg => [1, 1, 1], |
2071 | fg => [1, 1, 1], |
2056 | active_fg => [0, 0, 0], |
2072 | active_fg => [0, 0, 0], |
2057 | bg => [0, 0, 0, 0.2], |
2073 | bg => [0, 0, 0, 0.2], |
2058 | active_bg => [1, 1, 1, 0.5], |
2074 | active_bg => [1, 1, 1, 0.5], |
2059 | range => [0, 0, 100, 10], |
2075 | range => [0, 0, 100, 10, 0], |
2060 | req_w => $::WIDTH / 80, |
2076 | req_w => $::WIDTH / 80, |
2061 | req_h => $::WIDTH / 80, |
2077 | req_h => $::WIDTH / 80, |
2062 | vertical => 0, |
2078 | vertical => 0, |
2063 | can_hover => 1, |
2079 | can_hover => 1, |
2064 | inner_pad => 0.02, |
2080 | inner_pad => 0.02, |
… | |
… | |
2069 | $self->update; |
2085 | $self->update; |
2070 | |
2086 | |
2071 | $self |
2087 | $self |
2072 | } |
2088 | } |
2073 | |
2089 | |
|
|
2090 | sub set_range { |
|
|
2091 | my ($self, $range) = @_; |
|
|
2092 | |
|
|
2093 | $self->{range} = $range; |
|
|
2094 | |
|
|
2095 | $self->update; |
|
|
2096 | } |
|
|
2097 | |
2074 | sub set_value { |
2098 | sub set_value { |
2075 | my ($self, $value) = @_; |
2099 | my ($self, $value) = @_; |
2076 | |
2100 | |
2077 | my ($old_value, $lo, $hi, $page, $unit) = @{$self->{range}}; |
2101 | my ($old_value, $lo, $hi, $page, $unit) = @{$self->{range}}; |
2078 | |
2102 | |
2079 | $hi = $lo + 1 if $hi <= $lo; |
2103 | $hi = $lo + 1 if $hi <= $lo; |
2080 | |
2104 | |
|
|
2105 | $page = $hi - $lo if $page > $hi - $lo; |
|
|
2106 | |
2081 | $value = $lo if $value < $lo; |
2107 | $value = $lo if $value < $lo; |
2082 | $value = $hi if $value > $hi; |
2108 | $value = $hi - $page if $value > $hi - $page; |
2083 | |
2109 | |
2084 | $value = $lo + $unit * int +($value - $lo + $unit * 0.5) / $unit |
2110 | $value = $lo + $unit * int +($value - $lo + $unit * 0.5) / $unit |
2085 | if $unit; |
2111 | if $unit; |
2086 | |
|
|
2087 | $page = $hi - $lo if $page > $hi - $lo; |
|
|
2088 | |
2112 | |
2089 | @{$self->{range}} = ($value, $lo, $hi, $page, $unit); |
2113 | @{$self->{range}} = ($value, $lo, $hi, $page, $unit); |
2090 | |
2114 | |
2091 | if ($value != $old_value) { |
2115 | if ($value != $old_value) { |
2092 | $self->emit (changed => $value); |
2116 | $self->emit (changed => $value); |
… | |
… | |
2105 | |
2129 | |
2106 | sub button_down { |
2130 | sub button_down { |
2107 | my ($self, $ev, $x, $y) = @_; |
2131 | my ($self, $ev, $x, $y) = @_; |
2108 | |
2132 | |
2109 | $self->SUPER::button_down ($ev, $x, $y); |
2133 | $self->SUPER::button_down ($ev, $x, $y); |
|
|
2134 | |
|
|
2135 | $self->{click} = [$self->{range}[0], $self->{vertical} ? $y : $x]; |
|
|
2136 | |
2110 | $self->mouse_motion ($ev, $x, $y); |
2137 | $self->mouse_motion ($ev, $x, $y); |
2111 | } |
2138 | } |
2112 | |
2139 | |
2113 | sub mouse_motion { |
2140 | sub mouse_motion { |
2114 | my ($self, $ev, $x, $y) = @_; |
2141 | my ($self, $ev, $x, $y) = @_; |
… | |
… | |
2116 | if ($GRAB == $self) { |
2143 | if ($GRAB == $self) { |
2117 | my ($x, $w) = $self->{vertical} ? ($y, $self->{h}) : ($x, $self->{w}); |
2144 | my ($x, $w) = $self->{vertical} ? ($y, $self->{h}) : ($x, $self->{w}); |
2118 | |
2145 | |
2119 | my (undef, $lo, $hi, $page) = @{$self->{range}}; |
2146 | my (undef, $lo, $hi, $page) = @{$self->{range}}; |
2120 | |
2147 | |
2121 | $x = $x / ($w * (1 - 2 * $self->{inner_pad})) - $self->{inner_pad}; |
2148 | $x = ($x - $self->{click}[1]) / ($w * $self->{scale}); |
2122 | |
2149 | |
2123 | $self->set_value ($x * ($hi - $lo) + $lo); |
2150 | $self->set_value ($self->{click}[0] + $x * ($hi - $page - $lo)); |
2124 | } |
2151 | } |
2125 | } |
2152 | } |
2126 | |
2153 | |
2127 | sub update { |
2154 | sub update { |
2128 | my ($self) = @_; |
2155 | my ($self) = @_; |
2129 | |
2156 | |
2130 | $CFClient::UI::ROOT->on_post_alloc ($self => sub { |
2157 | $CFClient::UI::ROOT->on_post_alloc ($self => sub { |
2131 | $self->set_value ($self->{range}[0]); |
2158 | $self->set_value ($self->{range}[0]); |
2132 | |
2159 | |
2133 | my ($value, $lo, $hi, $page) = @{$self->{range}}; |
2160 | my ($value, $lo, $hi, $page) = @{$self->{range}}; |
|
|
2161 | my $range = ($hi - $page - $lo) || 1e-100; |
2134 | |
2162 | |
2135 | my $inner_w = 1 - 2 * $self->{inner_pad}; |
2163 | my $knob_w = List::Util::min 1, $page / ($hi - $lo) || 0.1; |
2136 | |
2164 | |
2137 | $self->{scale} = ($inner_w / ($hi - $lo)) || 1; |
2165 | $self->{offset} = List::Util::max $self->{inner_pad}, $knob_w * 0.5; |
|
|
2166 | $self->{scale} = 1 - 2 * $self->{offset} || 1e-100; |
2138 | |
2167 | |
2139 | $page = $self->{scale} * $page || 10 / ($self->{w} || 1); |
2168 | $value = ($value - $lo) / $range; |
2140 | $value = $self->{scale} * ($value - $lo); |
2169 | $value = $value * $self->{scale} + $self->{offset}; |
2141 | |
2170 | |
2142 | $value = $self->{inner_pad} + ($value - $page * 0.5); |
|
|
2143 | |
|
|
2144 | $value = 0 if $value < 0; |
|
|
2145 | $page = 1 - $value if $value + $page > 1; |
|
|
2146 | |
|
|
2147 | $self->{knob_x} = $value; |
2171 | $self->{knob_x} = $value - $knob_w * 0.5; |
2148 | $self->{knob_w} = $page; |
2172 | $self->{knob_w} = $knob_w; |
2149 | }); |
2173 | }); |
2150 | |
2174 | |
2151 | $self->SUPER::update; |
2175 | $self->SUPER::update; |
2152 | } |
2176 | } |
2153 | |
2177 | |
… | |
… | |
2177 | # draw handle |
2201 | # draw handle |
2178 | $tex[0]->draw_quad_alpha ($self->{knob_x}, 0, $self->{knob_w}, 1); |
2202 | $tex[0]->draw_quad_alpha ($self->{knob_x}, 0, $self->{knob_w}, 1); |
2179 | |
2203 | |
2180 | glDisable GL_TEXTURE_2D; |
2204 | glDisable GL_TEXTURE_2D; |
2181 | } |
2205 | } |
|
|
2206 | |
|
|
2207 | ############################################################################# |
|
|
2208 | |
|
|
2209 | package CFClient::UI::ValSlider; |
|
|
2210 | |
|
|
2211 | our @ISA = CFClient::UI::HBox::; |
|
|
2212 | |
|
|
2213 | sub new { |
|
|
2214 | my ($class, %arg) = @_; |
|
|
2215 | |
|
|
2216 | my $range = delete $arg{range}; |
|
|
2217 | |
|
|
2218 | my $self = $class->SUPER::new ( |
|
|
2219 | slider => (new CFClient::UI::Slider expand => 1, range => $range), |
|
|
2220 | entry => (new CFClient::UI::Label text => "", template => delete $arg{template}), |
|
|
2221 | to_value => sub { shift }, |
|
|
2222 | from_value => sub { shift }, |
|
|
2223 | %arg, |
|
|
2224 | ); |
|
|
2225 | |
|
|
2226 | $self->{slider}->connect (changed => sub { |
|
|
2227 | my ($self, $value) = @_; |
|
|
2228 | $self->{parent}{entry}->set_text ($self->{parent}{to_value}->($value)); |
|
|
2229 | $self->{parent}->emit (changed => $value); |
|
|
2230 | }); |
|
|
2231 | |
|
|
2232 | # $self->{entry}->connect (changed => sub { |
|
|
2233 | # my ($self, $value) = @_; |
|
|
2234 | # $self->{parent}{slider}->set_value ($self->{parent}{from_value}->($value)); |
|
|
2235 | # $self->{parent}->emit (changed => $value); |
|
|
2236 | # }); |
|
|
2237 | |
|
|
2238 | $self->add ($self->{slider}, $self->{entry}); |
|
|
2239 | |
|
|
2240 | $self->{slider}->emit (changed => $self->{slider}{range}[0]); |
|
|
2241 | |
|
|
2242 | $self |
|
|
2243 | } |
|
|
2244 | |
|
|
2245 | sub set_range { shift->{slider}->set_range (@_) } |
|
|
2246 | sub set_value { shift->{slider}->set_value (@_) } |
2182 | |
2247 | |
2183 | ############################################################################# |
2248 | ############################################################################# |
2184 | |
2249 | |
2185 | package CFClient::UI::TextView; |
2250 | package CFClient::UI::TextView; |
2186 | |
2251 | |
… | |
… | |
2247 | |
2312 | |
2248 | $self->{need_reflow}++; |
2313 | $self->{need_reflow}++; |
2249 | $self->update; |
2314 | $self->update; |
2250 | } |
2315 | } |
2251 | |
2316 | |
|
|
2317 | sub set_offset { |
|
|
2318 | my ($self, $offset) = @_; |
|
|
2319 | |
|
|
2320 | # todo: base offset on lines or so, not on pixels |
|
|
2321 | $self->{children}[1]->set_value ($offset); |
|
|
2322 | } |
|
|
2323 | |
|
|
2324 | sub clear { |
|
|
2325 | my ($self) = @_; |
|
|
2326 | |
|
|
2327 | $self->{par} = []; |
|
|
2328 | $self->{height} = 0; |
|
|
2329 | $self->{children}[1]->set_range ([0, 0, 0, 1, 1]); |
|
|
2330 | } |
|
|
2331 | |
2252 | sub add_paragraph { |
2332 | sub add_paragraph { |
2253 | my ($self, $color, $text, $indent) = @_; |
2333 | my ($self, $color, $text, $indent) = @_; |
2254 | |
|
|
2255 | #TODO: intelligently "reformat" paragraph |
|
|
2256 | |
2334 | |
2257 | for my $line (split /\n/, $text) { |
2335 | for my $line (split /\n/, $text) { |
2258 | my $height = $self->text_height ($line); |
2336 | my $height = $self->text_height ($line); |
2259 | $self->{height} += $height; |
2337 | $self->{height} += $height; |
2260 | push @{$self->{par}}, [$height, $color, $indent, $line]; |
2338 | push @{$self->{par}}, [$height, $color, $indent, $line]; |
2261 | } |
2339 | } |
2262 | |
2340 | |
2263 | $self->{children}[1]{range} = [$self->{height} - $self->{h}, 0, $self->{height}, $self->{h}]; |
2341 | $self->{children}[1]->set_range ([$self->{height}, 0, $self->{height}, $self->{h}, 1]); |
2264 | $self->{children}[1]->update; |
|
|
2265 | } |
2342 | } |
2266 | |
2343 | |
2267 | sub update { |
2344 | sub update { |
2268 | my ($self) = @_; |
2345 | my ($self) = @_; |
2269 | |
2346 | |
… | |
… | |
2280 | $height += $_->[0] = $self->text_height ($_->[3], $_->[2]) |
2357 | $height += $_->[0] = $self->text_height ($_->[3], $_->[2]) |
2281 | for @{$self->{par}}; |
2358 | for @{$self->{par}}; |
2282 | |
2359 | |
2283 | $self->{height} = $height; |
2360 | $self->{height} = $height; |
2284 | |
2361 | |
2285 | $self->{children}[1]{range} = [$height - $self->{h}, 0, $height, $self->{h}]; |
2362 | $self->{children}[1]->set_range ([$height, 0, $height, $self->{h}, 1]); |
2286 | $self->{children}[1]->update; |
|
|
2287 | |
2363 | |
2288 | delete $self->{texture}; |
2364 | delete $self->{texture}; |
2289 | } |
2365 | } |
2290 | |
2366 | |
2291 | $self->{texture} ||= new_from_opengl CFClient::Texture $self->{children}[0]{w}, $self->{children}[0]{h}, sub { |
2367 | $self->{texture} ||= new_from_opengl CFClient::Texture $self->{children}[0]{w}, $self->{children}[0]{h}, sub { |
… | |
… | |
2539 | } |
2615 | } |
2540 | |
2616 | |
2541 | sub _draw { |
2617 | sub _draw { |
2542 | my ($self) = @_; |
2618 | my ($self) = @_; |
2543 | |
2619 | |
2544 | return unless $::CONN;#d# manage and cache textures differently |
2620 | return unless $::CONN; |
2545 | |
2621 | |
2546 | my $face; |
2622 | my $face; |
2547 | |
2623 | |
2548 | if ($self->{frame}) { |
2624 | if ($self->{frame}) { |
2549 | my $anim = $::CONN->{anim}[$self->{anim}]; |
2625 | my $anim = $::CONN->{anim}[$self->{anim}]; |
… | |
… | |
2615 | |
2691 | |
2616 | my $self = $class->SUPER::new ( |
2692 | my $self = $class->SUPER::new ( |
2617 | can_hover => 1, |
2693 | can_hover => 1, |
2618 | can_events => 1, |
2694 | can_events => 1, |
2619 | tooltip => ((CFClient::UI::Label::escape $desc) |
2695 | tooltip => ((CFClient::UI::Label::escape $desc) |
2620 | . "\n<small>leftclick - pick up\nmiddle click - apply\nrightclick - menu</small>"), |
2696 | . "\n<small>leftclick - examine\nshift+leftclick - move/pickup/drop\nmiddle click - apply\nrightclick - menu</small>"), |
2621 | connect_button_down => sub { |
2697 | connect_button_down => sub { |
2622 | my ($self, $ev, $x, $y) = @_; |
2698 | my ($self, $ev, $x, $y) = @_; |
2623 | |
2699 | |
2624 | # todo: maybe put examine on 1? but should just be a tooltip :( |
2700 | # todo: maybe put examine on 1? but should just be a tooltip :( |
2625 | if ($ev->{button} == 1) { |
2701 | if (($ev->{mod} & CFClient::KMOD_SHIFT) && $ev->{button} == 1) { |
2626 | my $targ = $::CONN->{player}{tag}; |
2702 | my $targ = $::CONN->{player}{tag}; |
2627 | |
2703 | |
2628 | if ($item->{container} == $::CONN->{player}{tag}) { |
2704 | if ($item->{container} == $::CONN->{player}{tag}) { |
2629 | $targ = $main::OPENCONT; |
2705 | $targ = $main::OPENCONT; |
2630 | } |
2706 | } |
2631 | |
2707 | |
2632 | $::CONN->send ("move $targ $item->{tag} 0"); |
2708 | $::CONN->send ("move $targ $item->{tag} 0"); |
|
|
2709 | } elsif ($ev->{button} == 1) { |
|
|
2710 | $::CONN->send ("examine $item->{tag}"); |
2633 | } elsif ($ev->{button} == 2) { |
2711 | } elsif ($ev->{button} == 2) { |
2634 | $::CONN->send ("apply $item->{tag}"); |
2712 | $::CONN->send ("apply $item->{tag}"); |
2635 | } elsif ($ev->{button} == 3) { |
2713 | } elsif ($ev->{button} == 3) { |
2636 | my @menu_items = ( |
2714 | my @menu_items = ( |
2637 | ["examine", sub { $::CONN->send ("examine $item->{tag}") }], |
2715 | ["examine", sub { $::CONN->send ("examine $item->{tag}") }], |
… | |
… | |
2690 | |
2768 | |
2691 | sub new { |
2769 | sub new { |
2692 | my $class = shift; |
2770 | my $class = shift; |
2693 | |
2771 | |
2694 | my $self = $class->SUPER::new ( |
2772 | my $self = $class->SUPER::new ( |
2695 | scrolled => (new CFClient::UI::VBox), |
2773 | scrolled => (new CFClient::UI::Table), |
2696 | @_, |
2774 | @_, |
2697 | ); |
2775 | ); |
2698 | |
2776 | |
2699 | $self |
2777 | $self |
2700 | } |
2778 | } |
… | |
… | |
2711 | } @$items; |
2789 | } @$items; |
2712 | |
2790 | |
2713 | $self->{real_items} = \@items; |
2791 | $self->{real_items} = \@items; |
2714 | |
2792 | |
2715 | for my $item (@items) { |
2793 | for my $item (@items) { |
|
|
2794 | $item->{item} = $item; |
2716 | $item = $item->{widget} ||= new CFClient::UI::InventoryItem item => $item; |
2795 | $item = $item->{widget} ||= new CFClient::UI::InventoryItem item => $item; |
2717 | $item->update_item (); |
2796 | $item->update_item (); |
2718 | } |
2797 | } |
2719 | |
2798 | |
|
|
2799 | my $i = 0; |
|
|
2800 | for (@items) { |
2720 | $self->{scrolled}->add (@items); |
2801 | $self->{scrolled}->add (0, $i, $_); |
|
|
2802 | my $nrof = $_->{item}->{nrof} || 1; |
|
|
2803 | $self->{scrolled}->add (1, $i++, new CFClient::UI::Label text => ($_->{item}->{weight} * $nrof) / 1000); |
|
|
2804 | } |
2721 | |
2805 | |
2722 | # $range->{range} = [$self->{pos}, 0, $self->{max_pos}, $page]; |
2806 | # $range->{range} = [$self->{pos}, 0, $self->{max_pos}, $page]; |
2723 | } |
2807 | } |
2724 | |
2808 | |
2725 | sub size_request { |
2809 | sub size_request { |