… | |
… | |
25 | # need to do it again because that pile of garbage called PAR nukes it before main |
25 | # need to do it again because that pile of garbage called PAR nukes it before main |
26 | unshift @INC, $ENV{PAR_TEMP} |
26 | unshift @INC, $ENV{PAR_TEMP} |
27 | if %PAR::LibCache; |
27 | if %PAR::LibCache; |
28 | |
28 | |
29 | use Time::HiRes 'time'; |
29 | use Time::HiRes 'time'; |
30 | use Pod::POM; |
|
|
31 | use Event; |
30 | use Event; |
32 | |
31 | |
33 | use Crossfire; |
32 | use Crossfire; |
34 | use Crossfire::Protocol; |
33 | use Crossfire::Protocol::Constants; |
35 | |
34 | |
36 | use Compress::LZF; |
35 | use Compress::LZF; |
37 | |
36 | |
38 | use CFClient; |
37 | use CFClient; |
|
|
38 | use CFClient::OpenGL (); |
|
|
39 | use CFClient::Protocol; |
39 | use CFClient::UI; |
40 | use CFClient::UI; |
|
|
41 | use CFClient::Pod; |
|
|
42 | use CFClient::BindingEditor; |
40 | use CFClient::MapWidget; |
43 | use CFClient::MapWidget; |
|
|
44 | |
|
|
45 | $SIG{QUIT} = sub { Carp::cluck "QUIT" }; |
|
|
46 | $SIG{PIPE} = 'IGNORE'; |
41 | |
47 | |
42 | $Event::DIED = sub { |
48 | $Event::DIED = sub { |
43 | # TODO: display dialog box or so |
49 | # TODO: display dialog box or so |
|
|
50 | Carp::confess $_[1];#d#TODO: remove when stable |
44 | CFClient::error $_[1]; |
51 | CFClient::error $_[1]; |
45 | }; |
52 | }; |
46 | |
53 | |
47 | #$SIG{__WARN__} = sub { Carp::cluck $_[0] };#d# |
54 | #$SIG{__WARN__} = sub { Carp::cluck $_[0] };#d# |
48 | |
55 | |
… | |
… | |
50 | |
57 | |
51 | my $MAX_FPS = 60; |
58 | my $MAX_FPS = 60; |
52 | my $MIN_FPS = 5; # unused as of yet |
59 | my $MIN_FPS = 5; # unused as of yet |
53 | |
60 | |
54 | our $META_SERVER = "crossfire.real-time.com:13326"; |
61 | our $META_SERVER = "crossfire.real-time.com:13326"; |
55 | |
|
|
56 | our $FACEMAP; |
|
|
57 | our $TILECACHE; |
|
|
58 | our $MAPCACHE; |
|
|
59 | |
62 | |
60 | our $LAST_REFRESH; |
63 | our $LAST_REFRESH; |
61 | our $NOW; |
64 | our $NOW; |
62 | |
65 | |
63 | our $CFG; |
66 | our $CFG; |
… | |
… | |
83 | our $LOGVIEW; |
86 | our $LOGVIEW; |
84 | our $CONSOLE; |
87 | our $CONSOLE; |
85 | our $METASERVER; |
88 | our $METASERVER; |
86 | our $LOGIN_BUTTON; |
89 | our $LOGIN_BUTTON; |
87 | our $QUIT_DIALOG; |
90 | our $QUIT_DIALOG; |
|
|
91 | our $HOST_ENTRY; |
|
|
92 | our $FULLSCREEN_ENABLE; |
|
|
93 | our $PICKUP_ENABLE; |
|
|
94 | our $SERVER_INFO; |
88 | |
95 | |
|
|
96 | our $SETUP_DIALOG; |
|
|
97 | our $SETUP_NOTEBOOK; |
|
|
98 | our $SETUP_SERVER; |
|
|
99 | our $SETUP_KEYBOARD; |
|
|
100 | |
|
|
101 | our $PL_NOTEBOOK; |
|
|
102 | our $PL_WINDOW; |
|
|
103 | |
|
|
104 | our $INVENTORY_PAGE; |
|
|
105 | our $STATS_PAGE; |
|
|
106 | our $SKILL_PAGE; |
|
|
107 | our $SPELL_PAGE; |
|
|
108 | |
|
|
109 | our $HELP_WINDOW; |
|
|
110 | our $MESSAGE_WINDOW; |
89 | our $FLOORBOX; |
111 | our $FLOORBOX; |
90 | our $GAUGES; |
112 | our $GAUGES; |
91 | our $STATWIDS; |
113 | our $STATWIDS; |
92 | |
114 | |
93 | our $SDL_ACTIVE; |
115 | our $SDL_ACTIVE; |
… | |
… | |
99 | |
121 | |
100 | our $ALT_ENTER_MESSAGE; |
122 | our $ALT_ENTER_MESSAGE; |
101 | our $STATUSBOX; |
123 | our $STATUSBOX; |
102 | our $DEBUG_STATUS; |
124 | our $DEBUG_STATUS; |
103 | |
125 | |
104 | our $INVWIN; |
|
|
105 | our $INV; |
126 | our $INV; |
106 | our $INVR; |
127 | our $INVR; |
107 | our $INVR_LBL; |
128 | our $INV_RIGHT_HB; |
|
|
129 | |
|
|
130 | our $BIND_EDITOR; |
|
|
131 | our $BIND_UPD_CB; |
|
|
132 | |
|
|
133 | our $PICKUP_CFG; |
108 | |
134 | |
109 | sub status { |
135 | sub status { |
110 | $STATUSBOX->add (CFClient::UI::Label::escape $_[0], pri => -10, group => "status", timeout => 10, fg => [1, 1, 0, 1]); |
136 | $STATUSBOX->add (CFClient::asxml $_[0], pri => -10, group => "status", timeout => 10, fg => [1, 1, 0, 1]); |
111 | } |
137 | } |
112 | |
138 | |
113 | sub debug { |
139 | sub debug { |
114 | $DEBUG_STATUS->set_text ($_[0]); |
140 | $DEBUG_STATUS->set_text ($_[0]); |
115 | my ($w, $h) = $DEBUG_STATUS->size_request; |
141 | } |
116 | $DEBUG_STATUS->move ($WIDTH - $w, 0); |
142 | |
|
|
143 | sub destroy_query_dialog { |
|
|
144 | (delete $_[0]{query_dialog})->destroy |
|
|
145 | if $_[0]{query_dialog}; |
|
|
146 | } |
|
|
147 | |
|
|
148 | # server query dialog |
|
|
149 | sub server_query { |
|
|
150 | my ($conn, $flags, $prompt) = @_; |
|
|
151 | |
|
|
152 | $conn->{query_dialog} = my $dialog = new CFClient::UI::FancyFrame |
|
|
153 | x => "center", |
|
|
154 | y => "center", |
|
|
155 | title => "Server Query", |
|
|
156 | child => my $vbox = new CFClient::UI::VBox, |
|
|
157 | ; |
|
|
158 | |
|
|
159 | my @dialog = my $label = new CFClient::UI::Label |
|
|
160 | max_w => $::WIDTH * 0.4, |
|
|
161 | ellipsise => 0, |
|
|
162 | text => $prompt; |
|
|
163 | |
|
|
164 | if ($flags & CS_QUERY_YESNO) { |
|
|
165 | push @dialog, my $hbox = new CFClient::UI::HBox; |
|
|
166 | |
|
|
167 | $hbox->add (new CFClient::UI::Button |
|
|
168 | text => "No", |
|
|
169 | on_activate => sub { |
|
|
170 | $conn->send ("reply n"); |
|
|
171 | $dialog->destroy; |
|
|
172 | 0 |
|
|
173 | } |
|
|
174 | ); |
|
|
175 | $hbox->add (new CFClient::UI::Button |
|
|
176 | text => "Yes", |
|
|
177 | on_activate => sub { |
|
|
178 | $conn->send ("reply y"); |
|
|
179 | destroy_query_dialog $conn; |
|
|
180 | 0 |
|
|
181 | }, |
|
|
182 | ); |
|
|
183 | |
|
|
184 | $dialog->grab_focus; |
|
|
185 | |
|
|
186 | } elsif ($flags & CS_QUERY_SINGLECHAR) { |
|
|
187 | $dialog->{tooltip} = "Press a key (click on the entry to make sure it has keyboard focus)"; |
|
|
188 | |
|
|
189 | if ($prompt =~ /Now choose a character|Press any key for the next race/i) { |
|
|
190 | $MESSAGE_WINDOW->show; |
|
|
191 | |
|
|
192 | unshift @dialog, new CFClient::UI::Label |
|
|
193 | max_w => $::WIDTH * 0.4, |
|
|
194 | ellipsise => 0, |
|
|
195 | markup => "\nOr use your keyboard and the text entry below:\n"; |
|
|
196 | |
|
|
197 | unshift @dialog, my $table = new CFClient::UI::Table; |
|
|
198 | |
|
|
199 | $table->add (0, 0, new CFClient::UI::Button |
|
|
200 | text => "Next Race", |
|
|
201 | on_activate => sub { |
|
|
202 | $conn->send ("reply n"); |
|
|
203 | destroy_query_dialog $conn; |
|
|
204 | 0 |
|
|
205 | }, |
|
|
206 | ); |
|
|
207 | $table->add (2, 0, new CFClient::UI::Button |
|
|
208 | text => "Accept", |
|
|
209 | on_activate => sub { |
|
|
210 | $conn->send ("reply d"); |
|
|
211 | destroy_query_dialog $conn; |
|
|
212 | 0 |
|
|
213 | }, |
|
|
214 | ); |
|
|
215 | |
|
|
216 | unshift @dialog, new CFClient::UI::Label |
|
|
217 | max_w => $::WIDTH * 0.4, |
|
|
218 | ellipsise => 0, |
|
|
219 | markup => |
|
|
220 | "<big><b>Character Creation: Race</b></big>\n\n" |
|
|
221 | . "Look at the <b>Messages</b> window to see a description of this race " |
|
|
222 | . "and the center of the screen to see how this race looks like " |
|
|
223 | . "(<small>below this dialog window: you may need to move the dialog away and " |
|
|
224 | . "click into the display area to make it visible</small>).\n\n" |
|
|
225 | . "You can look at another race, or accept this race (you will cycle back to " |
|
|
226 | . "this race eventually, so you can take your time making this important choice." |
|
|
227 | ; |
|
|
228 | |
|
|
229 | } elsif ($prompt =~ /roll new stats/) { |
|
|
230 | if (my $stat = delete $conn->{stat_change_with}) { |
|
|
231 | $conn->send ("reply $stat"); |
|
|
232 | destroy_query_dialog $conn; |
|
|
233 | return; |
|
|
234 | } |
|
|
235 | |
|
|
236 | $STATS_PAGE->show; |
|
|
237 | $MESSAGE_WINDOW->hide; |
|
|
238 | |
|
|
239 | unshift @dialog, new CFClient::UI::Label |
|
|
240 | max_w => $::WIDTH * 0.4, |
|
|
241 | ellipsise => 0, |
|
|
242 | markup => "\nOr use your keyboard and the text entry below:\n"; |
|
|
243 | |
|
|
244 | unshift @dialog, my $table = new CFClient::UI::Table; |
|
|
245 | |
|
|
246 | # left: re-roll |
|
|
247 | $table->add (0, 0, new CFClient::UI::Button |
|
|
248 | text => "Roll Again", |
|
|
249 | on_activate => sub { |
|
|
250 | $conn->send ("reply y"); |
|
|
251 | destroy_query_dialog $conn; |
|
|
252 | 0 |
|
|
253 | }, |
|
|
254 | ); |
|
|
255 | |
|
|
256 | # center: swap stats |
|
|
257 | my ($sw1, $sw2) = map +(new CFClient::UI::Selector |
|
|
258 | expand => 1, |
|
|
259 | value => $_, |
|
|
260 | options => [ |
|
|
261 | [1 => "Str", "Strength ($conn->{stat}{+CS_STAT_STR})"], |
|
|
262 | [2 => "Dex", "Dexterity ($conn->{stat}{+CS_STAT_DEX})"], |
|
|
263 | [3 => "Con", "Constitution ($conn->{stat}{+CS_STAT_CON})"], |
|
|
264 | [4 => "Int", "Intelligence ($conn->{stat}{+CS_STAT_INT})"], |
|
|
265 | [5 => "Wis", "Wisdom ($conn->{stat}{+CS_STAT_WIS})"], |
|
|
266 | [6 => "Pow", "Power ($conn->{stat}{+CS_STAT_POW})"], |
|
|
267 | [7 => "Cha", "Charisma ($conn->{stat}{+CS_STAT_CHA})"], |
|
|
268 | ], |
|
|
269 | ), 1 .. 2; |
|
|
270 | |
|
|
271 | $table->add (2, 0, new CFClient::UI::Button |
|
|
272 | text => "Swap Stats", |
|
|
273 | on_activate => sub { |
|
|
274 | $conn->{stat_change_with} = $sw2->{value}; |
|
|
275 | $conn->send ("reply $sw1->{value}"); |
|
|
276 | destroy_query_dialog $conn; |
|
|
277 | 0 |
|
|
278 | }, |
|
|
279 | ); |
|
|
280 | $table->add (2, 1, new CFClient::UI::HBox children => [$sw1, $sw2]); |
|
|
281 | |
|
|
282 | # right: accept |
|
|
283 | $table->add (4, 0, new CFClient::UI::Button |
|
|
284 | text => "Accept", |
|
|
285 | on_activate => sub { |
|
|
286 | $conn->send ("reply n"); |
|
|
287 | $STATS_PAGE->hide; |
|
|
288 | destroy_query_dialog $conn; |
|
|
289 | 0 |
|
|
290 | }, |
|
|
291 | ); |
|
|
292 | |
|
|
293 | unshift @dialog, my $hbox = new CFClient::UI::HBox; |
|
|
294 | for ( |
|
|
295 | [Str => CS_STAT_STR], |
|
|
296 | [Dex => CS_STAT_DEX], |
|
|
297 | [Con => CS_STAT_CON], |
|
|
298 | [Int => CS_STAT_INT], |
|
|
299 | [Wis => CS_STAT_WIS], |
|
|
300 | [Pow => CS_STAT_POW], |
|
|
301 | [Cha => CS_STAT_CHA], |
|
|
302 | ) { |
|
|
303 | my ($name, $id) = @$_; |
|
|
304 | $hbox->add (new CFClient::UI::Label |
|
|
305 | markup => "$conn->{stat}{$id} <span foreground='yellow'>$name</span>", |
|
|
306 | align => 0, |
|
|
307 | expand => 1, |
|
|
308 | can_events => 1, |
|
|
309 | can_hover => 1, |
|
|
310 | tooltip => $CFClient::STAT_TOOLTIP{$name}, |
|
|
311 | ); |
|
|
312 | } |
|
|
313 | |
|
|
314 | unshift @dialog, new CFClient::UI::Label |
|
|
315 | max_w => $::WIDTH * 0.4, |
|
|
316 | ellipsise => 0, |
|
|
317 | markup => |
|
|
318 | "<big><b>Character Creation: Stats</b></big>\n\n" |
|
|
319 | . "<b>Stats</b> are a very important aspect of your character. You can use the tooltips to learn what each Stat governs.\n\n" |
|
|
320 | . "The stats generated by the server are always sorted from Str (highest) to Cha (lowest). " |
|
|
321 | . "They will be modified later by both the race and the class you choose.\n\n" |
|
|
322 | . "You can create another set of stats, swap two stat values with each other or accept the stats as shown below and continue.\n" |
|
|
323 | ; |
|
|
324 | } |
|
|
325 | |
|
|
326 | push @dialog, my $entry = new CFClient::UI::Entry |
|
|
327 | on_changed => sub { |
|
|
328 | $conn->send ("reply $_[1]"); |
|
|
329 | destroy_query_dialog $conn; |
|
|
330 | 0 |
|
|
331 | }, |
|
|
332 | ; |
|
|
333 | |
|
|
334 | $entry->grab_focus; |
|
|
335 | |
|
|
336 | } else { |
|
|
337 | $dialog->{tooltip} = "Enter the reply and press return (click on the entry to make sure it has keyboard focus)"; |
|
|
338 | |
|
|
339 | push @dialog, my $entry = new CFClient::UI::Entry |
|
|
340 | $flags & CS_QUERY_HIDEINPUT ? (hidden => "*") : (), |
|
|
341 | on_activate => sub { |
|
|
342 | $conn->send ("reply $_[1]"); |
|
|
343 | destroy_query_dialog $conn; |
|
|
344 | 0 |
|
|
345 | }, |
|
|
346 | ; |
|
|
347 | |
|
|
348 | $entry->grab_focus; |
|
|
349 | } |
|
|
350 | |
|
|
351 | $vbox->add (@dialog); |
|
|
352 | $dialog->show; |
117 | } |
353 | } |
118 | |
354 | |
119 | sub start_game { |
355 | sub start_game { |
120 | status "logging in..."; |
356 | status "logging in..."; |
121 | |
357 | |
|
|
358 | $LOGIN_BUTTON->set_text ("Logout"); |
|
|
359 | $SETUP_DIALOG->hide; |
|
|
360 | |
122 | my $mapsize = List::Util::min 32, List::Util::max 11, int $WIDTH * $CFG->{mapsize} * 0.01 / 32; |
361 | my $mapsize = List::Util::min 32, List::Util::max 11, int $WIDTH * $CFG->{mapsize} * 0.01 / 32; |
123 | |
362 | |
124 | $MAPCACHE = CFClient::db_table "mapcache_$CFG->{host}"; |
363 | my ($host, $port) = split /:/, $CFG->{profile}{default}{host}; |
|
|
364 | |
125 | $MAP = new CFClient::Map $mapsize, $mapsize; |
365 | $MAP = new CFClient::Map $mapsize, $mapsize; |
126 | |
|
|
127 | my ($host, $port) = split /:/, $CFG->{host}; |
|
|
128 | |
366 | |
129 | $CONN = eval { |
367 | $CONN = eval { |
130 | new conn |
368 | new CFClient::Protocol |
131 | host => $host, |
369 | host => $host, |
132 | port => $port || 13327, |
370 | port => $port || 13327, |
133 | user => $CFG->{user}, |
371 | user => $CFG->{profile}{default}{user}, |
134 | pass => $CFG->{password}, |
372 | pass => $CFG->{profile}{default}{password}, |
135 | mapw => $mapsize, |
373 | mapw => $mapsize, |
136 | maph => $mapsize, |
374 | maph => $mapsize, |
137 | ; |
375 | |
|
|
376 | map_widget => $MAPWIDGET, |
|
|
377 | logview => $LOGVIEW, |
|
|
378 | statusbox => $STATUSBOX, |
|
|
379 | map => $MAP, |
|
|
380 | mapmap => $MAPMAP, |
|
|
381 | query => \&server_query, |
|
|
382 | |
|
|
383 | sound_play => sub { |
|
|
384 | my ($x, $y, $soundnum, $type) = @_; |
|
|
385 | |
|
|
386 | $SDL_MIXER |
|
|
387 | or return; |
|
|
388 | |
|
|
389 | my $chunk = $AUDIO_CHUNKS{$SOUNDS[$soundnum]} |
|
|
390 | or return; |
|
|
391 | |
|
|
392 | $chunk->play; |
|
|
393 | }, |
138 | }; |
394 | }; |
139 | |
395 | |
140 | if ($CONN) { |
396 | if ($CONN) { |
141 | CFClient::lowdelay fileno $CONN->{fh}; |
397 | CFClient::lowdelay fileno $CONN->{fh}; |
142 | |
398 | |
143 | $LOGIN_BUTTON->set_text ("Logout"); |
|
|
144 | status "login successful"; |
399 | status "login successful"; |
145 | |
|
|
146 | $BUTTONBAR->{children}[1]->emit ("activate") |
|
|
147 | if $BUTTONBAR->{children}[1]->{state}; |
|
|
148 | |
|
|
149 | } else { |
400 | } else { |
150 | status "unable to connect"; |
401 | status "unable to connect"; |
151 | stop_game(); |
402 | stop_game(); |
152 | } |
403 | } |
153 | } |
404 | } |
154 | |
405 | |
155 | sub stop_game { |
406 | sub stop_game { |
|
|
407 | $LOGIN_BUTTON->set_text ("Login"); |
|
|
408 | $SETUP_NOTEBOOK->set_current_page ($SETUP_SERVER); |
|
|
409 | $SETUP_DIALOG->show; |
|
|
410 | $PL_WINDOW->hide; |
|
|
411 | $SPELL_PAGE->clear_spells; |
|
|
412 | |
156 | return unless $CONN; |
413 | return unless $CONN; |
157 | |
414 | |
158 | status "connection closed"; |
415 | status "connection closed"; |
159 | $LOGIN_BUTTON->set_text ("Login"); |
416 | |
|
|
417 | destroy_query_dialog $CONN; |
160 | $CONN->destroy; |
418 | $CONN->destroy; |
161 | $CONN = 0; # false, does not autovivify |
419 | $CONN = 0; # false, does not autovivify |
162 | |
420 | |
163 | $BUTTONBAR->{children}[1]->emit ("activate") |
|
|
164 | unless $BUTTONBAR->{children}[1]->{state}; |
|
|
165 | |
|
|
166 | undef $MAPCACHE; |
|
|
167 | undef $MAP; |
421 | undef $MAP; |
168 | } |
422 | } |
169 | |
423 | |
170 | sub client_setup { |
424 | sub graphics_setup { |
171 | my $dialog = new CFClient::UI::FancyFrame |
|
|
172 | title => "Client Setup", |
|
|
173 | child => (my $vbox = new CFClient::UI::VBox); |
425 | my $vbox = new CFClient::UI::VBox; |
|
|
426 | |
174 | $vbox->add (my $table = new CFClient::UI::Table expand => 1, col_expand => [0, 1]); |
427 | $vbox->add (my $table = new CFClient::UI::Table expand => 1, col_expand => [0, 1]); |
175 | |
428 | |
176 | $table->add (0, 0, new CFClient::UI::Label valign => 0, align => 1, text => "Video Mode"); |
429 | $table->add (0, 0, new CFClient::UI::Label valign => 0, align => 1, text => "Video Mode"); |
177 | $table->add (1, 0, my $hbox = new CFClient::UI::HBox); |
430 | $table->add (1, 0, my $hbox = new CFClient::UI::HBox); |
178 | |
431 | |
179 | $hbox->add (my $mode_slider = new CFClient::UI::Slider expand => 1, req_w => 100, range => [$CFG->{sdl_mode}, 0, $#SDL_MODES, 0, 1]); |
432 | $hbox->add (my $mode_slider = new CFClient::UI::Slider force_w => $WIDTH * 0.1, expand => 1, range => [$CFG->{sdl_mode}, 0, $#SDL_MODES, 0, 1]); |
180 | $hbox->add (my $mode_label = new CFClient::UI::Label align => 0, valign => 0, height => 0.8, template => "9999x9999"); |
433 | $hbox->add (my $mode_label = new CFClient::UI::Label align => 0, valign => 0, height => 0.8, template => "9999x9999"); |
181 | |
434 | |
182 | $mode_slider->connect (changed => sub { |
435 | $mode_slider->connect (changed => sub { |
183 | my ($self, $value) = @_; |
436 | my ($self, $value) = @_; |
184 | |
437 | |
… | |
… | |
188 | $mode_slider->emit (changed => $mode_slider->{range}[0]); |
441 | $mode_slider->emit (changed => $mode_slider->{range}[0]); |
189 | |
442 | |
190 | my $row = 1; |
443 | my $row = 1; |
191 | |
444 | |
192 | $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Fullscreen"); |
445 | $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Fullscreen"); |
193 | $table->add (1, $row++, new CFClient::UI::CheckBox |
446 | $table->add (1, $row++, $FULLSCREEN_ENABLE = new CFClient::UI::CheckBox |
194 | state => $CFG->{fullscreen}, |
447 | state => $CFG->{fullscreen}, |
195 | tooltip => "Bring the client into fullscreen mode.", |
448 | tooltip => "Bring the client into fullscreen mode.", |
196 | connect_changed => sub { |
449 | on_changed => sub { my ($self, $value) = @_; $CFG->{fullscreen} = $value; 0 } |
197 | my ($self, $value) = @_; |
|
|
198 | $CFG->{fullscreen} = $value; |
|
|
199 | } |
|
|
200 | ); |
450 | ); |
201 | |
451 | |
202 | $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Fast & Ugly"); |
452 | $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Fast & Ugly"); |
203 | $table->add (1, $row++, new CFClient::UI::CheckBox |
453 | $table->add (1, $row++, new CFClient::UI::CheckBox |
204 | state => $CFG->{fast}, |
454 | state => $CFG->{fast}, |
205 | tooltip => "Lower the visual quality considerably to speed up rendering.", |
455 | tooltip => "Lower the visual quality considerably to speed up rendering.", |
206 | connect_changed => sub { |
456 | on_changed => sub { my ($self, $value) = @_; $CFG->{fast} = $value; 0 } |
207 | my ($self, $value) = @_; |
|
|
208 | $CFG->{fast} = $value; |
|
|
209 | } |
|
|
210 | ); |
457 | ); |
211 | |
458 | |
212 | $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Map Scale"); |
459 | $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Map Scale"); |
213 | $table->add (1, $row++, new CFClient::UI::Slider |
460 | $table->add (1, $row++, new CFClient::UI::Slider |
214 | range => [(log $CFG->{map_scale}) / (log 2), -3, 1, 0, 1], |
461 | range => [(log $CFG->{map_scale}) / (log 2), -3, 1, 0, 1], |
215 | tooltip => "Enlarge or shrink the displayed map. Changes are instant.", |
462 | tooltip => "Enlarge or shrink the displayed map. Changes are instant.", |
216 | connect_changed => sub { |
463 | on_changed => sub { my ($self, $value) = @_; $CFG->{map_scale} = 2 ** $value; 0 } |
217 | my ($self, $value) = @_; |
|
|
218 | $CFG->{map_scale} = 2 ** $value; |
|
|
219 | } |
|
|
220 | ); |
464 | ); |
221 | |
465 | |
222 | $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Fog of War"); |
466 | $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Fog of War"); |
223 | $table->add (1, $row++, new CFClient::UI::CheckBox |
467 | $table->add (1, $row++, new CFClient::UI::CheckBox |
224 | state => $CFG->{fow_enable}, |
468 | state => $CFG->{fow_enable}, |
225 | tooltip => "<b>Fog-of-War</b> marks areas that cannot be seen by the player. Changes are instant.", |
469 | tooltip => "<b>Fog-of-War</b> marks areas that cannot be seen by the player. Changes are instant.", |
226 | connect_changed => sub { |
470 | on_changed => sub { my ($self, $value) = @_; $CFG->{fow_enable} = $value; 0 } |
227 | my ($self, $value) = @_; |
|
|
228 | $CFG->{fow_enable} = $value; |
|
|
229 | } |
|
|
230 | ); |
471 | ); |
231 | |
472 | |
232 | $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "FoW Intensity"); |
473 | $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "FoW Intensity"); |
233 | $table->add (1, $row++, new CFClient::UI::Slider |
474 | $table->add (1, $row++, new CFClient::UI::Slider |
234 | range => [$CFG->{fow_intensity}, 0, 1, 0, 1 / 256], |
475 | range => [$CFG->{fow_intensity}, 0, 1, 0, 1 / 256], |
235 | tooltip => "<b>Fog of War Lightness.</b> The higher the intensity, the lighter the Fog-of-War color. Changes are instant.", |
476 | tooltip => "<b>Fog of War Lightness.</b> The higher the intensity, the lighter the Fog-of-War color. Changes are instant.", |
236 | connect_changed => sub { |
477 | on_changed => sub { my ($self, $value) = @_; $CFG->{fow_intensity} = $value; 0 } |
237 | my ($self, $value) = @_; |
|
|
238 | $CFG->{fow_intensity} = $value; |
|
|
239 | } |
|
|
240 | ); |
478 | ); |
241 | |
479 | |
242 | $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "FoW Smooth"); |
480 | $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "FoW Smooth"); |
243 | $table->add (1, $row++, new CFClient::UI::CheckBox |
481 | $table->add (1, $row++, new CFClient::UI::CheckBox |
244 | state => $CFG->{fow_smooth}, |
482 | state => $CFG->{fow_smooth}, |
245 | tooltip => "Smooth the Fog-of-War a bit to make it more realistic. Changes are instant.", |
483 | tooltip => "Smooth the Fog-of-War a bit to make it more realistic. Changes are instant.", |
246 | connect_changed => sub { |
484 | on_changed => sub { |
247 | my ($self, $value) = @_; |
485 | my ($self, $value) = @_; |
248 | $CFG->{fow_smooth} = $value; |
486 | $CFG->{fow_smooth} = $value; |
249 | status "Fog of War smoothing requires OpenGL 1.2 or higher" if $CFClient::GL_VERSION < 1.2; |
487 | status "Fog of War smoothing requires OpenGL 1.2 or higher" if $CFClient::OpenGL::GL_VERSION < 1.2; |
|
|
488 | 0 |
250 | } |
489 | } |
251 | ); |
490 | ); |
252 | |
491 | |
253 | $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "GUI Fontsize"); |
492 | $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "GUI Fontsize"); |
254 | $table->add (1, $row++, new CFClient::UI::Slider |
493 | $table->add (1, $row++, new CFClient::UI::Slider |
255 | range => [$CFG->{gui_fontsize}, 0.5, 2, 0, 0.1], |
494 | range => [$CFG->{gui_fontsize}, 0.5, 2, 0, 0.1], |
256 | tooltip => "The base font size used by most GUI elements that do not have their own setting.", |
495 | tooltip => "The base font size used by most GUI elements that do not have their own setting.", |
257 | connect_changed => sub { $CFG->{gui_fontsize} = $_[1] }, |
496 | on_changed => sub { $CFG->{gui_fontsize} = $_[1]; 0 }, |
258 | ); |
497 | ); |
259 | |
498 | |
260 | $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Message Fontsize"); |
499 | $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Message Fontsize"); |
261 | $table->add (1, $row++, new CFClient::UI::Slider |
500 | $table->add (1, $row++, new CFClient::UI::Slider |
262 | range => [$CFG->{log_fontsize}, 0.5, 2, 0, 0.1], |
501 | range => [$CFG->{log_fontsize}, 0.5, 2, 0, 0.1], |
263 | tooltip => "The font size used by the <b>message/server log</b> window only. Changes are instant.", |
502 | tooltip => "The font size used by the <b>message/server log</b> window only. Changes are instant.", |
264 | connect_changed => sub { $LOGVIEW->set_fontsize ($CFG->{log_fontsize} = $_[1]) }, |
503 | on_changed => sub { $LOGVIEW->set_fontsize ($CFG->{log_fontsize} = $_[1]); 0 }, |
265 | ); |
|
|
266 | |
|
|
267 | $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Stats Fontsize"); |
|
|
268 | |
|
|
269 | $table->add (1, $row++, new CFClient::UI::Slider |
|
|
270 | range => [$CFG->{stat_fontsize}, 0.5, 2, 0, 0.1], |
|
|
271 | tooltip => "The font size used by the <b>statistics window</b> only. Changes are instant.", |
|
|
272 | connect_changed => sub { |
|
|
273 | $CFG->{stat_fontsize} = $_[1]; |
|
|
274 | &set_stats_window_fontsize; |
|
|
275 | } |
|
|
276 | ); |
504 | ); |
277 | |
505 | |
278 | $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Gauge fontsize"); |
506 | $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Gauge fontsize"); |
279 | $table->add (1, $row++, new CFClient::UI::Slider |
507 | $table->add (1, $row++, new CFClient::UI::Slider |
280 | range => [$CFG->{gauge_fontsize}, 0.5, 2, 0, 0.1], |
508 | range => [$CFG->{gauge_fontsize}, 0.5, 2, 0, 0.1], |
281 | tooltip => "Adjusts the fontsize of the gauges at the bottom right. Changes are instant.", |
509 | tooltip => "Adjusts the fontsize of the gauges at the bottom right. Changes are instant.", |
282 | connect_changed => sub { |
510 | on_changed => sub { |
283 | $CFG->{gauge_fontsize} = $_[1]; |
511 | $CFG->{gauge_fontsize} = $_[1]; |
284 | &set_gauge_window_fontsize; |
512 | &set_gauge_window_fontsize; |
|
|
513 | 0 |
285 | } |
514 | } |
286 | ); |
515 | ); |
287 | |
516 | |
288 | $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Gauge size"); |
517 | $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Gauge size"); |
289 | $table->add (1, $row++, new CFClient::UI::Slider |
518 | $table->add (1, $row++, new CFClient::UI::Slider |
290 | range => [$CFG->{gauge_size}, 0.2, 0.8], |
519 | range => [$CFG->{gauge_size}, 0.2, 0.8], |
291 | tooltip => "Adjust the size of the stats gauges at the bottom right. Changes are instant.", |
520 | tooltip => "Adjust the size of the stats gauges at the bottom right. Changes are instant.", |
292 | connect_changed => sub { |
521 | on_changed => sub { |
293 | $CFG->{gauge_size} = $_[1]; |
522 | $CFG->{gauge_size} = $_[1]; |
294 | $GAUGES->{win}->set_size ($WIDTH, int $HEIGHT * $CFG->{gauge_size}); |
523 | $GAUGES->{win}->set_size ($WIDTH, int $HEIGHT * $CFG->{gauge_size}); |
|
|
524 | 0 |
295 | } |
525 | } |
296 | ); |
526 | ); |
297 | |
527 | |
298 | $table->add (1, $row++, new CFClient::UI::Button |
528 | $table->add (1, $row++, new CFClient::UI::Button |
299 | expand => 1, align => 0, text => "Apply", |
529 | expand => 1, align => 0, text => "Apply", |
300 | tooltip => "Apply the video settings", |
530 | tooltip => "Apply the video settings", |
301 | connect_activate => sub { |
531 | on_activate => sub { |
302 | video_shutdown (); |
532 | video_shutdown (); |
303 | video_init (); |
533 | video_init (); |
|
|
534 | 0 |
304 | } |
535 | } |
305 | ); |
536 | ); |
|
|
537 | |
|
|
538 | $vbox |
|
|
539 | } |
|
|
540 | |
|
|
541 | sub audio_setup { |
|
|
542 | my $vbox = new CFClient::UI::VBox; |
|
|
543 | |
|
|
544 | $vbox->add (my $table = new CFClient::UI::Table expand => 1, col_expand => [0, 1]); |
|
|
545 | |
|
|
546 | my $row = 0; |
306 | |
547 | |
307 | $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Audio Enable"); |
548 | $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Audio Enable"); |
308 | $table->add (1, $row++, new CFClient::UI::CheckBox |
549 | $table->add (1, $row++, new CFClient::UI::CheckBox |
309 | state => $CFG->{audio_enable}, |
550 | state => $CFG->{audio_enable}, |
310 | tooltip => "<b>Master Audio Enable.</b> If enabled, sound effects and music will be played. If disabled, no audio will be used and the soundcard will not be opened.", |
551 | tooltip => "<b>Master Audio Enable.</b> If enabled, sound effects and music will be played. If disabled, no audio will be used and the soundcard will not be opened.", |
311 | connect_changed => sub { |
552 | on_changed => sub { $CFG->{audio_enable} = $_[1]; 0 } |
312 | $CFG->{audio_enable} = $_[1]; |
|
|
313 | } |
|
|
314 | ); |
553 | ); |
315 | # $table->add (0, 9, new CFClient::UI::Label valign => 0, align => 1, text => "Effects Volume"); |
554 | # $table->add (0, 9, new CFClient::UI::Label valign => 0, align => 1, text => "Effects Volume"); |
316 | # $table->add (1, 8, new CFClient::UI::Slider range => [$CFG->{effects_volume}, 0, 128, 1], connect_changed => sub { |
555 | # $table->add (1, 8, new CFClient::UI::Slider range => [$CFG->{effects_volume}, 0, 128, 1], on_changed => sub { |
317 | # $CFG->{effects_volume} = $_[1]; |
556 | # $CFG->{effects_volume} = $_[1]; |
318 | # }); |
557 | # }); |
319 | $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Background Music"); |
558 | $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Background Music"); |
320 | $table->add (1, $row++, my $hbox = new CFClient::UI::HBox); |
559 | $table->add (1, $row++, my $hbox = new CFClient::UI::HBox); |
321 | $hbox->add (new CFClient::UI::CheckBox |
560 | $hbox->add (new CFClient::UI::CheckBox |
322 | expand => 1, state => $CFG->{bgm_enable}, |
561 | expand => 1, state => $CFG->{bgm_enable}, |
323 | tooltip => "If enabled, playing of background music is enabled. If disabled, no background music will be played.", |
562 | tooltip => "If enabled, playing of background music is enabled. If disabled, no background music will be played.", |
324 | connect_changed => sub { |
563 | on_changed => sub { $CFG->{bgm_enable} = $_[1]; 0 } |
325 | $CFG->{bgm_enable} = $_[1]; |
|
|
326 | } |
|
|
327 | ); |
564 | ); |
328 | $hbox->add (new CFClient::UI::Slider |
565 | $hbox->add (new CFClient::UI::Slider |
329 | expand => 1, range => [$CFG->{bgm_volume}, 0, 1, 0, 1/128], |
566 | expand => 1, range => [$CFG->{bgm_volume}, 0, 1, 0, 1/128], |
330 | tooltip => "The volume of the background music. Changes are instant.", |
567 | tooltip => "The volume of the background music. Changes are instant.", |
331 | connect_changed => sub { |
568 | on_changed => sub { $CFG->{bgm_volume} = $_[1]; CFClient::MixMusic::volume $_[1] * 128; 0 } |
332 | $CFG->{bgm_volume} = $_[1]; |
|
|
333 | CFClient::MixMusic::volume $_[1] * 128; |
|
|
334 | } |
|
|
335 | ); |
569 | ); |
336 | |
570 | |
337 | $table->add (1, $row++, new CFClient::UI::Button |
571 | $table->add (1, $row++, new CFClient::UI::Button |
338 | expand => 1, align => 0, text => "Apply", |
572 | expand => 1, align => 0, text => "Apply", |
339 | tooltip => "Apply the audio settings", |
573 | tooltip => "Apply the audio settings", |
340 | connect_activate => sub { |
574 | on_activate => sub { |
341 | audio_shutdown (); |
575 | audio_shutdown (); |
342 | audio_init (); |
576 | audio_init (); |
|
|
577 | 0 |
343 | } |
578 | } |
344 | ); |
579 | ); |
345 | |
580 | |
346 | $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Chat Command"); |
581 | $vbox |
347 | $table->add (1, $row++, my $saycmd = new CFClient::UI::Entry |
|
|
348 | text => $CFG->{say_command}, |
|
|
349 | tooltip => "This is the command that will be used if you write a line in the message window entry or press <b>\"</b> in the map window. " |
|
|
350 | . "Usually you want to enter something like 'say' or 'shout' or 'gsay' here. " |
|
|
351 | . "But you could also set it to <b>tell <i>playername</i></b> to only chat with that user.", |
|
|
352 | connect_changed => sub { |
|
|
353 | my ($self, $value) = @_; |
|
|
354 | $CFG->{say_command} = $value; |
|
|
355 | } |
|
|
356 | ); |
|
|
357 | |
|
|
358 | $dialog |
|
|
359 | } |
|
|
360 | |
|
|
361 | sub set_stats_window_fontsize { |
|
|
362 | for (values %{$STATWIDS}) { |
|
|
363 | $_->set_fontsize ($::CFG->{stat_fontsize}); |
|
|
364 | } |
|
|
365 | } |
582 | } |
366 | |
583 | |
367 | sub set_gauge_window_fontsize { |
584 | sub set_gauge_window_fontsize { |
368 | for (map { $GAUGES->{$_} } grep { $_ ne 'win' } keys %{$GAUGES}) { |
585 | for (map { $GAUGES->{$_} } grep { $_ ne 'win' } keys %{$GAUGES}) { |
369 | $_->set_fontsize ($::CFG->{gauge_fontsize}); |
586 | $_->set_fontsize ($::CFG->{gauge_fontsize}); |
… | |
… | |
372 | |
589 | |
373 | sub make_gauge_window { |
590 | sub make_gauge_window { |
374 | my $gh = int $HEIGHT * $CFG->{gauge_size}; |
591 | my $gh = int $HEIGHT * $CFG->{gauge_size}; |
375 | |
592 | |
376 | my $win = new CFClient::UI::Frame ( |
593 | my $win = new CFClient::UI::Frame ( |
377 | req_y => -1, |
594 | force_x => 0, |
|
|
595 | force_y => "max", |
378 | user_w => $WIDTH, |
596 | force_w => $WIDTH, |
379 | user_h => $gh, |
597 | force_h => $gh, |
380 | ); |
598 | ); |
381 | |
599 | |
382 | $win->add (my $hbox = new CFClient::UI::HBox |
600 | $win->add (my $hbox = new CFClient::UI::HBox |
383 | children => [ |
601 | children => [ |
384 | (new CFClient::UI::HBox expand => 1), |
602 | (new CFClient::UI::HBox expand => 1), |
… | |
… | |
420 | &set_gauge_window_fontsize; |
638 | &set_gauge_window_fontsize; |
421 | |
639 | |
422 | $win |
640 | $win |
423 | } |
641 | } |
424 | |
642 | |
|
|
643 | sub debug_setup { |
|
|
644 | my $table = new CFClient::UI::Table; |
|
|
645 | |
|
|
646 | $table->add (0, 0, new CFClient::UI::Label text => "Widget Borders"); |
|
|
647 | $table->add (1, 0, new CFClient::UI::CheckBox on_changed => sub { $ENV{CFPLUS_DEBUG} ^= 1; 0 }); |
|
|
648 | $table->add (0, 1, new CFClient::UI::Label text => "Tooltip Widget Info"); |
|
|
649 | $table->add (1, 1, new CFClient::UI::CheckBox on_changed => sub { $ENV{CFPLUS_DEBUG} ^= 2; 0 }); |
|
|
650 | $table->add (0, 2, new CFClient::UI::Label text => "Show FPS"); |
|
|
651 | $table->add (1, 2, new CFClient::UI::CheckBox on_changed => sub { $ENV{CFPLUS_DEBUG} ^= 4; 0 }); |
|
|
652 | $table->add (0, 3, new CFClient::UI::Label text => "Suppress Tooltips"); |
|
|
653 | $table->add (1, 3, new CFClient::UI::CheckBox on_changed => sub { $ENV{CFPLUS_DEBUG} ^= 8; 0 }); |
|
|
654 | |
|
|
655 | my @default_smooth = (0.05, 0.13, 0.05, 0.13, 0.30, 0.13, 0.05, 0.13, 0.05); |
|
|
656 | |
|
|
657 | for my $x (0..2) { |
|
|
658 | for my $y (0 .. 2) { |
|
|
659 | $table->add ($x + 3, $y, |
|
|
660 | new CFClient::UI::Entry |
|
|
661 | text => $default_smooth[$x * 3 + $y], |
|
|
662 | on_changed => sub { $MAP->{smooth_matrix}[$x * 3 + $y] = $_[1] if $MAP; 0 }, |
|
|
663 | ); |
|
|
664 | } |
|
|
665 | } |
|
|
666 | |
|
|
667 | |
|
|
668 | $table |
|
|
669 | } |
|
|
670 | |
425 | sub make_stats_window { |
671 | sub stats_window { |
426 | my $tgw = new CFClient::UI::FancyFrame title => "Stats"; |
672 | my $r = new CFClient::UI::ScrolledWindow ( |
|
|
673 | expand => 1, |
|
|
674 | scroll_y => 1 |
|
|
675 | ); |
|
|
676 | $r->add (my $vb = new CFClient::UI::VBox); |
427 | |
677 | |
428 | $tgw->add (new CFClient::UI::Window child => my $vb = new CFClient::UI::VBox); |
|
|
429 | $vb->add ($STATWIDS->{title} = new CFClient::UI::Label valign => 0, align => -1, text => "Title:", expand => 1, |
678 | $vb->add ($STATWIDS->{title} = new CFClient::UI::Label valign => 0, align => -1, text => "Title:", expand => 1, |
430 | can_hover => 1, can_events => 1, |
679 | can_hover => 1, can_events => 1, |
431 | tooltip => "Your name and title. You can change your title by using the <b>title</b> command, if supported by the server."); |
680 | tooltip => "Your name and title. You can change your title by using the <b>title</b> command, if supported by the server."); |
432 | $vb->add ($STATWIDS->{map} = new CFClient::UI::Label valign => 0, align => -1, text => "Map:", expand => 1, |
681 | $vb->add ($STATWIDS->{map} = new CFClient::UI::Label valign => 0, align => -1, text => "Map:", expand => 1, |
433 | can_hover => 1, can_events => 1, |
682 | can_hover => 1, can_events => 1, |
434 | tooltip => "The map you are currently on (if supported by the server)."); |
683 | tooltip => "The map you are currently on (if supported by the server)."); |
435 | |
684 | |
436 | $vb->add (my $hb0 = new CFClient::UI::HBox); |
685 | $vb->add (my $hb0 = new CFClient::UI::HBox); |
437 | $hb0->add ($STATWIDS->{weight} = new CFClient::UI::Label valign => 0, align => -1, text => "Weight:", expand => 1, |
686 | $hb0->add ($STATWIDS->{weight} = new CFClient::UI::Label valign => 0, align => -1, text => "Weight:", expand => 1, |
438 | can_hover => 1, can_events => 1, |
687 | can_hover => 1, can_events => 1, |
439 | tooltip => "This is the amount the Player weights."); |
688 | tooltip => "The weight of the player including all inventory items."); |
440 | $hb0->add ($STATWIDS->{m_weight} = new CFClient::UI::Label valign => 0, align => -1, text => "Max weight:", expand => 1, |
689 | $hb0->add ($STATWIDS->{m_weight} = new CFClient::UI::Label valign => 0, align => -1, text => "Max weight:", expand => 1, |
441 | can_hover => 1, can_events => 1, |
690 | can_hover => 1, can_events => 1, |
442 | tooltip => "The weight limit, you can't carry more than this."); |
691 | tooltip => "The weight limit: you cannot carry more than this."); |
443 | |
|
|
444 | |
692 | |
445 | $vb->add (my $hb = new CFClient::UI::HBox expand => 1); |
693 | $vb->add (my $hb = new CFClient::UI::HBox expand => 1); |
446 | $hb->add (my $tbl = new CFClient::UI::Table expand => 1); |
694 | $hb->add (my $tbl = new CFClient::UI::Table expand => 1); |
447 | |
695 | |
448 | my $color2 = [1, 1, 0]; |
696 | my $color2 = [1, 1, 0]; |
449 | |
697 | |
450 | for ( |
698 | for ( |
451 | [0, 0, st_str => "Str", 30, "<b>Physical Strength</b>, determines damage dealt with weapons, how much you can carry, and how often you can attack"], |
699 | [0, 0, st_str => "Str", 30], |
452 | [0, 1, st_dex => "Dex", 30, "<b>Dexterity</b>, your physical agility. Determines chance of being hit and affects armor class and speed"], |
700 | [0, 1, st_dex => "Dex", 30], |
453 | [0, 2, st_con => "Con", 30, "<b>Constitution</b>, physical health and toughness. Determines how many healthpoints you can have"], |
701 | [0, 2, st_con => "Con", 30], |
454 | [0, 3, st_int => "Int", 30, "<b>Intelligence</b>, your ability to learn and use skills and incantations (both prayers and magic) and determines how much spell points you can have"], |
702 | [0, 3, st_int => "Int", 30], |
455 | [0, 4, st_wis => "Wis", 30, "<b>Wisdom</b>, the ability to learn and use divine magic (prayers). Determines how many grace points you can have"], |
703 | [0, 4, st_wis => "Wis", 30], |
456 | [0, 5, st_pow => "Pow", 30, "<b>Power</b>, your magical potential. Influences the strength of spell effects, and also how much your spell and grace points increase when leveling up"], |
704 | [0, 5, st_pow => "Pow", 30], |
457 | [0, 6, st_cha => "Cha", 30, "<b>Charisma</b>, how well you are received by NPCs. Affects buying and selling prices in shops."], |
705 | [0, 6, st_cha => "Cha", 30], |
458 | |
706 | |
459 | [2, 0, st_wc => "Wc", -120, "<b>Weapon Class</b>, effectiveness of melee/missile attacks. Lower is more potent. Current weapon, level and Str are some things which effect the value of Wc. The value of Wc may range between 25 and -72."], |
707 | [2, 0, st_wc => "Wc", -120], |
460 | [2, 1, st_ac => "Ac", -120, "<b>Armour Class</b>, how protected you are from being hit by any attack. Lower values are better. Ac is based on your race and is modified by the Dex and current armour worn. For characters that cannot wear armour, Ac improves as their level increases."], |
708 | [2, 1, st_ac => "Ac", -120], |
461 | [2, 2, st_dam => "Dam", 120, "<b>Damage</b>, how much damage your melee/missile attack inflicts. Higher values indicate a greater amount of damage will be inflicted with each attack."], |
709 | [2, 2, st_dam => "Dam", 120], |
462 | [2, 3, st_arm => "Arm", 120, "<b>Armour</b>, how much damage (from physical attacks) will be subtracted from successful hits made upon you. This value ranges between 0 to 99%. Current armour worn primarily determines Arm value."], |
710 | [2, 3, st_arm => "Arm", 120], |
463 | [2, 4, st_spd => "Spd", 10.54, "<b>Speed</b>, how fast you can move. The value of speed may range between nearly 0 (\"very slow\") to higher than 5 (\"lightning fast\"). Base speed is determined from the Dex and modified downward proportionally by the amount of weight carried which exceeds the Max Carry limit. The armour worn also sets the upper limit on speed."], |
711 | [2, 4, st_spd => "Spd", 10.54], |
464 | [2, 5, st_wspd => "WSp", 10.54, "<b>Weapon Speed</b>, how many attacks you may make per unit of time (0.120s). Higher values indicate faster attack speed. Current weapon and Dex effect the value of weapon speed."], |
712 | [2, 5, st_wspd => "WSp", 10.54], |
465 | ) { |
713 | ) { |
466 | my ($col, $row, $id, $label, $template, $tooltip) = @$_; |
714 | my ($col, $row, $id, $label, $template) = @$_; |
467 | |
715 | |
468 | $tbl->add ($col , $row, $STATWIDS->{$id} = new CFClient::UI::Label |
716 | $tbl->add ($col , $row, $STATWIDS->{$id} = new CFClient::UI::Label |
469 | font => $FONT_FIXED, can_hover => 1, can_events => 1, valign => 0, align => +1, template => $template, tooltip => $tooltip); |
717 | font => $FONT_FIXED, can_hover => 1, can_events => 1, valign => 0, |
|
|
718 | align => +1, template => $template, tooltip => $CFClient::STAT_TOOLTIP{$label}); |
470 | $tbl->add ($col + 1, $row, $STATWIDS->{"$id\_lbl"} = new CFClient::UI::Label |
719 | $tbl->add ($col + 1, $row, $STATWIDS->{"$id\_lbl"} = new CFClient::UI::Label |
471 | font => $FONT_FIXED, can_hover => 1, can_events => 1, fg => $color2, valign => 0, align => -1, text => $label, tooltip => $tooltip); |
720 | font => $FONT_FIXED, can_hover => 1, can_events => 1, fg => $color2, valign => 0, |
|
|
721 | align => -1, text => $label, tooltip => $CFClient::STAT_TOOLTIP{$label}); |
472 | } |
722 | } |
473 | |
723 | |
474 | $hb->add (my $tbl2 = new CFClient::UI::Table expand => 1); |
724 | $vb->add (my $tbl2 = new CFClient::UI::Table expand => 1); |
475 | |
725 | |
476 | my $row = 0; |
726 | my $row = 0; |
477 | my $col = 0; |
727 | my $col = 0; |
478 | |
728 | |
479 | my %resist_names = ( |
729 | my %resist_names = ( |
|
|
730 | slow => ["Slow", |
480 | slow => "<b>Slow</b> (slows you down when you are hit by the spell. Monsters will have an opportunity to come near you faster and hit you more often.)", |
731 | "<b>Slow</b> (slows you down when you are hit by the spell. Monsters will have an opportunity to come near you faster and hit you more often.)"], |
|
|
732 | holyw => ["Holy Word", |
481 | holyw => "<b>Holy Word</b> (resistance you against getting the fear when someone whose god doesn't like you spells the holy word on you.)", |
733 | "<b>Holy Word</b> (resistance you against getting the fear when someone whose god doesn't like you spells the holy word on you.)"], |
|
|
734 | conf => ["Confusion", |
482 | conf => "<b>Confusion</b> (If you are hit by confusion you will move into random directions, and likely into monsters.)", |
735 | "<b>Confusion</b> (If you are hit by confusion you will move into random directions, and likely into monsters.)"], |
|
|
736 | fire => ["Fire", |
483 | fire => "<b>Fire</b> (just your resistance to fire spells like burning hands, dragonbreath, meteor swarm fire, ...)", |
737 | "<b>Fire</b> (just your resistance to fire spells like burning hands, dragonbreath, meteor swarm fire, ...)"], |
|
|
738 | depl => ["Depletion", |
484 | depl => "<b>Depletion</b> (some monsters and other effects can cause stats depletion)", |
739 | "<b>Depletion</b> (some monsters and other effects can cause stats depletion)"], |
|
|
740 | magic => ["Magic", |
485 | magic => "<b>Magic</b> (resistance to magic spells like magic missile or similar)", |
741 | "<b>Magic</b> (resistance to magic spells like magic missile or similar)"], |
|
|
742 | drain => ["Draining", |
486 | drain => "<b>Draining</b> (some monsters (e.g. vampires) and other effects can steal experience)", |
743 | "<b>Draining</b> (some monsters (e.g. vampires) and other effects can steal experience)"], |
|
|
744 | acid => ["Acid", |
487 | acid => "<b>Acid</b> (resistance to acid, acid hurts pretty much and also corrodes your weapons)", |
745 | "<b>Acid</b> (resistance to acid, acid hurts pretty much and also corrodes your weapons)"], |
|
|
746 | pois => ["Poison", |
488 | pois => "<b>Poison</b> (resistance to getting poisoned)", |
747 | "<b>Poison</b> (resistance to getting poisoned)"], |
|
|
748 | para => ["Paralysation", |
489 | para => "<b>Paralysation</b> (this resistance affects the chance you get paralysed)", |
749 | "<b>Paralysation</b> (this resistance affects the chance you get paralysed)"], |
|
|
750 | deat => ["Death", |
490 | deat => "<b>Death</b> (resistance against death spells)", |
751 | "<b>Death</b> (resistance against death spells)"], |
491 | phys => "<b>Physical</b> (this is the resistance against physical attacks, like when a monster hit you in melee combat)", |
752 | phys => ["Physical", |
|
|
753 | "<b>Physical</b> (this is the resistance against physical attacks, like when a monster hit you in melee combat. The value displayed here is also displayed in the 'Arm' field on the left.)"], |
|
|
754 | blind => ["Blind", |
492 | blind => "<b>Blind</b> (blind resistance affects the chance of a successful blinding attack)", |
755 | "<b>Blind</b> (blind resistance affects the chance of a successful blinding attack)"], |
|
|
756 | fear => ["Fear", |
493 | fear => "<b>Fear</b> (this attack will drive you away from monsters who cast this and hit you successfully, being resistant to this helps a lot when fighting those monsters)", |
757 | "<b>Fear</b> (this attack will drive you away from monsters who cast this and hit you successfully, being resistant to this helps a lot when fighting those monsters)"], |
|
|
758 | tund => ["Turn undead", |
494 | tund => "<b>Turn undead</b> (affects your resistancy to various forms of 'turn undead' spells. Only relevant when you are, in fact, undead...", |
759 | "<b>Turn undead</b> (affects your resistancy to various forms of 'turn undead' spells. Only relevant when you are, in fact, undead..."], |
|
|
760 | elec => ["Electricity", |
495 | elec => "<b>Electricity</b> (resistance against electricity, spells like large lightning, small lightning, ...)", |
761 | "<b>Electricity</b> (resistance against electricity, spells like large lightning, small lightning, ...)"], |
|
|
762 | cold => ["Cold", |
496 | cold => "<b>Cold</b> (this is your resistance against cold spells like icestorm, snowstorm, ...)", |
763 | "<b>Cold</b> (this is your resistance against cold spells like icestorm, snowstorm, ...)"], |
|
|
764 | ghit => ["Ghost hit", |
497 | ghit => "<b>Ghost hit</b> (special attack used by ghosts and ghost-like beings)", |
765 | "<b>Ghost hit</b> (special attack used by ghosts and ghost-like beings)"], |
498 | ); |
766 | ); |
499 | for (qw/slow holyw conf fire depl magic |
767 | for (qw/slow holyw conf fire depl magic |
500 | drain acid pois para deat phys |
768 | drain acid pois para deat phys |
501 | blind fear tund elec cold ghit/) |
769 | blind fear tund elec cold ghit/) |
502 | { |
770 | { |
… | |
… | |
507 | template => "-100%", |
775 | template => "-100%", |
508 | align => +1, |
776 | align => +1, |
509 | valign => 0, |
777 | valign => 0, |
510 | can_events => 1, |
778 | can_events => 1, |
511 | can_hover => 1, |
779 | can_hover => 1, |
512 | tooltip => $resist_names{$_}, |
780 | tooltip => $resist_names{$_}->[1], |
513 | ); |
781 | ); |
514 | $tbl2->add ($col + 1, $row, new CFClient::UI::Image |
782 | $tbl2->add ($col + 1, $row, new CFClient::UI::Image |
515 | font => $FONT_FIXED, |
783 | font => $FONT_FIXED, |
516 | can_hover => 1, |
784 | can_hover => 1, |
517 | can_events => 1, |
785 | can_events => 1, |
518 | image => "ui/resist/resist_$_.png", |
786 | path => "ui/resist/resist_$_.png", |
519 | tooltip => $resist_names{$_}, |
787 | tooltip => $resist_names{$_}->[1], |
|
|
788 | ); |
|
|
789 | $tbl2->add ($col + 2, $row, new CFClient::UI::Label |
|
|
790 | text => $resist_names{$_}->[0], |
|
|
791 | font => $FONT_FIXED, |
|
|
792 | can_hover => 1, |
|
|
793 | can_events => 1, |
|
|
794 | tooltip => $resist_names{$_}->[1], |
520 | ); |
795 | ); |
521 | |
796 | |
522 | $row++; |
797 | $row++; |
523 | if ($row % 6 == 0) { |
798 | if ($row % 6 == 0) { |
524 | $col += 2; |
799 | $col += 3; |
525 | $row = 0; |
800 | $row = 0; |
526 | } |
801 | } |
527 | } |
802 | } |
528 | |
803 | |
529 | &set_stats_window_fontsize; |
|
|
530 | update_stats_window ({}); |
804 | #update_stats_window ({}); |
531 | |
805 | |
|
|
806 | $r |
|
|
807 | } |
|
|
808 | |
|
|
809 | sub skill_window { |
|
|
810 | my $sw = new CFClient::UI::ScrolledWindow (expand => 1); |
|
|
811 | $sw->add ($STATWIDS->{skill_tbl} = new CFClient::UI::Table expand => 1, col_expand => [0, 0, 1, 0, 0, 1]); |
532 | $tgw |
812 | $sw |
533 | } |
813 | } |
534 | |
814 | |
535 | sub formsep { |
815 | sub formsep($) { |
536 | reverse join ",", grep length, split /(...)/, reverse $_[0] * 1 |
816 | scalar reverse join ",", unpack "(A3)*", reverse $_[0] * 1 |
537 | } |
|
|
538 | |
|
|
539 | sub update_stats_window { |
|
|
540 | my ($stats) = @_; |
|
|
541 | |
|
|
542 | # i love text protocols!!! |
|
|
543 | my $hp = $stats->{Crossfire::Protocol::CS_STAT_HP} * 1; |
|
|
544 | my $hp_m = $stats->{Crossfire::Protocol::CS_STAT_MAXHP} * 1; |
|
|
545 | my $sp = $stats->{Crossfire::Protocol::CS_STAT_SP} * 1; |
|
|
546 | my $sp_m = $stats->{Crossfire::Protocol::CS_STAT_MAXSP} * 1; |
|
|
547 | my $fo = $stats->{Crossfire::Protocol::CS_STAT_FOOD} * 1; |
|
|
548 | my $fo_m = 999; |
|
|
549 | my $gr = $stats->{Crossfire::Protocol::CS_STAT_GRACE} * 1; |
|
|
550 | my $gr_m = $stats->{Crossfire::Protocol::CS_STAT_MAXGRACE} * 1; |
|
|
551 | |
|
|
552 | $GAUGES->{hp} ->set_value ($hp, $hp_m); |
|
|
553 | $GAUGES->{mana} ->set_value ($sp, $sp_m); |
|
|
554 | $GAUGES->{food} ->set_value ($fo, $fo_m); |
|
|
555 | $GAUGES->{grace} ->set_value ($gr, $gr_m); |
|
|
556 | $GAUGES->{exp} ->set_text ("Exp: " . (formsep $stats->{Crossfire::Protocol::CS_STAT_EXP64}) |
|
|
557 | . " (lvl " . ($stats->{Crossfire::Protocol::CS_STAT_LEVEL} * 1) . ")"); |
|
|
558 | my $rng = $stats->{Crossfire::Protocol::CS_STAT_RANGE}; |
|
|
559 | $rng =~ s/^Range: //; # thank you so much dear server |
|
|
560 | $GAUGES->{range} ->set_text ("Rng: " . $rng); |
|
|
561 | my $title = $stats->{Crossfire::Protocol::CS_STAT_TITLE}; |
|
|
562 | $title =~ s/^Player: //; |
|
|
563 | $STATWIDS->{title} ->set_text ("Title: " . $title); |
|
|
564 | |
|
|
565 | $STATWIDS->{st_str} ->set_text (sprintf "%d", $stats->{5}); |
|
|
566 | $STATWIDS->{st_dex} ->set_text (sprintf "%d", $stats->{8}); |
|
|
567 | $STATWIDS->{st_con} ->set_text (sprintf "%d", $stats->{9}); |
|
|
568 | $STATWIDS->{st_int} ->set_text (sprintf "%d", $stats->{6}); |
|
|
569 | $STATWIDS->{st_wis} ->set_text (sprintf "%d", $stats->{7}); |
|
|
570 | $STATWIDS->{st_pow} ->set_text (sprintf "%d", $stats->{22}); |
|
|
571 | $STATWIDS->{st_cha} ->set_text (sprintf "%d", $stats->{10}); |
|
|
572 | $STATWIDS->{st_wc} ->set_text (sprintf "%d", $stats->{13}); |
|
|
573 | $STATWIDS->{st_ac} ->set_text (sprintf "%d", $stats->{14}); |
|
|
574 | $STATWIDS->{st_dam} ->set_text (sprintf "%d", $stats->{15}); |
|
|
575 | $STATWIDS->{st_arm} ->set_text (sprintf "%d", $stats->{16}); |
|
|
576 | $STATWIDS->{st_spd} ->set_text (sprintf "%.1f", $stats->{Crossfire::Protocol::CS_STAT_SPEED}); |
|
|
577 | $STATWIDS->{st_wspd}->set_text (sprintf "%.1f", $stats->{Crossfire::Protocol::CS_STAT_WEAP_SP}); |
|
|
578 | |
|
|
579 | $STATWIDS->{m_weight}->set_text (sprintf "Max weight: %.1fkg", $stats->{Crossfire::Protocol::CS_STAT_WEIGHT_LIM} / 1000); |
|
|
580 | |
|
|
581 | my %tbl = ( |
|
|
582 | phys => 100, |
|
|
583 | magic => 101, |
|
|
584 | fire => 102, |
|
|
585 | elec => 103, |
|
|
586 | cold => 104, |
|
|
587 | conf => 105, |
|
|
588 | acid => 106, |
|
|
589 | drain => 107, |
|
|
590 | ghit => 108, |
|
|
591 | pois => 109, |
|
|
592 | slow => 110, |
|
|
593 | para => 111, |
|
|
594 | tund => 112, |
|
|
595 | fear => 113, |
|
|
596 | depl => 113, |
|
|
597 | deat => 115, |
|
|
598 | holyw => 116, |
|
|
599 | blind => 117 |
|
|
600 | ); |
|
|
601 | |
|
|
602 | for (keys %tbl) { |
|
|
603 | $STATWIDS->{"res_$_"}->set_text (sprintf "%d%", $stats->{$tbl{$_}}); |
|
|
604 | } |
|
|
605 | |
|
|
606 | } |
|
|
607 | |
|
|
608 | sub metaserver_dialog { |
|
|
609 | my $dialog = new CFClient::UI::FancyFrame |
|
|
610 | title => "Server List", |
|
|
611 | child => (my $vbox = new CFClient::UI::VBox); |
|
|
612 | |
|
|
613 | $vbox->add ($dialog->{table} = new CFClient::UI::Table); |
|
|
614 | |
|
|
615 | $dialog |
|
|
616 | } |
817 | } |
617 | |
818 | |
618 | my $METASERVER_ATIME; |
819 | my $METASERVER_ATIME; |
619 | |
820 | |
620 | sub update_metaserver { |
821 | sub update_metaserver { |
621 | my ($HOST) = @_; |
822 | my ($metaserver_dialog) = @_; |
|
|
823 | |
|
|
824 | $METASERVER = $metaserver_dialog |
|
|
825 | if defined $metaserver_dialog; |
622 | |
826 | |
623 | return if $METASERVER_ATIME > time; |
827 | return if $METASERVER_ATIME > time; |
624 | $METASERVER_ATIME = time + 60; |
828 | $METASERVER_ATIME = time + 60; |
625 | |
829 | |
626 | my $table = $METASERVER->{table}; |
830 | my $table = $METASERVER->{table}; |
… | |
… | |
648 | |
852 | |
649 | utf8::decode $buf if utf8::valid $buf; |
853 | utf8::decode $buf if utf8::valid $buf; |
650 | |
854 | |
651 | $table->clear; |
855 | $table->clear; |
652 | |
856 | |
|
|
857 | my @tip = ( |
|
|
858 | "The current number of users logged in on the server.", |
|
|
859 | "The hostname of the server.", |
|
|
860 | "The time this server has been running without being restarted.", |
|
|
861 | "The server software version - a '+' indicates a Crossfire+ server.", |
|
|
862 | "Short information about this server provided by its admins.", |
|
|
863 | ); |
653 | my @col = qw(Use #Users Host Uptime Version Description); |
864 | my @col = qw(#Users Host Uptime Version Description); |
654 | $table->add ($_, 0, new CFClient::UI::Label align => 0, fg => [1, 1, 0], text => $col[$_]) |
865 | $table->add ($_, 0, new CFClient::UI::Label |
|
|
866 | can_hover => 1, can_events => 1, |
|
|
867 | align => 0, fg => [1, 1, 0], |
|
|
868 | text => $col[$_], tooltip => $tip[$_]) |
655 | for 0 .. $#col; |
869 | for 0 .. $#col; |
656 | |
870 | |
657 | my @align = qw(1 0 1 1 -1); |
871 | my @align = qw(1 0 1 1 -1); |
658 | |
872 | |
659 | my $y = 0; |
873 | my $y = 0; |
660 | for my $m (sort { $b->[3] <=> $a->[3] } map [split /\|/], split /\015?\012/, $buf) { |
874 | for my $m (sort { $b->[3] <=> $a->[3] } map [split /\|/], split /\015?\012/, $buf) { |
… | |
… | |
677 | |
891 | |
678 | $m = [$users, $host, $uptime, $version, $desc]; |
892 | $m = [$users, $host, $uptime, $version, $desc]; |
679 | |
893 | |
680 | $y++; |
894 | $y++; |
681 | |
895 | |
682 | $table->add (0, $y, new CFClient::UI::VBox children => [ |
896 | $table->add (scalar @$m, $y, new CFClient::UI::VBox children => [ |
683 | (new CFClient::UI::Button text => "Use", connect_activate => sub { |
897 | (new CFClient::UI::Button |
|
|
898 | text => "Use", |
|
|
899 | tooltip => "Put this server into the <b>Host:Port</b> field", |
|
|
900 | on_activate => sub { |
684 | $HOST->set_text ($CFG->{host} = $host); |
901 | $HOST_ENTRY->set_text ($CFG->{profile}{default}{host} = $host); |
|
|
902 | $METASERVER->hide; |
|
|
903 | 0 |
|
|
904 | }, |
685 | }), |
905 | ), |
686 | (new CFClient::UI::Empty expand => 1), |
906 | (new CFClient::UI::Empty expand => 1), |
687 | ]); |
907 | ]); |
688 | |
908 | |
689 | $table->add ($_ + 1, $y, new CFClient::UI::Label |
909 | $table->add ($_, $y, new CFClient::UI::Label |
690 | ellipsise => 0, align => $align[$_], text => $m->[$_], fontsize => 0.8) |
910 | ellipsise => 0, |
|
|
911 | align => $align[$_], |
|
|
912 | text => $m->[$_], |
|
|
913 | tooltip => $tip[$_], |
|
|
914 | can_hover => 1, |
|
|
915 | can_events => 1, |
|
|
916 | fontsize => 0.8) |
691 | for 0 .. $#$m; |
917 | for 0 .. $#$m; |
692 | } |
918 | } |
693 | } |
919 | } |
694 | }); |
920 | }); |
695 | } |
921 | } |
696 | |
922 | |
|
|
923 | sub metaserver_dialog { |
|
|
924 | my $vbox = new CFClient::UI::VBox; |
|
|
925 | my $table = new CFClient::UI::Table; |
|
|
926 | $vbox->add (new CFClient::UI::ScrolledWindow expand => 1, child => $table); |
|
|
927 | |
|
|
928 | my $dialog = new CFClient::UI::FancyFrame |
|
|
929 | title => "Server List", |
|
|
930 | name => 'metaserver_dialog', |
|
|
931 | x => 'center', |
|
|
932 | y => 'center', |
|
|
933 | z => 3, |
|
|
934 | force_h => $::HEIGHT * 0.4, |
|
|
935 | child => $vbox, |
|
|
936 | has_close_button => 1, |
|
|
937 | table => $table, |
|
|
938 | on_visibility_change => sub { |
|
|
939 | update_metaserver ($_[0]) if $_[1]; |
|
|
940 | 0 |
|
|
941 | }, |
|
|
942 | ; |
|
|
943 | |
|
|
944 | $dialog |
|
|
945 | } |
|
|
946 | |
697 | sub server_setup { |
947 | sub server_setup { |
698 | my $dialog = new CFClient::UI::FancyFrame |
|
|
699 | title => "Server Setup", |
|
|
700 | child => (my $vbox = new CFClient::UI::VBox); |
948 | my $vbox = new CFClient::UI::VBox; |
701 | |
949 | |
702 | $vbox->add (my $table = new CFClient::UI::Table expand => 1, col_expand => [0, 1]); |
950 | $vbox->add (my $table = new CFClient::UI::Table expand => 1, col_expand => [0, 1]); |
703 | $table->add (0, 2, new CFClient::UI::Label valign => 0, align => 1, text => "Host:Port"); |
951 | $table->add (0, 2, new CFClient::UI::Label valign => 0, align => 1, text => "Host:Port"); |
704 | |
952 | |
705 | { |
953 | { |
706 | $table->add (1, 2, my $vbox = new CFClient::UI::VBox); |
954 | $table->add (1, 2, my $vbox = new CFClient::UI::VBox); |
707 | |
955 | |
708 | $vbox->add ( |
956 | $vbox->add ( |
709 | my $HOST = new CFClient::UI::Entry |
957 | $HOST_ENTRY = new CFClient::UI::Entry |
710 | expand => 1, |
958 | expand => 1, |
711 | text => $CFG->{host}, |
959 | text => $CFG->{profile}{default}{host}, |
712 | tooltip => "The hostname or ip address of the Crossfire(+) server to connect to", |
960 | tooltip => "The hostname or ip address of the Crossfire(+) server to connect to", |
713 | connect_changed => sub { |
961 | on_changed => sub { |
714 | my ($self, $value) = @_; |
962 | my ($self, $value) = @_; |
715 | $CFG->{host} = $value; |
963 | $CFG->{profile}{default}{host} = $value; |
|
|
964 | 0 |
716 | } |
965 | } |
717 | ); |
966 | ); |
718 | |
967 | |
719 | $METASERVER = metaserver_dialog; |
|
|
720 | |
|
|
721 | $vbox->add (new CFClient::UI::Flopper |
968 | $vbox->add (new CFClient::UI::Button |
722 | expand => 1, |
969 | expand => 1, |
723 | text => "Server List", |
970 | text => "Server List", |
724 | other => $METASERVER, |
971 | other => $METASERVER, |
725 | tooltip => "Show a list of available crossfire servers", |
972 | tooltip => "Show a list of available crossfire servers", |
726 | connect_open => sub { |
973 | on_activate => sub { $METASERVER->toggle_visibility; 0 }, |
727 | update_metaserver $HOST; |
974 | on_visibility_change => sub { $METASERVER->hide unless $_[1]; 0 }, |
728 | } |
|
|
729 | ); |
975 | ); |
730 | } |
976 | } |
731 | |
977 | |
732 | $table->add (0, 4, new CFClient::UI::Label valign => 0, align => 1, text => "Username"); |
978 | $table->add (0, 4, new CFClient::UI::Label valign => 0, align => 1, text => "Username"); |
733 | $table->add (1, 4, new CFClient::UI::Entry |
979 | $table->add (1, 4, new CFClient::UI::Entry |
734 | text => $CFG->{user}, |
980 | text => $CFG->{profile}{default}{user}, |
735 | tooltip => "The name of your character on the server", |
981 | tooltip => "The name of your character on the server", |
736 | connect_changed => sub { |
982 | on_changed => sub { my ($self, $value) = @_; $CFG->{profile}{default}{user} = $value } |
737 | my ($self, $value) = @_; |
|
|
738 | $CFG->{user} = $value; |
|
|
739 | } |
|
|
740 | ); |
983 | ); |
741 | |
984 | |
742 | $table->add (0, 5, new CFClient::UI::Label valign => 0, align => 1, text => "Password"); |
985 | $table->add (0, 5, new CFClient::UI::Label valign => 0, align => 1, text => "Password"); |
743 | $table->add (1, 5, new CFClient::UI::Entry |
986 | $table->add (1, 5, new CFClient::UI::Entry |
744 | text => $CFG->{password}, |
987 | text => $CFG->{profile}{default}{password}, |
745 | hidden => 1, |
988 | hidden => 1, |
746 | tooltip => "The password for your character", |
989 | tooltip => "The password for your character", |
747 | connect_changed => sub { |
990 | on_changed => sub { my ($self, $value) = @_; $CFG->{profile}{default}{password} = $value } |
748 | my ($self, $value) = @_; |
|
|
749 | $CFG->{password} = $value; |
|
|
750 | } |
|
|
751 | ); |
991 | ); |
752 | |
992 | |
753 | $table->add (0, 7, new CFClient::UI::Label valign => 0, align => 1, text => "Map Size"); |
993 | $table->add (0, 7, new CFClient::UI::Label valign => 0, align => 1, text => "Map Size"); |
754 | $table->add (1, 7, new CFClient::UI::Slider |
994 | $table->add (1, 7, new CFClient::UI::Slider |
755 | req_w => 100, |
995 | force_w => 100, |
756 | range => [$CFG->{mapsize}, 10, 100, 0, 1], |
996 | range => [$CFG->{mapsize}, 10, 100, 0, 1], |
757 | tooltip => "This is the size of the portion of the map update the server sends you. " |
997 | tooltip => "This is the size of the portion of the map update the server sends you. " |
758 | . "If you set this to a high value you will be able to see further, " |
998 | . "If you set this to a high value you will be able to see further, " |
759 | . "but you also increase bandwidth requirements and latency. " |
999 | . "but you also increase bandwidth requirements and latency. " |
760 | . "This option is only used once at log-in.", |
1000 | . "This option is only used once at log-in.", |
761 | connect_changed => sub { |
1001 | on_changed => sub { my ($self, $value) = @_; $CFG->{mapsize} = $self->{range}[0] = $value = int $value; 0 }, |
762 | my ($self, $value) = @_; |
|
|
763 | |
|
|
764 | $CFG->{mapsize} = $self->{range}[0] = $value = int $value; |
|
|
765 | }, |
|
|
766 | ); |
1002 | ); |
767 | |
1003 | |
768 | $table->add (0, 8, new CFClient::UI::Label valign => 0, align => 1, text => "Face Prefetch"); |
1004 | $table->add (0, 8, new CFClient::UI::Label valign => 0, align => 1, text => "Face Prefetch"); |
769 | $table->add (1, 8, new CFClient::UI::CheckBox |
1005 | $table->add (1, 8, new CFClient::UI::CheckBox |
770 | state => $CFG->{face_prefetch}, |
1006 | state => $CFG->{face_prefetch}, |
… | |
… | |
773 | . "This might increase or create lag, but increases the chances " |
1009 | . "This might increase or create lag, but increases the chances " |
774 | . "of faces being ready for display when you encounter them. " |
1010 | . "of faces being ready for display when you encounter them. " |
775 | . "It also uses up server bandwidth on every connect, " |
1011 | . "It also uses up server bandwidth on every connect, " |
776 | . "so only set it if you really need to prefetch images. " |
1012 | . "so only set it if you really need to prefetch images. " |
777 | . "This option can be set and unset any time.", |
1013 | . "This option can be set and unset any time.", |
778 | connect_changed => sub { $CFG->{face_prefetch} = $_[1] }, |
1014 | on_changed => sub { $CFG->{face_prefetch} = $_[1]; 0 }, |
779 | ); |
1015 | ); |
780 | |
1016 | |
781 | $table->add (0, 9, new CFClient::UI::Label valign => 0, align => 1, text => "Output-Count"); |
1017 | $table->add (0, 9, new CFClient::UI::Label valign => 0, align => 1, text => "Output-Count"); |
782 | $table->add (1, 9, new CFClient::UI::Entry |
1018 | $table->add (1, 9, new CFClient::UI::Entry |
783 | text => $CFG->{output_count}, |
1019 | text => $CFG->{output_count}, |
784 | tooltip => "Should be set to 1 unless you know what you are doing. This option is only used once at log-in.", |
1020 | tooltip => "Should be set to 1 unless you know what you are doing. This option is only used once at log-in.", |
785 | connect_changed => sub { $CFG->{output_count} = $_[1] }, |
1021 | on_changed => sub { $CFG->{output_count} = $_[1]; 0 }, |
786 | ); |
1022 | ); |
787 | |
1023 | |
788 | $table->add (0, 10, new CFClient::UI::Label valign => 0, align => 1, text => "Output-Sync"); |
1024 | $table->add (0, 10, new CFClient::UI::Label valign => 0, align => 1, text => "Output-Sync"); |
789 | $table->add (1, 10, new CFClient::UI::Entry |
1025 | $table->add (1, 10, new CFClient::UI::Entry |
790 | text => $CFG->{output_sync}, |
1026 | text => $CFG->{output_sync}, |
791 | tooltip => "Should be set to 1 unless you know what you are doing. This option is only used once at log-in.", |
1027 | tooltip => "Should be set to 1 unless you know what you are doing. This option is only used once at log-in.", |
792 | connect_changed => sub { $CFG->{output_sync} = $_[1] }, |
1028 | on_changed => sub { $CFG->{output_sync} = $_[1]; 0 }, |
793 | ); |
1029 | ); |
794 | |
1030 | |
795 | $table->add (1, 11, $LOGIN_BUTTON = new CFClient::UI::Button |
1031 | $table->add (1, 11, $LOGIN_BUTTON = new CFClient::UI::Button |
796 | expand => 1, |
1032 | expand => 1, |
797 | align => 0, |
1033 | align => 0, |
798 | text => "Login", |
1034 | text => "Login", |
799 | connect_activate => sub { |
1035 | on_activate => sub { |
800 | $CONN ? stop_game |
1036 | $CONN ? stop_game |
801 | : start_game; |
1037 | : start_game; |
|
|
1038 | 0 |
802 | }, |
1039 | }, |
803 | ); |
1040 | ); |
804 | |
1041 | |
805 | $dialog |
1042 | $table->add (0, 12, new CFClient::UI::Label valign => 0, align => 1, text => "Chat Command"); |
|
|
1043 | $table->add (1, 12, my $saycmd = new CFClient::UI::Entry |
|
|
1044 | text => $CFG->{say_command}, |
|
|
1045 | tooltip => "This is the command that will be used if you write a line in the message window entry or press <b>\"</b> in the map window. " |
|
|
1046 | . "Usually you want to enter something like 'say' or 'shout' or 'gsay' here. " |
|
|
1047 | . "But you could also set it to <b>tell <i>playername</i></b> to only chat with that user.", |
|
|
1048 | on_changed => sub { |
|
|
1049 | my ($self, $value) = @_; |
|
|
1050 | $CFG->{say_command} = $value; |
|
|
1051 | 0 |
|
|
1052 | } |
|
|
1053 | ); |
|
|
1054 | |
|
|
1055 | $vbox->add (new CFClient::UI::Label |
|
|
1056 | text => "Server Info", |
|
|
1057 | fontsize => 1.2, |
|
|
1058 | padding_y => 8, |
|
|
1059 | fg => [1, 1, 0, 1], |
|
|
1060 | ); |
|
|
1061 | |
|
|
1062 | $vbox->add ($SERVER_INFO = new CFClient::UI::Label ellipsise => 0); |
|
|
1063 | |
|
|
1064 | $vbox |
806 | } |
1065 | } |
807 | |
1066 | |
808 | sub message_window { |
1067 | sub message_window { |
809 | my $window = new CFClient::UI::FancyFrame |
1068 | my $window = new CFClient::UI::FancyFrame |
|
|
1069 | name => "message_window", |
810 | title => "Messages", |
1070 | title => "Messages", |
811 | border_bg => [1, 1, 1, 1], |
1071 | border_bg => [1, 1, 1, 1], |
812 | bg => [0, 0, 0, 0.75], |
1072 | x => "max", |
813 | user_w => int $::WIDTH / 3, |
1073 | y => 0, |
|
|
1074 | force_w => $::WIDTH * 0.4, |
814 | user_h => int $::HEIGHT / 5, |
1075 | force_h => $::HEIGHT * 0.5, |
815 | child => (my $vbox = new CFClient::UI::VBox); |
1076 | child => (my $vbox = new CFClient::UI::VBox), |
|
|
1077 | has_close_button => 1; |
816 | |
1078 | |
817 | $vbox->add ($LOGVIEW); |
1079 | $vbox->add ($LOGVIEW); |
818 | |
1080 | |
819 | $vbox->add (my $input = new CFClient::UI::Entry |
1081 | $vbox->add (my $input = new CFClient::UI::Entry |
820 | tooltip => "<b>Chat Box</b>. If you enter a text and press return/enter here, the current <i>communication command</i> " |
1082 | tooltip => "<b>Chat Box</b>. If you enter a text and press return/enter here, the current <i>communication command</i> " |
821 | . "from the client setup will be prepended (e.g. <b>shout</b>, <b>chat</b>...). " |
1083 | . "from the client setup will be prepended (e.g. <b>shout</b>, <b>chat</b>...). " |
822 | . "If you prepend a slash (/), you will submit a command instead (similar to IRC). " |
1084 | . "If you prepend a slash (/), you will submit a command instead (similar to IRC). " |
823 | . "A better way to submit commands (and the occasional chat command) is often the map command completer.", |
1085 | . "A better way to submit commands (and the occasional chat command) is often the map command completer.", |
824 | connect_focus_in => sub { |
1086 | on_focus_in => sub { |
825 | my ($input, $prev_focus) = @_; |
1087 | my ($input, $prev_focus) = @_; |
826 | |
1088 | |
827 | delete $input->{refocus_map}; |
1089 | delete $input->{refocus_map}; |
828 | |
1090 | |
829 | if ($prev_focus == $MAPWIDGET && $input->{auto_activated}) { |
1091 | if ($prev_focus == $MAPWIDGET && $input->{auto_activated}) { |
830 | $input->{refocus_map} = 1; |
1092 | $input->{refocus_map} = 1; |
831 | } |
1093 | } |
832 | delete $input->{auto_activated}; |
1094 | delete $input->{auto_activated}; |
|
|
1095 | |
|
|
1096 | 0 |
833 | }, |
1097 | }, |
834 | connect_activate => sub { |
1098 | on_activate => sub { |
835 | my ($input, $text) = @_; |
1099 | my ($input, $text) = @_; |
836 | $input->set_text (''); |
1100 | $input->set_text (''); |
837 | |
1101 | |
838 | if ($text =~ /^\/(.*)/) { |
1102 | if ($text =~ /^\/(.*)/) { |
839 | $::CONN->user_send ($1); |
1103 | $::CONN->user_send ($1); |
… | |
… | |
843 | } |
1107 | } |
844 | if ($input->{refocus_map}) { |
1108 | if ($input->{refocus_map}) { |
845 | delete $input->{refocus_map}; |
1109 | delete $input->{refocus_map}; |
846 | $MAPWIDGET->focus_in |
1110 | $MAPWIDGET->focus_in |
847 | } |
1111 | } |
|
|
1112 | |
|
|
1113 | 0 |
848 | }, |
1114 | }, |
849 | connect_escape => sub { |
1115 | on_escape => sub { |
850 | $MAPWIDGET->focus_in |
1116 | $MAPWIDGET->grab_focus; |
|
|
1117 | |
|
|
1118 | 0 |
851 | }, |
1119 | }, |
852 | ); |
1120 | ); |
853 | |
1121 | |
854 | $CONSOLE = { |
1122 | $CONSOLE = { |
855 | window => $window, |
1123 | window => $window, |
856 | input => $input |
1124 | input => $input, |
857 | }; |
1125 | }; |
858 | |
1126 | |
859 | $window |
1127 | $window |
860 | } |
1128 | } |
861 | |
1129 | |
862 | sub open_quit_dialog { |
1130 | sub open_quit_dialog { |
863 | unless ($QUIT_DIALOG) { |
1131 | unless ($QUIT_DIALOG) { |
864 | |
|
|
865 | $QUIT_DIALOG = new CFClient::UI::FancyFrame title => "Really Quit?"; |
1132 | $QUIT_DIALOG = new CFClient::UI::FancyFrame |
|
|
1133 | x => "center", |
|
|
1134 | y => "center", |
|
|
1135 | z => 50, |
|
|
1136 | title => "Really Quit?", |
|
|
1137 | on_key_down => sub { |
|
|
1138 | my ($dialog, $ev) = @_; |
|
|
1139 | $ev->{sym} == 27 and $dialog->hide; |
|
|
1140 | } |
|
|
1141 | ; |
866 | |
1142 | |
867 | $QUIT_DIALOG->add (my $vb = new CFClient::UI::VBox expand => 1); |
1143 | $QUIT_DIALOG->add (my $vb = new CFClient::UI::VBox expand => 1); |
868 | |
1144 | |
869 | $vb->add (new CFClient::UI::Label |
1145 | $vb->add (new CFClient::UI::Label |
870 | text => "You should find a savebed and apply it first!", |
1146 | text => "You should find a savebed and apply it first!", |
… | |
… | |
873 | ); |
1149 | ); |
874 | $vb->add (my $hb = new CFClient::UI::HBox expand => 1); |
1150 | $vb->add (my $hb = new CFClient::UI::HBox expand => 1); |
875 | $hb->add (new CFClient::UI::Button |
1151 | $hb->add (new CFClient::UI::Button |
876 | text => "Ok", |
1152 | text => "Ok", |
877 | expand => 1, |
1153 | expand => 1, |
878 | connect_activate => sub { $QUIT_DIALOG->hide }, |
1154 | on_activate => sub { $QUIT_DIALOG->hide; 0 }, |
879 | ); |
1155 | ); |
880 | $hb->add (new CFClient::UI::Button |
1156 | $hb->add (new CFClient::UI::Button |
881 | text => "Quit anyway", |
1157 | text => "Quit anyway", |
882 | expand => 1, |
1158 | expand => 1, |
883 | connect_activate => sub { exit }, |
1159 | on_activate => sub { exit }, |
884 | ); |
|
|
885 | |
|
|
886 | $QUIT_DIALOG->show_centered; |
|
|
887 | } else { |
|
|
888 | $QUIT_DIALOG->show_centered; |
|
|
889 | } |
|
|
890 | } |
|
|
891 | |
|
|
892 | sub make_inventory_window { |
|
|
893 | my $invwin = new CFClient::UI::FancyFrame |
|
|
894 | user_w => $WIDTH * (7/8), user_h => $HEIGHT * (7/8), title => "Inventory"; |
|
|
895 | |
|
|
896 | $invwin->add (my $hb = new CFClient::UI::HBox expand => 1); |
|
|
897 | |
|
|
898 | $hb->add (my $vb1 = new CFClient::UI::VBox expand => 1); |
|
|
899 | $vb1->add (my $lbl = new CFClient::UI::Label xalign => 0.5); |
|
|
900 | $lbl->set_text ("Player"); |
|
|
901 | $vb1->add ($INV = new CFClient::UI::Inventory expand => 1); |
|
|
902 | |
|
|
903 | $hb->add (my $vb2 = new CFClient::UI::VBox expand => 1); |
|
|
904 | $vb2->add ($INVR_LBL = new CFClient::UI::Label xalign => 0.5); |
|
|
905 | $INVR_LBL->set_text ("Floor"); |
|
|
906 | $vb2->add ($INVR = new CFClient::UI::Inventory expand => 1); |
|
|
907 | |
|
|
908 | $invwin |
|
|
909 | } |
|
|
910 | |
|
|
911 | sub make_help_window { |
|
|
912 | my $win = new CFClient::UI::FancyFrame |
|
|
913 | user_w => $WIDTH * (7/8), user_h => $HEIGHT * (7/8), title => "Documentation"; |
|
|
914 | |
|
|
915 | $win->add (my $vbox = new CFClient::UI::VBox); |
|
|
916 | |
|
|
917 | $vbox->add (my $buttons = new CFClient::UI::HBox); |
|
|
918 | $vbox->add (my $viewer = new CFClient::UI::TextView expand => 1, fontsize => 0.8); |
|
|
919 | |
|
|
920 | for ( |
|
|
921 | [intro => "Introduction"], |
|
|
922 | [manual => "Manual"], |
|
|
923 | [command_help => "Commands"], |
|
|
924 | [skill_help => "Skills"], |
|
|
925 | ) { |
|
|
926 | my ($pod, $label) = @$_; |
|
|
927 | |
|
|
928 | $buttons->add (new CFClient::UI::Button |
|
|
929 | text => $label, |
|
|
930 | connect_activate => sub { |
|
|
931 | my $parser = new Pod::POM; |
|
|
932 | my $pom = $parser->parse_file (CFClient::find_rcfile "pod/$pod.pod"); |
|
|
933 | |
|
|
934 | $viewer->clear; |
|
|
935 | |
|
|
936 | $viewer->add_paragraph ([1, 1, 1, 1], $_->[1], $_->[0]) |
|
|
937 | for @{ CFClient::pod_to_pango_list $pom }; |
|
|
938 | |
|
|
939 | $viewer->set_offset (0); |
|
|
940 | }, |
|
|
941 | ); |
1160 | ); |
942 | } |
1161 | } |
943 | |
1162 | |
944 | $viewer->add_paragraph ([1, 1, 0, 1], "<big>Use one of the buttons above to display a document.</big>"); |
1163 | $QUIT_DIALOG->show; |
|
|
1164 | $QUIT_DIALOG->grab_focus; |
|
|
1165 | } |
|
|
1166 | |
|
|
1167 | sub autopickup_setup { |
|
|
1168 | my $table = new CFClient::UI::Table; |
|
|
1169 | |
|
|
1170 | for ( |
|
|
1171 | ["General", 0, 0, |
|
|
1172 | ["Enable autopickup" => PICKUP_NEWMODE, \$PICKUP_ENABLE], |
|
|
1173 | ["Inhibit autopickup" => PICKUP_INHIBIT], |
|
|
1174 | ["Stop before pickup" => PICKUP_STOP], |
|
|
1175 | ["Debug autopickup" => PICKUP_DEBUG], |
|
|
1176 | ], |
|
|
1177 | ["Weapons", 0, 6, |
|
|
1178 | ["All weapons" => PICKUP_ALLWEAPON], |
|
|
1179 | ["Missile weapons" => PICKUP_MISSILEWEAPON], |
|
|
1180 | ["Bows" => PICKUP_BOW], |
|
|
1181 | ["Arrows" => PICKUP_ARROW], |
|
|
1182 | ], |
|
|
1183 | ["Armour", 0, 12, |
|
|
1184 | ["Helmets" => PICKUP_HELMET], |
|
|
1185 | ["Shields" => PICKUP_SHIELD], |
|
|
1186 | ["Body Armour" => PICKUP_ARMOUR], |
|
|
1187 | ["Boots" => PICKUP_BOOTS], |
|
|
1188 | ["Gloves" => PICKUP_GLOVES], |
|
|
1189 | ["Cloaks" => PICKUP_CLOAK], |
|
|
1190 | ], |
|
|
1191 | |
|
|
1192 | ["Readables", 2, 2, |
|
|
1193 | ["Spellbooks" => PICKUP_SPELLBOOK], |
|
|
1194 | ["Skillscrolls" => PICKUP_SKILLSCROLL], |
|
|
1195 | ["Normal Books/Scrolls" => PICKUP_READABLES], |
|
|
1196 | ], |
|
|
1197 | ["Misc", 2, 7, |
|
|
1198 | ["Food" => PICKUP_FOOD], |
|
|
1199 | ["Drinks" => PICKUP_DRINK], |
|
|
1200 | ["Valuables (Money, Gems)" => PICKUP_VALUABLES], |
|
|
1201 | ["Keys" => PICKUP_KEY], |
|
|
1202 | ["Magical Items" => PICKUP_MAGICAL], |
|
|
1203 | ["Potions" => PICKUP_POTION], |
|
|
1204 | ["Magic Devices" => PICKUP_MAGIC_DEVICE], |
|
|
1205 | ["Ignore cursed" => PICKUP_NOT_CURSED], |
|
|
1206 | ["Jewelery" => PICKUP_JEWELS], |
|
|
1207 | ], |
|
|
1208 | ["Weight/Value ratio", 2, 17] |
|
|
1209 | ) |
|
|
1210 | { |
|
|
1211 | my ($title, $x, $y, @bits) = @$_; |
|
|
1212 | $table->add ($x, $y, new CFClient::UI::Label text => $title, align => 1, fg => [1, 1, 0]); |
|
|
1213 | |
|
|
1214 | for (@bits) { |
|
|
1215 | ++$y; |
|
|
1216 | |
|
|
1217 | my $mask = $_->[1]; |
|
|
1218 | $table->add ($x , $y, new CFClient::UI::Label text => $_->[0], align => 1, expand => 1); |
|
|
1219 | $table->add ($x+1, $y, my $checkbox = new CFClient::UI::CheckBox |
|
|
1220 | state => $::CFG->{pickup} & $mask, |
|
|
1221 | on_changed => sub { |
|
|
1222 | my ($box, $value) = @_; |
|
|
1223 | |
|
|
1224 | if ($value) { |
|
|
1225 | $::CFG->{pickup} |= $mask; |
|
|
1226 | } else { |
|
|
1227 | $::CFG->{pickup} &= ~$mask; |
|
|
1228 | } |
|
|
1229 | |
|
|
1230 | $::CONN->send_command ("pickup $::CFG->{pickup}") |
|
|
1231 | if defined $::CONN; |
|
|
1232 | |
|
|
1233 | 0 |
|
|
1234 | }); |
|
|
1235 | |
|
|
1236 | ${$_->[2]} = $checkbox if $_->[2]; |
|
|
1237 | } |
|
|
1238 | } |
|
|
1239 | |
|
|
1240 | $table->add (2, 18, new CFClient::UI::ValSlider |
|
|
1241 | range => [$::CFG->{pickup} & 0xF, 0, 16, 1, 1], |
|
|
1242 | template => ">= 99", |
|
|
1243 | to_value => sub { ">= " . 5 * $_[0] }, |
|
|
1244 | on_changed => sub { |
|
|
1245 | my ($slider, $value) = @_; |
|
|
1246 | |
|
|
1247 | $::CFG->{pickup} &= ~0xF; |
|
|
1248 | $::CFG->{pickup} |= int $value |
|
|
1249 | if $value; |
|
|
1250 | 1; |
|
|
1251 | }); |
|
|
1252 | |
|
|
1253 | $table->add (3, 18, new CFClient::UI::Button |
|
|
1254 | text => "set", |
|
|
1255 | on_activate => sub { |
|
|
1256 | $::CONN->send_command ("pickup $::CFG->{pickup}") |
|
|
1257 | if defined $::CONN; |
|
|
1258 | 0 |
|
|
1259 | }); |
|
|
1260 | |
|
|
1261 | $table |
|
|
1262 | } |
|
|
1263 | |
|
|
1264 | my %SORT_ORDER = ( |
|
|
1265 | type => undef, |
|
|
1266 | mtime => sub { sort { |
|
|
1267 | ($a->{flags} & F_LOCKED) <=> ($b->{flags} & F_LOCKED) |
|
|
1268 | or $b->{mtime} <=> $a->{mtime} |
|
|
1269 | or $a->{type} <=> $b->{type} |
|
|
1270 | } @_ }, |
|
|
1271 | weight => sub { sort { |
|
|
1272 | $a->{weight} * ($a->{nrof} || 1) <=> $b->{weight} * ($b->{nrof} || 1) |
|
|
1273 | or $a->{type} <=> $b->{type} |
|
|
1274 | } @_ }, |
|
|
1275 | ); |
|
|
1276 | |
|
|
1277 | sub inventory_widget { |
|
|
1278 | my $hb = new CFClient::UI::HBox homogeneous => 1; |
|
|
1279 | |
|
|
1280 | $hb->add (my $vb1 = new CFClient::UI::VBox); |
|
|
1281 | $vb1->add (new CFClient::UI::Label align => 0, text => "Player"); |
|
|
1282 | |
|
|
1283 | $vb1->add (my $hb1 = new CFClient::UI::HBox); |
|
|
1284 | |
|
|
1285 | use sort 'stable'; |
|
|
1286 | |
|
|
1287 | $hb1->add (new CFClient::UI::Selector |
|
|
1288 | value => $::CFG->{inv_sort}, |
|
|
1289 | options => [ |
|
|
1290 | [type => "Type/Name"], |
|
|
1291 | [mtime => "Recent/Normal/Locked"], |
|
|
1292 | [weight => "Weight/Type"], |
|
|
1293 | ], |
|
|
1294 | on_changed => sub { |
|
|
1295 | $::CFG->{inv_sort} = $_[1]; |
|
|
1296 | $INV->set_sort_order ($SORT_ORDER{$_[1]}); |
|
|
1297 | }, |
|
|
1298 | ); |
|
|
1299 | $hb1->add (new CFClient::UI::Label text => "Weight: ", align => 1, expand => 1); |
|
|
1300 | #TODO# update to weigh/maxweight |
|
|
1301 | $hb1->add ($STATWIDS->{i_weight} = new CFClient::UI::Label align => -1); |
|
|
1302 | |
|
|
1303 | $vb1->add (my $sw1 = new CFClient::UI::ScrolledWindow expand => 1, scroll_y => 1); |
|
|
1304 | $sw1->add ($INV = new CFClient::UI::Inventory); |
|
|
1305 | |
|
|
1306 | $hb->add (my $vb2 = new CFClient::UI::VBox); |
|
|
1307 | |
|
|
1308 | $vb2->add ($INV_RIGHT_HB = new CFClient::UI::HBox); |
|
|
1309 | |
|
|
1310 | $vb2->add (my $sw2 = new CFClient::UI::ScrolledWindow expand => 1, scroll_y => 1); |
|
|
1311 | $sw2->add ($INVR = new CFClient::UI::Inventory); |
|
|
1312 | |
|
|
1313 | # XXX: Call after $INVR = ... because set_opencont sets the items |
|
|
1314 | CFClient::Protocol::set_opencont ($::CONN, 0, "Floor"); |
|
|
1315 | |
|
|
1316 | $hb |
|
|
1317 | } |
|
|
1318 | |
|
|
1319 | sub toggle_player_page { |
|
|
1320 | my ($widget) = @_; |
|
|
1321 | |
|
|
1322 | if ($PL_WINDOW->{visible} && $PL_NOTEBOOK->get_current_page == $widget) { |
|
|
1323 | $PL_WINDOW->hide; |
|
|
1324 | } else { |
|
|
1325 | $PL_NOTEBOOK->set_current_page ($widget); |
|
|
1326 | $PL_WINDOW->show; |
|
|
1327 | } |
|
|
1328 | } |
|
|
1329 | |
|
|
1330 | sub player_window { |
|
|
1331 | my $plwin = $PL_WINDOW = new CFClient::UI::FancyFrame |
|
|
1332 | x => "center", |
|
|
1333 | y => "center", |
|
|
1334 | force_w => $WIDTH * 9/10, |
|
|
1335 | force_h => $HEIGHT * 9/10, |
|
|
1336 | title => "Player", |
|
|
1337 | name => "playerbook", |
|
|
1338 | has_close_button => 1 |
|
|
1339 | ; |
|
|
1340 | |
|
|
1341 | my $ntb = |
|
|
1342 | $PL_NOTEBOOK = |
|
|
1343 | new CFClient::UI::Notebook expand => 1, debug => 1; |
|
|
1344 | |
|
|
1345 | $ntb->add ( |
|
|
1346 | "Statistics (F2)" => $STATS_PAGE = stats_window, |
|
|
1347 | "Shows statistics, where all your Stats and Resistances are shown." |
|
|
1348 | ); |
|
|
1349 | $ntb->add ( |
|
|
1350 | "Skills (F3)" => $SKILL_PAGE = skill_window, |
|
|
1351 | "Shows all your Skills." |
|
|
1352 | ); |
|
|
1353 | |
|
|
1354 | my $spellsw = new CFClient::UI::ScrolledWindow (expand => 1, scroll_y => 1); |
|
|
1355 | $spellsw->add ($SPELL_PAGE = new CFClient::UI::SpellList); |
|
|
1356 | $ntb->add ( |
|
|
1357 | "Spellbook (F4)" => $spellsw, |
|
|
1358 | "Displays all spells you have and lets you edit keyboard shortcuts for them." |
|
|
1359 | ); |
|
|
1360 | $ntb->add ( |
|
|
1361 | "Inventory (F5)" => $INVENTORY_PAGE = inventory_widget, |
|
|
1362 | "Toggles the inventory window, where you can manage your loot (or treasures :). " |
|
|
1363 | . "You can also hit the <b>Tab</b>-key to show/hide the Inventory." |
|
|
1364 | ); |
|
|
1365 | |
|
|
1366 | $ntb->set_current_page ($INVENTORY_PAGE); |
|
|
1367 | |
|
|
1368 | $plwin->add ($ntb); |
|
|
1369 | $plwin |
|
|
1370 | } |
|
|
1371 | |
|
|
1372 | sub update_bindings { |
|
|
1373 | $BIND_UPD_CB->() if $BIND_UPD_CB; |
|
|
1374 | } |
|
|
1375 | |
|
|
1376 | sub keyboard_setup { |
|
|
1377 | my $binding_list = new CFClient::UI::VBox; |
|
|
1378 | |
|
|
1379 | my $refresh; |
|
|
1380 | $refresh = $BIND_UPD_CB = sub { |
|
|
1381 | $binding_list->clear (); |
|
|
1382 | |
|
|
1383 | for my $mod (keys %{$::CFG->{profile}{default}{bindings}}) { |
|
|
1384 | for my $sym (keys %{$::CFG->{profile}{default}{bindings}{$mod}}) { |
|
|
1385 | my $cmds = $::CFG->{profile}{default}{bindings}{$mod}{$sym}; |
|
|
1386 | next unless ref $cmds eq 'ARRAY' and @$cmds > 0; |
|
|
1387 | |
|
|
1388 | my $lbl = join "; ", @$cmds; |
|
|
1389 | my $nam = CFClient::BindingEditor::keycombo_to_name ($mod, $sym); |
|
|
1390 | $binding_list->add (my $hb = new CFClient::UI::HBox); |
|
|
1391 | $hb->add (new CFClient::UI::Button |
|
|
1392 | text => "delete", |
|
|
1393 | tooltip => "Deletes the binding", |
|
|
1394 | on_activate => sub { |
|
|
1395 | $binding_list->remove ($hb); |
|
|
1396 | delete $::CFG->{profile}{default}{bindings}{$mod}{$sym}; |
|
|
1397 | 0 |
|
|
1398 | }); |
|
|
1399 | |
|
|
1400 | $hb->add (new CFClient::UI::Button |
|
|
1401 | text => "edit", |
|
|
1402 | tooltip => "Edits the binding", |
|
|
1403 | on_activate => sub { |
|
|
1404 | $::BIND_EDITOR->set_binding ( |
|
|
1405 | $mod, $sym, $::CFG->{profile}{default}{bindings}{$mod}{$sym}, |
|
|
1406 | sub { |
|
|
1407 | my ($nmod, $nsym, $ncmds) = @_; |
|
|
1408 | $::BIND_EDITOR->cfg_unbind ($mod, $sym); |
|
|
1409 | $::BIND_EDITOR->cfg_bind ($nmod, $nsym, $ncmds); |
|
|
1410 | $refresh->(); |
|
|
1411 | $SETUP_NOTEBOOK->set_current_page ($SETUP_KEYBOARD); |
|
|
1412 | $SETUP_DIALOG->show; |
|
|
1413 | }, |
|
|
1414 | sub { |
|
|
1415 | $SETUP_NOTEBOOK->set_current_page ($SETUP_KEYBOARD); |
|
|
1416 | $SETUP_DIALOG->show; |
|
|
1417 | }); |
|
|
1418 | $::BIND_EDITOR->show; |
|
|
1419 | $SETUP_DIALOG->hide; |
|
|
1420 | 0 |
|
|
1421 | }); |
|
|
1422 | |
|
|
1423 | $hb->add (new CFClient::UI::Label text => "(Key: $nam)"); |
|
|
1424 | $hb->add (new CFClient::UI::Label text => $lbl, expand => 1); |
|
|
1425 | } |
|
|
1426 | } |
|
|
1427 | }; |
|
|
1428 | |
|
|
1429 | my $vb = new CFClient::UI::VBox; |
|
|
1430 | $vb->add (my $hb = new CFClient::UI::HBox); |
|
|
1431 | $hb->add (new CFClient::UI::Label text => "only shift-up stops fire"); |
|
|
1432 | $hb->add (new CFClient::UI::CheckBox |
|
|
1433 | expand => 1, |
|
|
1434 | state => $CFG->{shift_fire_stop}, |
|
|
1435 | tooltip => "If this checkbox is enabled you will stop fire only if you stop pressing shift", |
|
|
1436 | on_changed => sub { |
|
|
1437 | my ($cbox, $value) = @_; |
|
|
1438 | $CFG->{shift_fire_stop} = $value; |
|
|
1439 | 0 |
|
|
1440 | }); |
|
|
1441 | |
|
|
1442 | $vb->add ($binding_list); |
|
|
1443 | $vb->add (my $hb = new CFClient::UI::HBox); |
|
|
1444 | |
|
|
1445 | $hb->add (new CFClient::UI::Button |
|
|
1446 | text => "record new", |
|
|
1447 | expand => 1, |
|
|
1448 | tooltip => "This button opens the binding editor with an empty binding.", |
|
|
1449 | on_activate => sub { |
|
|
1450 | $::BIND_EDITOR->set_binding (undef, undef, [], |
|
|
1451 | sub { |
|
|
1452 | my ($mod, $sym, $cmds) = @_; |
|
|
1453 | $::BIND_EDITOR->cfg_bind ($mod, $sym, $cmds); |
|
|
1454 | $refresh->(); |
|
|
1455 | $SETUP_NOTEBOOK->set_current_page ($SETUP_KEYBOARD); |
|
|
1456 | $SETUP_DIALOG->show; |
|
|
1457 | }, |
|
|
1458 | sub { |
|
|
1459 | $SETUP_NOTEBOOK->set_current_page ($SETUP_KEYBOARD); |
|
|
1460 | $SETUP_DIALOG->show; |
|
|
1461 | }, |
|
|
1462 | ); |
|
|
1463 | $SETUP_DIALOG->hide; |
|
|
1464 | $::BIND_EDITOR->show; |
|
|
1465 | 0 |
|
|
1466 | }, |
|
|
1467 | ); |
|
|
1468 | |
|
|
1469 | $hb->add (new CFClient::UI::Button |
|
|
1470 | text => "close", |
|
|
1471 | tooltip => "Closes the binding window", |
|
|
1472 | expand => 1, |
|
|
1473 | on_activate => sub { |
|
|
1474 | $SETUP_DIALOG->hide; |
|
|
1475 | 0 |
|
|
1476 | } |
|
|
1477 | ); |
|
|
1478 | |
|
|
1479 | $refresh->(); |
|
|
1480 | |
|
|
1481 | $vb |
|
|
1482 | } |
|
|
1483 | |
|
|
1484 | # just weirdness, pls. ignore |
|
|
1485 | sub load_html_page { |
|
|
1486 | my ($viewer, $base) = @_; |
|
|
1487 | |
|
|
1488 | $viewer->clear; |
|
|
1489 | |
|
|
1490 | require LWP::Simple; |
|
|
1491 | require HTML::Parser; |
|
|
1492 | require URI; |
|
|
1493 | |
|
|
1494 | my $page = LWP::Simple::get ($base) |
|
|
1495 | or return; |
|
|
1496 | |
|
|
1497 | my @s = { }; |
|
|
1498 | my %passthrough = map ($_ => undef), qw(b i u s tt big small sub sup); |
|
|
1499 | |
|
|
1500 | my $parser = HTML::Parser->new ( |
|
|
1501 | text_h => [sub { |
|
|
1502 | my ($text) = @_; |
|
|
1503 | $text =~ s/\s+/ /g; |
|
|
1504 | $s[-1]{text} .= CFClient::asxml $text; |
|
|
1505 | }, "dtext"], |
|
|
1506 | start_h => [sub { |
|
|
1507 | my ($tag, $attr) = @_; |
|
|
1508 | if ($passthrough{$tag}) { |
|
|
1509 | $s[-1]{text} .= "<$tag>"; |
|
|
1510 | } elsif ($tag eq "h1") { |
|
|
1511 | push @s, { text => "<span foreground='#ffff00' size='x-large'>" }; |
|
|
1512 | } elsif ($tag eq "h2") { |
|
|
1513 | push @s, { text => "<span foreground='#ccccff' size='large'>" }; |
|
|
1514 | } elsif ($tag eq "h3") { |
|
|
1515 | push @s, { text => "<span size='large'>" }; |
|
|
1516 | } elsif ($tag eq "a") { |
|
|
1517 | push @s, { text => "", url => $attr->{href} }; |
|
|
1518 | } elsif ($tag eq "p") { |
|
|
1519 | push @s, { }; |
|
|
1520 | } elsif ($tag eq "img") { |
|
|
1521 | eval { |
|
|
1522 | push @{$s[-1]{obj}}, new CFClient::UI::Image |
|
|
1523 | tex => (new_from_image CFClient::Texture LWP::Simple::get (URI->new ($attr->{src}, $base)->abs ($base))); |
|
|
1524 | $s[-1]{text} .= "\x{fffc}"; |
|
|
1525 | }; |
|
|
1526 | } |
|
|
1527 | }, "tagname, attr"], |
|
|
1528 | end_h => [sub { |
|
|
1529 | my ($tag) = @_; |
|
|
1530 | if ($passthrough{$tag}) { |
|
|
1531 | $s[-1]{text} .= "</$tag>"; |
|
|
1532 | } elsif ($tag =~ /^h\d$/) { |
|
|
1533 | $s[-1]{text} .= "</span>"; |
|
|
1534 | push @s, { }; |
|
|
1535 | } elsif ($tag eq "a") { |
|
|
1536 | my $S = pop @s; |
|
|
1537 | $s[-1]{text} .= "\x{fffc}"; |
|
|
1538 | push @{$s[-1]{obj}}, new CFClient::UI::Label |
|
|
1539 | fg => [0.8, 0.8, 1], |
|
|
1540 | markup => "<u>$S->{text}</u>", |
|
|
1541 | fontsize => 0.8, |
|
|
1542 | can_events => 1, |
|
|
1543 | can_focus => 1, |
|
|
1544 | on_button_up => sub { |
|
|
1545 | load_html_page ($viewer, URI->new ($S->{url}, $base)->abs ($base)); |
|
|
1546 | }, |
|
|
1547 | ; |
|
|
1548 | } |
|
|
1549 | }, "tagname"], |
|
|
1550 | ); |
|
|
1551 | |
|
|
1552 | $parser->parse ($page); |
|
|
1553 | $parser->eof; |
|
|
1554 | |
|
|
1555 | $viewer->add_paragraph ([1, 1, 1, 1], [$_->{text}, @{ $_->{obj} || [] }], $_->{indent}) |
|
|
1556 | for @s; |
|
|
1557 | |
|
|
1558 | $viewer->set_offset (0); |
|
|
1559 | } |
|
|
1560 | |
|
|
1561 | sub help_window { |
|
|
1562 | my $win = new CFClient::UI::FancyFrame |
|
|
1563 | x => 'center', |
|
|
1564 | y => 'center', |
|
|
1565 | z => 2, |
|
|
1566 | name => 'doc_browser', |
|
|
1567 | force_w => int $WIDTH * 7/8, |
|
|
1568 | force_h => int $HEIGHT * 7/8, |
|
|
1569 | title => "Help Browser", |
|
|
1570 | has_close_button => 1; |
|
|
1571 | |
|
|
1572 | $win->add (my $vbox = new CFClient::UI::VBox); |
|
|
1573 | |
|
|
1574 | $vbox->add (my $buttons = new CFClient::UI::HBox); |
|
|
1575 | $vbox->add (my $viewer = new CFClient::UI::TextScroller |
|
|
1576 | expand => 1, fontsize => 0.8, padding_x => 4); |
|
|
1577 | |
|
|
1578 | $buttons->add (new CFClient::UI::Label text => "Choose a document to display: "); |
|
|
1579 | $buttons->add (my $combo = new CFClient::UI::Selector |
|
|
1580 | value => undef, |
|
|
1581 | options => [ |
|
|
1582 | [intro => "Introduction"], |
|
|
1583 | [manual => "Main Manual"], |
|
|
1584 | [skill_help => "Skill Reference"], |
|
|
1585 | [command_help => "Command Reference"], |
|
|
1586 | [dmcommand_help => "DM Commands"], |
|
|
1587 | [COPYING => "License Terms"], |
|
|
1588 | [test => "test (do not select)"], #d#TODO |
|
|
1589 | ], |
|
|
1590 | on_changed => sub { |
|
|
1591 | my ($self, $pod) = @_; |
|
|
1592 | |
|
|
1593 | if ($pod eq "test") {#d#TODO |
|
|
1594 | eval { |
|
|
1595 | load_html_page $viewer, "http://crossfire.real-time.com/guides/walkthrough/newbie-tower.html"; |
|
|
1596 | }; |
|
|
1597 | warn "$@" if $@; |
|
|
1598 | return; |
|
|
1599 | } |
|
|
1600 | |
|
|
1601 | my $pom = CFClient::Pod::load CFClient::find_rcfile "pod/$pod.pod", |
|
|
1602 | doc_viewer => 1, sub { CFClient::Pod::as_paragraphs $_[0] }; |
|
|
1603 | |
|
|
1604 | #use Data::Dumper; warn Dumper $pom;#d# |
|
|
1605 | |
|
|
1606 | $viewer->clear; |
|
|
1607 | |
|
|
1608 | # $viewer->add_paragraph ([1, 1, 1, 1], ["<big>Test</big>\n\n \x{fffc} \x{fffc}\n", |
|
|
1609 | # (new CFClient::UI::Image path => "x.png", can_hover => 1, can_events => 1), |
|
|
1610 | # (new CFClient::UI::Label text => "üüüü", can_hover => 1, can_events => 1, tooltip => "??"), |
|
|
1611 | # ]);#d# |
|
|
1612 | |
|
|
1613 | $viewer->add_paragraph ([1, 1, 1, 1], [$_->{text}, @{ $_->{obj} || [] }], $_->{indent}) |
|
|
1614 | for @$pom; |
|
|
1615 | |
|
|
1616 | $viewer->set_offset (0); |
|
|
1617 | |
|
|
1618 | 0 |
|
|
1619 | }, |
|
|
1620 | on_visibility_change => sub { |
|
|
1621 | my ($self, $visible) = @_; |
|
|
1622 | return unless $visible; |
|
|
1623 | return if $self->{value}; |
|
|
1624 | $self->set_value ("intro"); |
|
|
1625 | 0 |
|
|
1626 | }, |
|
|
1627 | ); |
945 | |
1628 | |
946 | $win |
1629 | $win |
947 | } |
1630 | } |
948 | |
1631 | |
949 | sub sdl_init { |
1632 | sub sdl_init { |
… | |
… | |
966 | or die "SDL_SetVideoMode failed: " . (CFClient::SDL_GetError) . "\n"; |
1649 | or die "SDL_SetVideoMode failed: " . (CFClient::SDL_GetError) . "\n"; |
967 | |
1650 | |
968 | $SDL_ACTIVE = 1; |
1651 | $SDL_ACTIVE = 1; |
969 | $LAST_REFRESH = time - 0.01; |
1652 | $LAST_REFRESH = time - 0.01; |
970 | |
1653 | |
971 | CFClient::gl_init; |
1654 | CFClient::OpenGL::init; |
972 | |
1655 | |
973 | $FONTSIZE = int $HEIGHT / 40 * $CFG->{gui_fontsize}; |
1656 | $FONTSIZE = int $HEIGHT / 40 * $CFG->{gui_fontsize}; |
974 | |
1657 | |
975 | $CFClient::UI::ROOT->configure (0, 0, $WIDTH, $HEIGHT);#d# |
1658 | $CFClient::UI::ROOT->configure (0, 0, $WIDTH, $HEIGHT);#d# |
976 | |
1659 | |
… | |
… | |
979 | if ($DEBUG_STATUS) { |
1662 | if ($DEBUG_STATUS) { |
980 | CFClient::UI::rescale_widgets $WIDTH / $old_w, $HEIGHT / $old_h; |
1663 | CFClient::UI::rescale_widgets $WIDTH / $old_w, $HEIGHT / $old_h; |
981 | } else { |
1664 | } else { |
982 | # create the widgets |
1665 | # create the widgets |
983 | |
1666 | |
984 | $DEBUG_STATUS = new CFClient::UI::Label padding => 0, z => 100, req_x => -1; |
1667 | $DEBUG_STATUS = new CFClient::UI::Label |
|
|
1668 | padding => 0, |
|
|
1669 | z => 100, |
|
|
1670 | force_x => "max", |
|
|
1671 | force_y => 0; |
985 | $DEBUG_STATUS->show; |
1672 | $DEBUG_STATUS->show; |
986 | |
1673 | |
|
|
1674 | $BIND_EDITOR = new CFClient::BindingEditor (x => "max", y => 0); |
|
|
1675 | |
987 | $STATUSBOX = new CFClient::UI::Statusbox; |
1676 | $STATUSBOX = new CFClient::UI::Statusbox; |
988 | $STATUSBOX->add ("Use <b>Alt-Enter</b> to toggle fullscreen mode", pri => -100, color => [1, 1, 1, 0.8]); |
1677 | $STATUSBOX->add ("Use <b>Alt-Enter</b> to toggle fullscreen mode", timeout => 864000, pri => -100, color => [1, 1, 1, 0.8]); |
989 | |
1678 | |
990 | (new CFClient::UI::Frame |
1679 | (new CFClient::UI::Frame |
991 | bg => [0, 0, 0, 0.4], |
1680 | bg => [0, 0, 0, 0.4], |
992 | req_y => -1, |
1681 | force_x => 0, |
|
|
1682 | force_y => "max", |
993 | child => $STATUSBOX, |
1683 | child => $STATUSBOX, |
994 | )->show; |
1684 | )->show; |
995 | |
1685 | |
996 | CFClient::UI::FancyFrame->new ( |
1686 | CFClient::UI::FancyFrame->new ( |
|
|
1687 | title => "Map", |
|
|
1688 | name => "mapmap", |
|
|
1689 | x => 0, |
|
|
1690 | y => $FONTSIZE + 8, |
997 | border_bg => [1, 1, 1, 192/255], |
1691 | border_bg => [1, 1, 1, 192/255], |
998 | bg => [1, 1, 1, 0], |
1692 | bg => [1, 1, 1, 0], |
999 | child => ($MAPMAP = new CFClient::MapWidget::MapMap |
1693 | child => ($MAPMAP = new CFClient::MapWidget::MapMap |
1000 | tooltip => "<b>Map</b>. On servers that support this feature, this will display an overview of the surrounding areas.", |
1694 | tooltip => "<b>Map</b>. On servers that support this feature, this will display an overview of the surrounding areas.", |
1001 | ), |
1695 | ), |
… | |
… | |
1005 | $MAPWIDGET->connect (activate_console => sub { |
1699 | $MAPWIDGET->connect (activate_console => sub { |
1006 | my ($mapwidget, $preset) = @_; |
1700 | my ($mapwidget, $preset) = @_; |
1007 | |
1701 | |
1008 | if ($CONSOLE) { |
1702 | if ($CONSOLE) { |
1009 | $CONSOLE->{input}->{auto_activated} = 1; |
1703 | $CONSOLE->{input}->{auto_activated} = 1; |
1010 | $CONSOLE->{input}->focus_in; |
1704 | $CONSOLE->{input}->grab_focus; |
1011 | |
1705 | |
1012 | if ($preset && $CONSOLE->{input}->get_text eq '') { |
1706 | if ($preset && $CONSOLE->{input}->get_text eq '') { |
1013 | $CONSOLE->{input}->set_text ($preset); |
1707 | $CONSOLE->{input}->set_text ($preset); |
1014 | } |
1708 | } |
1015 | } |
1709 | } |
1016 | }); |
1710 | }); |
1017 | $MAPWIDGET->show; |
1711 | $MAPWIDGET->show; |
1018 | $MAPWIDGET->focus_in; |
1712 | $MAPWIDGET->grab_focus; |
1019 | |
1713 | |
1020 | $LOGVIEW = new CFClient::UI::TextView |
1714 | $LOGVIEW = new CFClient::UI::TextScroller |
1021 | expand => 1, |
1715 | expand => 1, |
1022 | font => $FONT_FIXED, |
1716 | font => $FONT_FIXED, |
1023 | fontsize => $::CFG->{log_fontsize}, |
1717 | fontsize => $::CFG->{log_fontsize}, |
|
|
1718 | indent => -4, |
1024 | can_hover => 1, |
1719 | can_hover => 1, |
1025 | can_events => 1, |
1720 | can_events => 1, |
1026 | tooltip => "<b>Server Log</b>. This text viewer contains all the messages sent by the server.", |
1721 | tooltip => "<b>Server Log</b>. This text viewer contains all the messages sent by the server.", |
1027 | ; |
1722 | ; |
1028 | |
1723 | |
1029 | $BUTTONBAR = new CFClient::UI::HBox; |
1724 | $SETUP_DIALOG = new CFClient::UI::FancyFrame |
|
|
1725 | title => "Setup", |
|
|
1726 | name => "setup_dialog", |
|
|
1727 | x => 'center', |
|
|
1728 | y => 'center', |
|
|
1729 | z => 2, |
|
|
1730 | force_w => $::WIDTH * 0.6, |
|
|
1731 | force_h => $::HEIGHT * 0.6, |
|
|
1732 | has_close_button => 1, |
|
|
1733 | ; |
1030 | |
1734 | |
|
|
1735 | $METASERVER = metaserver_dialog; |
|
|
1736 | |
|
|
1737 | $SETUP_DIALOG->add ($SETUP_NOTEBOOK = new CFClient::UI::Notebook expand => 1, debug => 1, |
|
|
1738 | filter => new CFClient::UI::ScrolledWindow expand => 1, scroll_y => 1); |
|
|
1739 | |
|
|
1740 | $SETUP_NOTEBOOK->add (Server => $SETUP_SERVER = server_setup, |
|
|
1741 | "Configure the server to play on, your username, password and other server-related options."); |
|
|
1742 | $SETUP_NOTEBOOK->add (Pickup => autopickup_setup, |
|
|
1743 | "Configure autopickup settings, i.e. which items you will pick up automatically when walking (or running) over them."); |
|
|
1744 | $SETUP_NOTEBOOK->add (Graphics => graphics_setup, |
|
|
1745 | "Configure the video mode, performance, fonts and other graphical aspects of the game."); |
|
|
1746 | $SETUP_NOTEBOOK->add (Audio => audio_setup, |
|
|
1747 | "Configure the use of audio, sound effects and background music."); |
|
|
1748 | $SETUP_NOTEBOOK->add (Keyboard => $SETUP_KEYBOARD = keyboard_setup, |
|
|
1749 | "Lets you define, edit and delete key bindings." |
|
|
1750 | . "There is a shortcut for making bindings: <b>Control-Insert</b> opens the binding editor " |
|
|
1751 | . "with nothing set and the recording started. After doing the actions you " |
|
|
1752 | . "want to record press <b>Insert</b> and you will be asked to press a key-combo. " |
|
|
1753 | . "After pressing the combo the binding will be saved automatically and the " |
|
|
1754 | . "binding editor closes"); |
|
|
1755 | $SETUP_NOTEBOOK->add (Debug => debug_setup, |
|
|
1756 | "Some debuggin' options. Do not ask."); |
|
|
1757 | |
|
|
1758 | $BUTTONBAR = new CFClient::UI::Buttonbar x => 0, y => 0, z => 200; # put on top |
|
|
1759 | |
1031 | $BUTTONBAR->add (new CFClient::UI::Flopper text => "Client Setup", other => client_setup, |
1760 | $BUTTONBAR->add (new CFClient::UI::Flopper text => "Setup", other => $SETUP_DIALOG, |
1032 | tooltip => "Toggles a dialog where you can configure various aspects of the client, such as graphics mode, performance, and audio options."); |
1761 | tooltip => "Toggles a dialog where you can configure all aspects of this client."); |
1033 | $BUTTONBAR->add (new CFClient::UI::Flopper text => "Server Setup", other => server_setup, |
1762 | |
1034 | tooltip => "Toggles a dialog where you can configure the server to play on, your username, password and other server-related options."); |
|
|
1035 | $BUTTONBAR->add (new CFClient::UI::Flopper text => "Message Window", other => message_window, |
1763 | $BUTTONBAR->add (new CFClient::UI::Flopper text => "Message Window", other => $MESSAGE_WINDOW = message_window, |
1036 | tooltip => "Toggles the server message log, where the client collects <i>all</i> messages from the server."); |
1764 | tooltip => "Toggles the server message log, where the client collects <i>all</i> messages from the server."); |
1037 | |
1765 | |
1038 | make_gauge_window->show; # XXX: this has to be set before make_stats_window as make_stats_window calls update_stats_window which updated the gauges also X-D |
1766 | make_gauge_window->show; # XXX: this has to be set before make_stats_window as make_stats_window calls update_stats_window which updated the gauges also X-D |
1039 | |
1767 | |
1040 | $BUTTONBAR->add (new CFClient::UI::Flopper text => "Stats Window", other => make_stats_window, |
|
|
1041 | tooltip => "Toggles the statistics window, where all your Stats and Resistances are being displayed at all times."); |
|
|
1042 | $BUTTONBAR->add (new CFClient::UI::Flopper text => "Inventory", other => make_inventory_window, |
1768 | $BUTTONBAR->add (new CFClient::UI::Flopper text => "Playerbook", other => player_window, |
1043 | tooltip => "Toggles the inventory window, where you can manage your loot (or treaures :)."); |
1769 | tooltip => "Toggles the player view, where you can manage Inventory, Spells, Skills and see your Stats."); |
1044 | |
1770 | |
1045 | $BUTTONBAR->add (new CFClient::UI::Button |
1771 | $BUTTONBAR->add (new CFClient::UI::Button |
1046 | text => "Save Config", |
1772 | text => "Save Config", |
1047 | tooltip => "Saves the options chosen in the client setting, server settings and the window layout to be restored on later runs.", |
1773 | tooltip => "Saves the options chosen in the client setting, server settings and the window layout to be restored on later runs.", |
1048 | connect_activate => sub { |
1774 | on_activate => sub { |
|
|
1775 | $::CFG->{layout} = CFClient::UI::get_layout; |
1049 | CFClient::write_cfg "$Crossfire::VARDIR/pclientrc"; |
1776 | CFClient::write_cfg "$Crossfire::VARDIR/cfplusrc"; |
1050 | status "Configuration Saved"; |
1777 | status "Configuration Saved"; |
|
|
1778 | 0 |
1051 | }, |
1779 | }, |
1052 | ); |
1780 | ); |
1053 | |
1781 | |
1054 | $BUTTONBAR->add (new CFClient::UI::Flopper text => "Help!", other => make_help_window, |
1782 | $BUTTONBAR->add (new CFClient::UI::Flopper text => "Help!", other => $HELP_WINDOW = help_window, |
1055 | tooltip => "View Documentation"); |
1783 | tooltip => "View Documentation"); |
1056 | |
1784 | |
1057 | $BUTTONBAR->add (new CFClient::UI::Button |
1785 | $BUTTONBAR->add (new CFClient::UI::Button |
1058 | text => "Quit", |
1786 | text => "Quit", |
1059 | tooltip => "Terminates the program", |
1787 | tooltip => "Terminates the program", |
1060 | connect_activate => sub { |
1788 | on_activate => sub { |
1061 | if ($CONN) { |
1789 | if ($CONN) { |
1062 | open_quit_dialog; |
1790 | open_quit_dialog; |
1063 | } else { |
1791 | } else { |
1064 | exit; |
1792 | exit; |
1065 | } |
1793 | } |
|
|
1794 | 0 |
1066 | }, |
1795 | }, |
1067 | ); |
1796 | ); |
1068 | |
1797 | |
1069 | $BUTTONBAR->show; |
1798 | $BUTTONBAR->show; |
1070 | |
1799 | $SETUP_DIALOG->show; |
1071 | $STATUSBOX->add ("Set video mode $WIDTH×$HEIGHT", timeout => 10, fg => [1, 1, 1, 0.5]); |
|
|
1072 | |
|
|
1073 | # delay till geometry is constant |
|
|
1074 | $CFClient::UI::ROOT->on_post_alloc (startup => sub { |
|
|
1075 | $BUTTONBAR->{children}[1]->emit ("activate"); # pop up server setup |
|
|
1076 | my $widget = $GAUGES->{win}; |
|
|
1077 | $widget->move (0, $HEIGHT - $widget->{h});#d# to in toplevel |
|
|
1078 | }); |
|
|
1079 | force_refresh (); |
|
|
1080 | } |
1800 | } |
|
|
1801 | |
|
|
1802 | $STATUSBOX->add ("Set video mode $WIDTH×$HEIGHT", timeout => 10, fg => [1, 1, 1, 0.5]); |
1081 | } |
1803 | } |
1082 | |
1804 | |
1083 | sub video_shutdown { |
1805 | sub video_shutdown { |
|
|
1806 | CFClient::OpenGL::shutdown; |
|
|
1807 | |
1084 | undef $SDL_ACTIVE; |
1808 | undef $SDL_ACTIVE; |
1085 | } |
1809 | } |
1086 | |
1810 | |
1087 | my @bgmusic = qw(game1.ogg game2.ogg game3.ogg game5.ogg game6.ogg ross1.ogg ross2.ogg ross3.ogg ross4.ogg ross5.ogg); #d# |
1811 | my @bgmusic = qw(game1.ogg game2.ogg game3.ogg game5.ogg game6.ogg ross1.ogg ross2.ogg ross3.ogg ross4.ogg ross5.ogg); #d# |
1088 | my $bgmusic;#TODO#hack#d# |
1812 | my $bgmusic;#TODO#hack#d# |
… | |
… | |
1152 | |
1876 | |
1153 | my %demo;#d# |
1877 | my %demo;#d# |
1154 | |
1878 | |
1155 | sub force_refresh { |
1879 | sub force_refresh { |
1156 | $fps = $fps * 0.95 + 1 / (($NOW - $LAST_REFRESH) || 0.1) * 0.05; |
1880 | $fps = $fps * 0.95 + 1 / (($NOW - $LAST_REFRESH) || 0.1) * 0.05; |
1157 | debug sprintf "%3.2f", $fps; |
1881 | debug sprintf "%3.2f", $fps if $ENV{CFPLUS_DEBUG} & 4; |
1158 | |
1882 | |
1159 | $CFClient::UI::ROOT->draw; |
1883 | $CFClient::UI::ROOT->draw; |
1160 | |
1884 | |
1161 | $WANT_REFRESH = 0; |
1885 | $WANT_REFRESH = 0; |
1162 | $CAN_REFRESH = 0; |
1886 | $CAN_REFRESH = 0; |
1163 | $LAST_REFRESH = $NOW; |
1887 | $LAST_REFRESH = $NOW; |
1164 | |
1888 | |
1165 | 0 && do { |
|
|
1166 | # some weird model-drawing code, just a joke right now |
|
|
1167 | use CFClient::OpenGL; |
|
|
1168 | |
|
|
1169 | $demo{t}{eye_auv} ||= new_from_file CFClient::Texture "eye2.png" or die; |
|
|
1170 | $demo{t}{body_auv} ||= new_from_file CFClient::Texture "body_auv3.png" or die; |
|
|
1171 | $demo{r} ||= do { |
|
|
1172 | my $mod = Compress::LZF::sthaw do { local $/; open my $fh, "<:raw:perlio", "dread.lz3"; <$fh> }; |
|
|
1173 | $mod->{v} = pack "f*", @{$mod->{v}}; |
|
|
1174 | $_ = [scalar @$_, pack "S!*", @$_] |
|
|
1175 | for values %{$mod->{g}}; |
|
|
1176 | $mod |
|
|
1177 | }; |
|
|
1178 | |
|
|
1179 | my $r = $demo{r} or die; |
|
|
1180 | |
|
|
1181 | glDepthMask 1; |
|
|
1182 | glClear GL_DEPTH_BUFFER_BIT; |
|
|
1183 | glEnable GL_TEXTURE_2D; |
|
|
1184 | glEnable GL_DEPTH_TEST; |
|
|
1185 | glEnable GL_CULL_FACE; |
|
|
1186 | glShadeModel $::FAST ? GL_FLAT : GL_SMOOTH; |
|
|
1187 | |
|
|
1188 | glMatrixMode GL_PROJECTION; |
|
|
1189 | glLoadIdentity; |
|
|
1190 | glFrustum -1 * ($::WIDTH / $::HEIGHT), 1 * ($::WIDTH / $::HEIGHT), 1, -1, 1, 10000; |
|
|
1191 | #glOrtho 0, $::WIDTH, 0, $::HEIGHT, -10000, 10000; |
|
|
1192 | glMatrixMode GL_MODELVIEW; |
|
|
1193 | glLoadIdentity; |
|
|
1194 | |
|
|
1195 | glPushMatrix; |
|
|
1196 | glTranslate 0, 0, -800; |
|
|
1197 | glScale 1, -1, 1; |
|
|
1198 | glRotate $NOW * 1000 % 36000 / 5, 0, 1, 0; |
|
|
1199 | glRotate $NOW * 1000 % 36000 / 6, 1, 0, 0; |
|
|
1200 | glRotate $NOW * 1000 % 36000 / 7, 0, 0, 1; |
|
|
1201 | glScale 50, 50, 50; |
|
|
1202 | |
|
|
1203 | glInterleavedArrays GL_T2F_N3F_V3F, 0, $r->{v}; |
|
|
1204 | while (my ($k, $v) = each %{$r->{g}}) { |
|
|
1205 | glBindTexture GL_TEXTURE_2D, ($demo{t}{$k}{name} or die); |
|
|
1206 | glDrawElements GL_TRIANGLES, $v->[0], GL_UNSIGNED_SHORT, $v->[1]; |
|
|
1207 | } |
|
|
1208 | |
|
|
1209 | glPopMatrix; |
|
|
1210 | |
|
|
1211 | glShadeModel GL_FLAT; |
|
|
1212 | glDisable GL_DEPTH_TEST; |
|
|
1213 | glDisable GL_TEXTURE_2D; |
|
|
1214 | glDepthMask 0; |
|
|
1215 | |
|
|
1216 | $WANT_REFRESH++; |
|
|
1217 | }; |
|
|
1218 | |
|
|
1219 | CFClient::SDL_GL_SwapBuffers; |
1889 | CFClient::SDL_GL_SwapBuffers; |
1220 | } |
1890 | } |
1221 | |
1891 | |
1222 | my $refresh_watcher = Event->timer (after => 0, hard => 1, interval => 1 / $MAX_FPS, cb => sub { |
1892 | my $refresh_watcher = Event->timer (after => 0, hard => 0, interval => 1 / $MAX_FPS, cb => sub { |
1223 | $NOW = time; |
1893 | $NOW = time; |
1224 | |
1894 | |
1225 | ($SDL_CB{$_->{type}} || sub { warn "unhandled event $_->{type}" })->($_) |
1895 | ($SDL_CB{$_->{type}} || sub { warn "unhandled event $_->{type}" })->($_) |
1226 | for CFClient::SDL_PollEvent; |
1896 | for CFClient::SDL_PollEvent; |
1227 | |
1897 | |
… | |
… | |
1245 | sub animation_stop { |
1915 | sub animation_stop { |
1246 | my ($widget) = @_; |
1916 | my ($widget) = @_; |
1247 | delete $animate_object{$widget}; |
1917 | delete $animate_object{$widget}; |
1248 | } |
1918 | } |
1249 | |
1919 | |
1250 | @conn::ISA = Crossfire::Protocol::; |
|
|
1251 | |
|
|
1252 | sub conn::new { |
|
|
1253 | my $class = shift; |
|
|
1254 | |
|
|
1255 | my $self = $class->Crossfire::Protocol::new (@_); |
|
|
1256 | |
|
|
1257 | $MAPWIDGET->clr_commands; |
|
|
1258 | |
|
|
1259 | my $parser = new Pod::POM; |
|
|
1260 | my $pod = $parser->parse_file (CFClient::find_rcfile "pod/command_help.pod"); |
|
|
1261 | |
|
|
1262 | for my $head2 ($pod->head2) { |
|
|
1263 | $head2->title =~ /^(\S+) (?:\s+ \( ([^\)]*) \) )?/x |
|
|
1264 | or next; |
|
|
1265 | |
|
|
1266 | my $cmd = $1; |
|
|
1267 | my @args = split /\|/, $2; |
|
|
1268 | @args = (".*") unless @args; |
|
|
1269 | |
|
|
1270 | my $text = CFClient::pod_to_pango $head2->content; |
|
|
1271 | |
|
|
1272 | for my $arg (@args) { |
|
|
1273 | $arg = $arg eq ".*" ? "" : " $arg"; |
|
|
1274 | |
|
|
1275 | $MAPWIDGET->add_command ("$cmd$arg", $text); |
|
|
1276 | } |
|
|
1277 | } |
|
|
1278 | |
|
|
1279 | $self->{noface} = new_from_file CFClient::Texture |
|
|
1280 | CFClient::find_rcfile "noface.png", minify => 1, mipmap => 1; |
|
|
1281 | |
|
|
1282 | $self->{open_container} = 0; |
|
|
1283 | |
|
|
1284 | $self |
|
|
1285 | } |
|
|
1286 | |
|
|
1287 | sub conn::stats_update { |
|
|
1288 | my ($self, $stats) = @_; |
|
|
1289 | |
|
|
1290 | if (my $exp = $stats->{Crossfire::Protocol::CS_STAT_EXP64}) { |
|
|
1291 | my $diff = $exp - $self->{prev_exp}; |
|
|
1292 | $STATUSBOX->add ("$diff experience gained", group => "experience $diff", fg => [0.5, 1, 0.5, 0.8], timeout => 5) |
|
|
1293 | if exists $self->{prev_exp} && $diff; |
|
|
1294 | $self->{prev_exp} = $exp; |
|
|
1295 | } |
|
|
1296 | |
|
|
1297 | update_stats_window ($stats); |
|
|
1298 | } |
|
|
1299 | |
|
|
1300 | sub conn::user_send { |
|
|
1301 | my ($self, $command) = @_; |
|
|
1302 | |
|
|
1303 | $self->send_command ($command); |
|
|
1304 | status $command; |
|
|
1305 | } |
|
|
1306 | |
|
|
1307 | sub conn::map_scroll { |
|
|
1308 | my ($self, $dx, $dy) = @_; |
|
|
1309 | |
|
|
1310 | $MAP->scroll ($dx, $dy); |
|
|
1311 | } |
|
|
1312 | |
|
|
1313 | sub conn::feed_map1a { |
|
|
1314 | my ($self, $data) = @_; |
|
|
1315 | |
|
|
1316 | # $self->Crossfire::Protocol::feed_map1a ($data); |
|
|
1317 | |
|
|
1318 | $MAP->map1a_update ($data); |
|
|
1319 | $MAPWIDGET->update; |
|
|
1320 | } |
|
|
1321 | |
|
|
1322 | sub conn::flush_map { |
|
|
1323 | my ($self) = @_; |
|
|
1324 | |
|
|
1325 | my $map_info = delete $self->{map_info} |
|
|
1326 | or return; |
|
|
1327 | |
|
|
1328 | my ($hash, $x, $y, $w, $h) = @$map_info; |
|
|
1329 | |
|
|
1330 | my $data = $MAP->get_rect ($x, $y, $w, $h); |
|
|
1331 | $MAPCACHE->put ($hash => Compress::LZF::compress $data); |
|
|
1332 | #warn sprintf "SAVEmap[%s] length %d\n", $hash, length $data;#d# |
|
|
1333 | } |
|
|
1334 | |
|
|
1335 | sub conn::map_clear { |
|
|
1336 | my ($self) = @_; |
|
|
1337 | |
|
|
1338 | $self->flush_map; |
|
|
1339 | delete $self->{neigh_map}; |
|
|
1340 | |
|
|
1341 | $MAP->clear; |
|
|
1342 | } |
|
|
1343 | |
|
|
1344 | |
|
|
1345 | sub conn::load_map($$$) { |
|
|
1346 | my ($self, $hash, $x, $y) = @_; |
|
|
1347 | |
|
|
1348 | if (defined (my $data = $MAPCACHE->get ($hash))) { |
|
|
1349 | $data = Compress::LZF::decompress $data; |
|
|
1350 | #warn sprintf "LOADmap[%s,%d,%d] length %d\n", $hash, $x, $y, length $data;#d# |
|
|
1351 | for my $id ($MAP->set_rect ($x, $y, $data)) { |
|
|
1352 | my $data = $TILECACHE->get ($id) |
|
|
1353 | or next; |
|
|
1354 | |
|
|
1355 | $self->set_texture ($id => $data); |
|
|
1356 | } |
|
|
1357 | } |
|
|
1358 | } |
|
|
1359 | |
|
|
1360 | # hardcode /world/world_xxx_xxx map names, the savings are enourmous, |
|
|
1361 | # (server resource,s latency, bandwidth), so this hack is warranted. |
|
|
1362 | # the right fix is to make real tiled maps with an overview file |
|
|
1363 | sub conn::send_mapinfo { |
|
|
1364 | my ($self, $data, $cb) = @_; |
|
|
1365 | |
|
|
1366 | if ($self->{map_info}[0] =~ m%^/world/world_(\d\d\d)_(\d\d\d)$%) { |
|
|
1367 | my ($wx, $wy) = ($1, $2); |
|
|
1368 | |
|
|
1369 | if ($data =~ /^spatial ([1-4]+)$/) { |
|
|
1370 | my @dx = (0, 0, 1, 0, -1); |
|
|
1371 | my @dy = (0, -1, 0, 1, 0); |
|
|
1372 | my ($dx, $dy); |
|
|
1373 | |
|
|
1374 | for (split //, $1) { |
|
|
1375 | $dx += $dx[$_]; |
|
|
1376 | $dy += $dy[$_]; |
|
|
1377 | } |
|
|
1378 | |
|
|
1379 | $cb->(spatial => 15, |
|
|
1380 | $self->{map_info}[1] - $MAP->ox + $dx * 50, |
|
|
1381 | $self->{map_info}[2] - $MAP->oy + $dy * 50, |
|
|
1382 | 50, 50, |
|
|
1383 | sprintf "/world/world_%03d_%03d", $wx + $dx, $wy + $dy |
|
|
1384 | ); |
|
|
1385 | |
|
|
1386 | return; |
|
|
1387 | } |
|
|
1388 | } |
|
|
1389 | |
|
|
1390 | $self->Crossfire::Protocol::send_mapinfo ($data, $cb); |
|
|
1391 | } |
|
|
1392 | |
|
|
1393 | # this method does a "flood fill" into every tile direction |
|
|
1394 | # it assumes that tiles are arranged in a rectangular grid, |
|
|
1395 | # i.e. a map is the same as the left of the right map etc. |
|
|
1396 | # failure to comply are harmless and result in display errors |
|
|
1397 | # at worst. |
|
|
1398 | sub conn::flood_fill { |
|
|
1399 | my ($self, $block, $gx, $gy, $path, $hash, $flags) = @_; |
|
|
1400 | |
|
|
1401 | # the server does not allow map paths > 6 |
|
|
1402 | return if 7 <= length $path; |
|
|
1403 | |
|
|
1404 | my ($x0, $y0, $x1, $y1) = @{$self->{neigh_rect}}; |
|
|
1405 | |
|
|
1406 | for ( |
|
|
1407 | [1, 3, 0, -1], |
|
|
1408 | [2, 4, 1, 0], |
|
|
1409 | [3, 1, 0, 1], |
|
|
1410 | [4, 2, -1, 0], |
|
|
1411 | ) { |
|
|
1412 | my ($tile, $tile2, $dx, $dy) = @$_; |
|
|
1413 | |
|
|
1414 | next if $block & (1 << $tile); |
|
|
1415 | my $block = $block | (1 << $tile2); |
|
|
1416 | |
|
|
1417 | my $gx = $gx + $dx; |
|
|
1418 | my $gy = $gy + $dy; |
|
|
1419 | |
|
|
1420 | next unless $flags & (1 << ($tile - 1)); |
|
|
1421 | next if $self->{neigh_grid}{$gx, $gy}++; |
|
|
1422 | |
|
|
1423 | my $neigh = $self->{neigh_map}{$hash} ||= []; |
|
|
1424 | if (my $info = $neigh->[$tile]) { |
|
|
1425 | my ($flags, $x, $y, $w, $h, $hash) = @$info; |
|
|
1426 | |
|
|
1427 | $self->flood_fill ($block, $gx, $gy, "$path$tile", $hash, $flags) |
|
|
1428 | if $x >= $x0 && $x + $w < $x1 && $y >= $y0 && $y + $h < $y1; |
|
|
1429 | |
|
|
1430 | } else { |
|
|
1431 | $self->send_mapinfo ("spatial $path$tile", sub { |
|
|
1432 | my ($mode, $flags, $x, $y, $w, $h, $hash) = @_; |
|
|
1433 | |
|
|
1434 | return if $mode ne "spatial"; |
|
|
1435 | |
|
|
1436 | $x += $MAP->ox; |
|
|
1437 | $y += $MAP->oy; |
|
|
1438 | |
|
|
1439 | $self->load_map ($hash, $x, $y) |
|
|
1440 | unless $self->{neigh_map}{$hash}[5]++;#d# |
|
|
1441 | |
|
|
1442 | $neigh->[$tile] = [$flags, $x, $y, $w, $h, $hash]; |
|
|
1443 | |
|
|
1444 | $self->flood_fill ($block, $gx, $gy, "$path$tile", $hash, $flags) |
|
|
1445 | if $x >= $x0 && $x + $w < $x1 && $y >= $y0 && $y + $h < $y1; |
|
|
1446 | }); |
|
|
1447 | } |
|
|
1448 | } |
|
|
1449 | } |
|
|
1450 | |
|
|
1451 | sub conn::map_change { |
|
|
1452 | my ($self, $mode, $flags, $x, $y, $w, $h, $hash) = @_; |
|
|
1453 | |
|
|
1454 | $self->flush_map; |
|
|
1455 | |
|
|
1456 | my ($ox, $oy) = ($::MAP->ox, $::MAP->oy); |
|
|
1457 | |
|
|
1458 | my $mapmapw = $MAPMAP->{w}; |
|
|
1459 | my $mapmaph = $MAPMAP->{h}; |
|
|
1460 | |
|
|
1461 | $self->{neigh_rect} = [ |
|
|
1462 | $ox - $mapmapw * 0.5, $oy - $mapmapw * 0.5, |
|
|
1463 | $ox + $mapmapw * 0.5 + $w, $oy + $mapmapw * 0.5 + $h, |
|
|
1464 | ]; |
|
|
1465 | |
|
|
1466 | delete $self->{neigh_grid}; |
|
|
1467 | |
|
|
1468 | $x += $ox; |
|
|
1469 | $y += $oy; |
|
|
1470 | |
|
|
1471 | $self->{map_info} = [$hash, $x, $y, $w, $h]; |
|
|
1472 | |
|
|
1473 | (my $map = $hash) =~ s/^.*?\/([^\/]+)$/\1/; |
|
|
1474 | $STATWIDS->{map}->set_text ("Map: " . $map); |
|
|
1475 | |
|
|
1476 | $self->load_map ($hash, $x, $y); |
|
|
1477 | $self->flood_fill (0, 0, 0, "", $hash, $flags); |
|
|
1478 | } |
|
|
1479 | |
|
|
1480 | sub conn::face_find { |
|
|
1481 | my ($self, $facenum, $face) = @_; |
|
|
1482 | |
|
|
1483 | my $hash = "$face->{chksum},$face->{name}"; |
|
|
1484 | |
|
|
1485 | my $id = $FACEMAP->get ($hash); |
|
|
1486 | |
|
|
1487 | unless ($id) { |
|
|
1488 | # create new id for face |
|
|
1489 | # I love transactions |
|
|
1490 | for (1..100) { |
|
|
1491 | my $txn = $CFClient::DB_ENV->txn_begin; |
|
|
1492 | my $status = $FACEMAP->db_get (id => $id, BerkeleyDB::DB_RMW); |
|
|
1493 | if ($status == 0 || $status == BerkeleyDB::DB_NOTFOUND) { |
|
|
1494 | $id = ($id || 16) + 1; |
|
|
1495 | if ($FACEMAP->put (id => $id) == 0 |
|
|
1496 | && $FACEMAP->put ($hash => $id) == 0) { |
|
|
1497 | $txn->txn_commit; |
|
|
1498 | |
|
|
1499 | goto gotid; |
|
|
1500 | } |
|
|
1501 | } |
|
|
1502 | $txn->abort; |
|
|
1503 | } |
|
|
1504 | |
|
|
1505 | CFClient::fatal "maximum number of transaction retries reached - database problems?"; |
|
|
1506 | } |
|
|
1507 | |
|
|
1508 | gotid: |
|
|
1509 | $face->{id} = $id; |
|
|
1510 | $MAP->set_face ($facenum => $id); |
|
|
1511 | $self->{faceid}[$facenum] = $id;#d# |
|
|
1512 | |
|
|
1513 | my $face = $TILECACHE->get ($id); |
|
|
1514 | |
|
|
1515 | if ($face) { |
|
|
1516 | #$self->face_prefetch; |
|
|
1517 | $face |
|
|
1518 | } else { |
|
|
1519 | my $tex = $self->{noface}; |
|
|
1520 | $MAP->set_texture ($id, @$tex{qw(name w h s t)}, @{$tex->{minified}}); |
|
|
1521 | undef |
|
|
1522 | }; |
|
|
1523 | } |
|
|
1524 | |
|
|
1525 | sub conn::face_update { |
|
|
1526 | my ($self, $facenum, $face) = @_; |
|
|
1527 | |
|
|
1528 | $TILECACHE->put ($face->{id} => $face->{image}); #TODO: try to avoid duplicate writes |
|
|
1529 | |
|
|
1530 | $self->set_texture ($face->{id} => delete $face->{image}); |
|
|
1531 | } |
|
|
1532 | |
|
|
1533 | sub conn::set_texture { |
|
|
1534 | my ($self, $id, $data) = @_; |
|
|
1535 | |
|
|
1536 | $self->{texture}[$id] ||= do { |
|
|
1537 | my $tex = |
|
|
1538 | new_from_image CFClient::Texture |
|
|
1539 | $data, minify => 1, mipmap => 1; |
|
|
1540 | |
|
|
1541 | $MAP->set_texture ($id, @$tex{qw(name w h s t)}, @{$tex->{minified}}); |
|
|
1542 | $MAPWIDGET->update; |
|
|
1543 | |
|
|
1544 | $tex |
|
|
1545 | }; |
|
|
1546 | } |
|
|
1547 | |
|
|
1548 | sub conn::sound_play { |
|
|
1549 | my ($self, $x, $y, $soundnum, $type) = @_; |
|
|
1550 | |
|
|
1551 | $SDL_MIXER |
|
|
1552 | or return; |
|
|
1553 | |
|
|
1554 | my $chunk = $AUDIO_CHUNKS{$SOUNDS[$soundnum]} |
|
|
1555 | or return; |
|
|
1556 | |
|
|
1557 | $chunk->play; |
|
|
1558 | # warn "sound $x,$y,$soundnum,$type\n";#d# |
|
|
1559 | } |
|
|
1560 | |
|
|
1561 | my $LAST_QUERY; # server is stupid, stupid, stupid |
|
|
1562 | |
|
|
1563 | sub conn::query { |
|
|
1564 | my ($self, $flags, $prompt) = @_; |
|
|
1565 | |
|
|
1566 | $prompt = $LAST_QUERY unless length $prompt; |
|
|
1567 | $LAST_QUERY = $prompt; |
|
|
1568 | |
|
|
1569 | my $dialog = new CFClient::UI::FancyFrame |
|
|
1570 | title => "Query", |
|
|
1571 | child => my $vbox = new CFClient::UI::VBox; |
|
|
1572 | |
|
|
1573 | $vbox->add (new CFClient::UI::Label |
|
|
1574 | max_w => $::WIDTH * 0.4, |
|
|
1575 | ellipsise => 0, |
|
|
1576 | text => $prompt); |
|
|
1577 | |
|
|
1578 | if ($flags & Crossfire::Protocol::CS_QUERY_YESNO) { |
|
|
1579 | $vbox->add (my $hbox = new CFClient::HBox); |
|
|
1580 | $hbox->add (new CFClient::Button |
|
|
1581 | text => "No", |
|
|
1582 | connect_activate => sub { |
|
|
1583 | $self->send ("reply n"); |
|
|
1584 | $dialog->destroy; |
|
|
1585 | $MAPWIDGET->focus_in; |
|
|
1586 | } |
|
|
1587 | ); |
|
|
1588 | $hbox->add (new CFClient::Button |
|
|
1589 | text => "Yes", |
|
|
1590 | connect_activate => sub { |
|
|
1591 | $self->send ("reply y"); |
|
|
1592 | $dialog->destroy; |
|
|
1593 | }, |
|
|
1594 | ); |
|
|
1595 | |
|
|
1596 | $dialog->focus_in; |
|
|
1597 | |
|
|
1598 | } elsif ($flags & Crossfire::Protocol::CS_QUERY_SINGLECHAR) { |
|
|
1599 | $dialog->{tooltip} = "Press a key (click on the entry to make sure it has keyboard focus)"; |
|
|
1600 | $vbox->add (my $entry = new CFClient::UI::Entry |
|
|
1601 | connect_changed => sub { |
|
|
1602 | $self->send ("reply $_[1]"); |
|
|
1603 | $dialog->destroy; |
|
|
1604 | }, |
|
|
1605 | ); |
|
|
1606 | |
|
|
1607 | $entry->focus_in; |
|
|
1608 | |
|
|
1609 | } else { |
|
|
1610 | $dialog->{tooltip} = "Enter the reply and press return (click on the entry to make sure it has keyboard focus)"; |
|
|
1611 | |
|
|
1612 | $vbox->add (my $entry = new CFClient::UI::Entry |
|
|
1613 | $flags & Crossfire::Protocol::CS_QUERY_HIDEINPUT ? (hiddenchar => "*") : (), |
|
|
1614 | connect_activate => sub { |
|
|
1615 | $self->send ("reply $_[1]"); |
|
|
1616 | $dialog->destroy; |
|
|
1617 | }, |
|
|
1618 | ); |
|
|
1619 | |
|
|
1620 | $entry->focus_in; |
|
|
1621 | } |
|
|
1622 | |
|
|
1623 | $dialog->show_centered; |
|
|
1624 | } |
|
|
1625 | |
|
|
1626 | sub conn::drawinfo { |
|
|
1627 | my ($self, $color, $text) = @_; |
|
|
1628 | |
|
|
1629 | my @color = ( |
|
|
1630 | [1.00, 1.00, 1.00], #[0.00, 0.00, 0.00], |
|
|
1631 | [1.00, 1.00, 1.00], |
|
|
1632 | [0.50, 0.50, 1.00], #[0.00, 0.00, 0.55] |
|
|
1633 | [1.00, 0.00, 0.00], |
|
|
1634 | [1.00, 0.54, 0.00], |
|
|
1635 | [0.11, 0.56, 1.00], |
|
|
1636 | [0.93, 0.46, 0.00], |
|
|
1637 | [0.18, 0.54, 0.34], |
|
|
1638 | [0.56, 0.73, 0.56], |
|
|
1639 | [0.80, 0.80, 0.80], |
|
|
1640 | [0.55, 0.41, 0.13], |
|
|
1641 | [0.99, 0.77, 0.26], |
|
|
1642 | [0.74, 0.65, 0.41], |
|
|
1643 | ); |
|
|
1644 | |
|
|
1645 | my $time = sprintf "%02d:%02d:%02d", (localtime time)[2,1,0]; |
|
|
1646 | |
|
|
1647 | $text = CFClient::UI::Label::escape $text; |
|
|
1648 | $text =~ s/\[b\](.*?)\[\/b\]/<b>\1<\/b>/g; |
|
|
1649 | $text =~ s/\[color=(.*?)\](.*?)\[\/color\]/<span foreground='\1'>\2<\/span>/g; |
|
|
1650 | |
|
|
1651 | $LOGVIEW->add_paragraph ($color[$color], |
|
|
1652 | join "\n", map "$time $_", split /\n/, $text); |
|
|
1653 | |
|
|
1654 | $STATUSBOX->add ($text, |
|
|
1655 | group => $text, |
|
|
1656 | fg => $color[$color], |
|
|
1657 | timeout => 10, |
|
|
1658 | tooltip_font => $::FONT_FIXED, |
|
|
1659 | ); |
|
|
1660 | } |
|
|
1661 | |
|
|
1662 | sub conn::drawextinfo { |
|
|
1663 | my ($self, $color, $type, $subtype, $message) = @_; |
|
|
1664 | |
|
|
1665 | $self->drawinfo ($color, $message); |
|
|
1666 | } |
|
|
1667 | |
|
|
1668 | sub conn::spell_add { |
|
|
1669 | my ($self, $spell) = @_; |
|
|
1670 | |
|
|
1671 | # TODO |
|
|
1672 | # create a widget dynamically, using spell face (CF::Protocol downloads them) |
|
|
1673 | $MAPWIDGET->add_command ("invoke $spell->{name}", CFClient::UI::Label::escape $spell->{message}); |
|
|
1674 | $MAPWIDGET->add_command ("cast $spell->{name}", CFClient::UI::Label::escape $spell->{message}); |
|
|
1675 | } |
|
|
1676 | |
|
|
1677 | sub conn::spell_delete { |
|
|
1678 | my ($self, $spell) = @_; |
|
|
1679 | } |
|
|
1680 | |
|
|
1681 | sub conn::addme_success { |
|
|
1682 | my ($self) = @_; |
|
|
1683 | |
|
|
1684 | $self->send ("command output-sync $CFG->{output_sync}"); |
|
|
1685 | $self->send ("command output-count $CFG->{output_count}"); |
|
|
1686 | |
|
|
1687 | my $parser = new Pod::POM; |
|
|
1688 | my $pod = $parser->parse_file (CFClient::find_rcfile "pod/skill_help.pod"); |
|
|
1689 | |
|
|
1690 | my %skill_tooltip; |
|
|
1691 | |
|
|
1692 | for my $head2 ($pod->head2) { |
|
|
1693 | $skill_tooltip{$head2->title} = CFClient::pod_to_pango $head2->content; |
|
|
1694 | } |
|
|
1695 | |
|
|
1696 | for my $skill (values %{$self->{skill_info}}) { |
|
|
1697 | $MAPWIDGET->add_command ("ready_skill $skill", |
|
|
1698 | (CFClient::UI::Label::escape "Ready the skill '$skill'\n\n") |
|
|
1699 | . $skill_tooltip{$skill}); |
|
|
1700 | $MAPWIDGET->add_command ("use_skill $skill", |
|
|
1701 | (CFClient::UI::Label::escape "Immediately use the skill '$skill'\n\n") |
|
|
1702 | . $skill_tooltip{$skill}); |
|
|
1703 | } |
|
|
1704 | } |
|
|
1705 | |
|
|
1706 | sub conn::eof { |
|
|
1707 | $MAPWIDGET->clr_commands; |
|
|
1708 | |
|
|
1709 | stop_game; |
|
|
1710 | } |
|
|
1711 | |
|
|
1712 | sub conn::image_info { |
|
|
1713 | my ($self, $numfaces) = @_; |
|
|
1714 | |
|
|
1715 | $self->{num_faces} = $numfaces; |
|
|
1716 | $self->{face_prefetch} = [1 .. $numfaces]; |
|
|
1717 | $self->face_prefetch; |
|
|
1718 | } |
|
|
1719 | |
|
|
1720 | sub conn::face_prefetch { |
|
|
1721 | my ($self) = @_; |
|
|
1722 | |
|
|
1723 | return unless $CFG->{face_prefetch}; |
|
|
1724 | |
|
|
1725 | if ($self->{num_faces}) { |
|
|
1726 | return if @{ $self->{send_queue} || [] }; |
|
|
1727 | my $todo = @{ $self->{face_prefetch} } |
|
|
1728 | or return; |
|
|
1729 | |
|
|
1730 | my ($face) = splice @{ $self->{face_prefetch} }, + rand @{ $self->{face_prefetch} }, 1, (); |
|
|
1731 | |
|
|
1732 | $self->send ("requestinfo image_sums $face $face"); |
|
|
1733 | |
|
|
1734 | $STATUSBOX->add (CFClient::UI::Label::escape "prefetching $todo", |
|
|
1735 | group => "prefetch", timeout => 2, fg => [1, 1, 0, 0.5]); |
|
|
1736 | } elsif (!exists $self->{num_faces}) { |
|
|
1737 | $self->send ("requestinfo image_info"); |
|
|
1738 | |
|
|
1739 | $self->{num_faces} = 0; |
|
|
1740 | |
|
|
1741 | $STATUSBOX->add (CFClient::UI::Label::escape "starting to prefetch", |
|
|
1742 | group => "prefetch", timeout => 2, fg => [1, 1, 0, 0.5]); |
|
|
1743 | } |
|
|
1744 | } |
|
|
1745 | |
|
|
1746 | # check once/second for faces that need to be prefetched |
1920 | # check once/second for faces that need to be prefetched |
1747 | # this should, of course, only run on demand, but |
1921 | # this should, of course, only run on demand, but |
1748 | # SDL forces worse things on us.... |
1922 | # SDL forces worse things on us.... |
1749 | |
1923 | |
1750 | Event->timer (after => 1, interval => 0.25, cb => sub { |
1924 | Event->timer (after => 1, interval => 0.25, cb => sub { |
1751 | $CONN->face_prefetch |
1925 | $CONN->face_prefetch |
1752 | if $CONN; |
1926 | if $CONN; |
1753 | }); |
1927 | }); |
1754 | |
|
|
1755 | sub update_floorbox { |
|
|
1756 | $CFClient::UI::ROOT->on_refresh ($FLOORBOX => sub { |
|
|
1757 | return unless $CONN; |
|
|
1758 | |
|
|
1759 | $FLOORBOX->clear; |
|
|
1760 | $FLOORBOX->add (0, 1, new CFClient::UI::Empty expand => 1); |
|
|
1761 | |
|
|
1762 | my $row; |
|
|
1763 | for (@{ $CONN->{container}{0} }) { |
|
|
1764 | if (++$row < 7) { |
|
|
1765 | local $_->{face_widget}; # hack to force recreation of widget |
|
|
1766 | local $_->{desc_widget}; # hack to force recreation of widget |
|
|
1767 | CFClient::Item::update_widgets $_; |
|
|
1768 | |
|
|
1769 | $FLOORBOX->add (0, $row, $_->{face_widget}); |
|
|
1770 | $FLOORBOX->add (1, $row, $_->{desc_widget}); |
|
|
1771 | } else { |
|
|
1772 | $FLOORBOX->add (1, ++$row, new CFClient::UI::Label text => "More..."); |
|
|
1773 | last; |
|
|
1774 | } |
|
|
1775 | } |
|
|
1776 | }); |
|
|
1777 | |
|
|
1778 | $WANT_REFRESH++; |
|
|
1779 | } |
|
|
1780 | |
|
|
1781 | sub set_opencont { |
|
|
1782 | my ($conn, $tag, $name) = @_; |
|
|
1783 | $conn->{open_container} = $tag; |
|
|
1784 | $INVR_LBL->set_text ($name); |
|
|
1785 | $INVR->set_items ($conn->{container}{$tag}); |
|
|
1786 | } |
|
|
1787 | |
|
|
1788 | sub update_container { |
|
|
1789 | my ($tag) = @_; |
|
|
1790 | $INVR->set_items ($::CONN->{container}{$CONN->{open_container}}) |
|
|
1791 | if $tag == $CONN->{open_container}; |
|
|
1792 | } |
|
|
1793 | |
|
|
1794 | sub conn::container_add { |
|
|
1795 | my ($self, $tag, $items) = @_; |
|
|
1796 | |
|
|
1797 | #d# print "container_add: container $tag ($self->{player}{tag})\n"; |
|
|
1798 | |
|
|
1799 | if ($tag == 0) { |
|
|
1800 | update_floorbox; |
|
|
1801 | update_container (0); |
|
|
1802 | } elsif ($tag == $self->{player}{tag}) { |
|
|
1803 | $INV->set_items ($self->{container}{$self->{player}{tag}}) |
|
|
1804 | } else { |
|
|
1805 | update_container ($tag); |
|
|
1806 | } |
|
|
1807 | |
|
|
1808 | # $self-<{player}{tag} => player inv |
|
|
1809 | #use PApp::Util; warn PApp::Util::dumpval $self->{container}{$self->{player}{tag}}; |
|
|
1810 | } |
|
|
1811 | |
|
|
1812 | sub conn::container_clear { |
|
|
1813 | my ($self, $tag) = @_; |
|
|
1814 | |
|
|
1815 | #d# print "container_clear: container $tag ($self->{player}{tag})\n"; |
|
|
1816 | |
|
|
1817 | if ($tag == 0) { |
|
|
1818 | update_floorbox; |
|
|
1819 | update_container (0); |
|
|
1820 | } elsif ($tag == $self->{player}{tag}) { |
|
|
1821 | $INV->set_items ($self->{container}{$tag}) |
|
|
1822 | } |
|
|
1823 | |
|
|
1824 | # use PApp::Util; warn PApp::Util::dumpval $self->{container}{0}; |
|
|
1825 | } |
|
|
1826 | |
|
|
1827 | sub conn::item_delete { |
|
|
1828 | my ($self, @items) = @_; |
|
|
1829 | |
|
|
1830 | for (@items) { |
|
|
1831 | #d# print "item_delete: $_->{tag} from $_->{container} ($self->{player}{tag})\n"; |
|
|
1832 | |
|
|
1833 | if ($_->{container} == 0) { |
|
|
1834 | update_floorbox; |
|
|
1835 | update_container ($_->{tag}); |
|
|
1836 | } elsif ($_->{container} == $self->{player}{tag}) { |
|
|
1837 | $INV->set_items ($self->{container}{$self->{player}{tag}}) |
|
|
1838 | } else { |
|
|
1839 | update_container ($_->{tag}); |
|
|
1840 | } |
|
|
1841 | } |
|
|
1842 | } |
|
|
1843 | |
|
|
1844 | sub conn::item_update { |
|
|
1845 | my ($self, $item) = @_; |
|
|
1846 | |
|
|
1847 | #d# print "item_update: $item->{tag} in $item->{container} ($self->{player}{tag}) ($CONN->{open_container})\n"; |
|
|
1848 | |
|
|
1849 | if ($item->{tag} == $self->{player}{tag}) { |
|
|
1850 | $STATWIDS->{weight}->set_text (sprintf "Weight: %.1fkg", $item->{weight} / 1000); |
|
|
1851 | return |
|
|
1852 | } |
|
|
1853 | |
|
|
1854 | CFClient::Item::update_widgets $item; |
|
|
1855 | |
|
|
1856 | if ($item->{tag} == $CONN->{open_container} && not ($item->{flags} & Crossfire::Protocol::F_OPEN)) { |
|
|
1857 | set_opencont ($CONN, 0, "Floor"); |
|
|
1858 | |
|
|
1859 | } elsif ($item->{flags} & Crossfire::Protocol::F_OPEN) { |
|
|
1860 | set_opencont ($CONN, $item->{tag}, CFClient::Item::desc_string $item); |
|
|
1861 | } else { |
|
|
1862 | if ($item->{container} == 0) { |
|
|
1863 | update_floorbox; |
|
|
1864 | update_container (0); |
|
|
1865 | } elsif ($item->{container} == $self->{player}{tag}) { |
|
|
1866 | $INV->set_items ($self->{container}{$item->{container}}) |
|
|
1867 | } |
|
|
1868 | } |
|
|
1869 | } |
|
|
1870 | |
1928 | |
1871 | %SDL_CB = ( |
1929 | %SDL_CB = ( |
1872 | CFClient::SDL_QUIT => sub { |
1930 | CFClient::SDL_QUIT => sub { |
1873 | Event::unloop -1; |
1931 | Event::unloop -1; |
1874 | }, |
1932 | }, |
… | |
… | |
1881 | # printf "active %x %x\n", $SDL_EV->active_gain, $SDL_EV->active_state;#d# |
1939 | # printf "active %x %x\n", $SDL_EV->active_gain, $SDL_EV->active_state;#d# |
1882 | }, |
1940 | }, |
1883 | CFClient::SDL_KEYDOWN => sub { |
1941 | CFClient::SDL_KEYDOWN => sub { |
1884 | if ($_[0]{mod} & CFClient::KMOD_ALT && $_[0]{sym} == 13) { |
1942 | if ($_[0]{mod} & CFClient::KMOD_ALT && $_[0]{sym} == 13) { |
1885 | # alt-enter |
1943 | # alt-enter |
|
|
1944 | $FULLSCREEN_ENABLE->toggle; |
1886 | video_shutdown; |
1945 | video_shutdown; |
1887 | $CFG->{fullscreen} = !$CFG->{fullscreen}; |
|
|
1888 | video_init; |
1946 | video_init; |
1889 | } else { |
1947 | } else { |
1890 | CFClient::UI::feed_sdl_key_down_event ($_[0]); |
1948 | CFClient::UI::feed_sdl_key_down_event ($_[0]); |
1891 | } |
1949 | } |
1892 | }, |
1950 | }, |
… | |
… | |
1906 | ############################################################################# |
1964 | ############################################################################# |
1907 | |
1965 | |
1908 | $SIG{INT} = $SIG{TERM} = sub { exit }; |
1966 | $SIG{INT} = $SIG{TERM} = sub { exit }; |
1909 | |
1967 | |
1910 | { |
1968 | { |
1911 | local $SIG{__DIE__} = sub { CFClient::fatal $_[0] }; |
1969 | local $SIG{__DIE__} = sub { |
|
|
1970 | return unless defined $^S && !$^S; |
|
|
1971 | Carp::confess $_[0];#d#TODO: remove when stable |
|
|
1972 | CFClient::fatal $_[0]; |
|
|
1973 | }; |
1912 | |
1974 | |
1913 | CFClient::read_cfg "$Crossfire::VARDIR/pclientrc"; |
1975 | CFClient::read_cfg "$Crossfire::VARDIR/cfplusrc"; |
1914 | |
1976 | CFClient::UI::set_layout ($::CFG->{layout}); |
1915 | $TILECACHE = CFClient::db_table "tilecache"; |
|
|
1916 | $FACEMAP = CFClient::db_table "facemap"; |
|
|
1917 | |
1977 | |
1918 | my %DEF_CFG = ( |
1978 | my %DEF_CFG = ( |
1919 | sdl_mode => 0, |
1979 | sdl_mode => 0, |
1920 | width => 640, |
1980 | width => 640, |
1921 | height => 480, |
1981 | height => 480, |
1922 | fullscreen => 0, |
1982 | fullscreen => 0, |
1923 | fast => 0, |
1983 | fast => 0, |
1924 | map_scale => 1, |
1984 | map_scale => 1, |
1925 | fow_enable => 1, |
1985 | fow_enable => 1, |
1926 | fow_intensity => 0.45, |
1986 | fow_intensity => 0.45, |
1927 | fow_smooth => 0, |
1987 | fow_smooth => 0, |
1928 | gui_fontsize => 1, |
1988 | gui_fontsize => 1, |
1929 | log_fontsize => 1, |
1989 | log_fontsize => 0.7, |
1930 | gauge_fontsize=> 1, |
1990 | gauge_fontsize => 1, |
1931 | gauge_size => 0.35, |
1991 | gauge_size => 0.35, |
1932 | stat_fontsize => 1, |
1992 | stat_fontsize => 0.7, |
1933 | mapsize => 100, |
1993 | mapsize => 100, |
1934 | host => "crossfire.schmorp.de", |
|
|
1935 | say_command => 'say', |
1994 | say_command => 'say', |
1936 | audio_enable => 1, |
1995 | audio_enable => 1, |
1937 | bgm_enable => 1, |
1996 | bgm_enable => 1, |
1938 | bgm_volume => 0.25, |
1997 | bgm_volume => 0.25, |
1939 | face_prefetch => 0, |
1998 | face_prefetch => 0, |
1940 | output_sync => 1, |
1999 | output_sync => 1, |
1941 | output_count => 1, |
2000 | output_count => 1, |
|
|
2001 | pickup => 0, |
|
|
2002 | inv_sort => "mtime", |
|
|
2003 | default => "profile", # default profile |
|
|
2004 | ); |
1942 | ); |
2005 | |
1943 | |
|
|
1944 | while (my ($k, $v) = each %DEF_CFG) { |
2006 | while (my ($k, $v) = each %DEF_CFG) { |
1945 | $CFG->{$k} = $v unless exists $CFG->{$k}; |
2007 | $CFG->{$k} = $v unless exists $CFG->{$k}; |
1946 | } |
2008 | } |
|
|
2009 | |
|
|
2010 | $CFG->{profile}{default}{host} ||= "crossfire.schmorp.de"; |
1947 | |
2011 | |
1948 | sdl_init; |
2012 | sdl_init; |
1949 | |
2013 | |
1950 | @SDL_MODES = reverse |
2014 | @SDL_MODES = reverse |
1951 | grep $_->[0] >= 640 && $_->[1] >= 480, |
2015 | grep $_->[0] >= 640 && $_->[1] >= 480, |
… | |
… | |
1993 | video_init; |
2057 | video_init; |
1994 | audio_init; |
2058 | audio_init; |
1995 | } |
2059 | } |
1996 | |
2060 | |
1997 | Event::loop; |
2061 | Event::loop; |
|
|
2062 | #CFClient::SDL_Quit; |
|
|
2063 | #CFClient::_exit 0; |
1998 | |
2064 | |
1999 | END { CFClient::SDL_Quit } |
2065 | END { CFClient::SDL_Quit } |
2000 | |
2066 | |
2001 | =head1 NAME |
2067 | =head1 NAME |
2002 | |
2068 | |
2003 | pclient - A Crossfire+ and Crossfire game client |
2069 | cfplus - A Crossfire+ and Crossfire game client |
2004 | |
2070 | |
2005 | =head1 SYNOPSIS |
2071 | =head1 SYNOPSIS |
2006 | |
2072 | |
2007 | Just run it - no commandline arguments are supported. |
2073 | Just run it - no commandline arguments are supported. |
2008 | |
2074 | |
2009 | =head1 USAGE |
2075 | =head1 USAGE |
2010 | |
2076 | |
2011 | Pclient utilises OpenGL for all UI elements and the game. It is supposed to be used |
2077 | cfplus utilises OpenGL for all UI elements and the game. It is supposed to be used |
2012 | fullscreen and interactively. |
2078 | fullscreen and interactively. |
2013 | |
2079 | |
|
|
2080 | =head1 DEBUGGING |
|
|
2081 | |
|
|
2082 | |
|
|
2083 | CFPLUS_DEBUG - environment variable |
|
|
2084 | |
|
|
2085 | 1 draw borders around widgets |
|
|
2086 | 2 add low-level widget info to tooltips |
|
|
2087 | 4 show fps |
|
|
2088 | 8 suppress tooltips |
|
|
2089 | |
2014 | =head1 AUTHOR |
2090 | =head1 AUTHOR |
2015 | |
2091 | |
2016 | Marc Lehmann <crossfire@schmorp.de>, Robin Redeker <elmex@ta-sa.org> |
2092 | Marc Lehmann <crossfire@schmorp.de>, Robin Redeker <elmex@ta-sa.org> |
2017 | |
2093 | |
2018 | |
2094 | |