1 | #!/opt/bin/perl |
1 | #!/opt/bin/perl |
|
|
2 | |
|
|
3 | my $startup_done = sub { }; |
|
|
4 | our $PANGO = "1.5.0"; |
|
|
5 | |
|
|
6 | # do splash-screen thingy on win32 |
|
|
7 | BEGIN { |
|
|
8 | if (%PAR::LibCache && $^O eq "MSWin32") { |
|
|
9 | while (my ($filename, $zip) = each %PAR::LibCache) { |
|
|
10 | $zip->extractMember ("SPLASH.bmp", "$ENV{PAR_TEMP}/SPLASH.bmp"); |
|
|
11 | } |
|
|
12 | |
|
|
13 | require Win32::GUI::SplashScreen; |
|
|
14 | |
|
|
15 | Win32::GUI::SplashScreen::Show ( |
|
|
16 | -file => "$ENV{PAR_TEMP}/SPLASH.bmp", |
|
|
17 | ); |
|
|
18 | |
|
|
19 | $startup_done = sub { |
|
|
20 | Win32::GUI::SplashScreen::Done (1); |
|
|
21 | }; |
|
|
22 | } |
|
|
23 | } |
2 | |
24 | |
3 | use strict; |
25 | use strict; |
4 | use utf8; |
26 | use utf8; |
|
|
27 | |
|
|
28 | use Carp 'verbose'; |
5 | |
29 | |
6 | # do things only needed for single-binary version (par) |
30 | # do things only needed for single-binary version (par) |
7 | BEGIN { |
31 | BEGIN { |
8 | if (%PAR::LibCache) { |
32 | if (%PAR::LibCache) { |
9 | @INC = grep ref, @INC; # weed out all paths except pars loader refs |
33 | @INC = grep ref, @INC; # weed out all paths except pars loader refs |
10 | |
34 | |
|
|
35 | my $tmp = $ENV{PAR_TEMP}; |
|
|
36 | |
11 | while (my ($filename, $zip) = each %PAR::LibCache) { |
37 | while (my ($filename, $zip) = each %PAR::LibCache) { |
12 | for ($zip->memberNames) { |
38 | for ($zip->memberNames) { |
13 | next unless /^\/root\/(.*)/; |
39 | next unless /^root\/(.*)/; |
14 | $zip->extractMember ($_, "$ENV{PAR_TEMP}/$1") |
40 | $zip->extractMember ($_, "$tmp/$1") |
15 | unless -e "$ENV{PAR_TEMP}/$1"; |
41 | unless -e "$tmp/$1"; |
16 | } |
42 | } |
17 | } |
43 | } |
18 | |
44 | |
19 | # TODO: pango-rc file, anybody? |
45 | if ($^O eq "MSWin32") { |
|
|
46 | # relocatable |
|
|
47 | } else { |
|
|
48 | # unix, need to patch pango rc file |
|
|
49 | open my $fh, "<:perlio", "$tmp/usr/lib/pango/$PANGO/module-files.d/libpango1.0-0.modules" |
|
|
50 | or die "$tmp/usr/lib/$PANGO/module-files.d/libpango1.0-0.modules: $!"; |
|
|
51 | local $/; |
|
|
52 | my $rc = <$fh>; |
|
|
53 | $rc =~ s/^\//$tmp\//gm; # replace abs paths by relative ones |
20 | |
54 | |
21 | unshift @INC, $ENV{PAR_TEMP}; |
55 | mkdir "$tmp/pango-modules"; |
|
|
56 | open my $fh, ">:perlio", "$tmp/pango-modules/pango.modules" |
|
|
57 | or die "$tmp/pango-modules/pango.modules: $!"; |
|
|
58 | print $fh $rc; |
|
|
59 | |
|
|
60 | $ENV{PANGO_RC_FILE} = "$tmp/pango.rc"; |
|
|
61 | open my $fh, ">:perlio", $ENV{PANGO_RC_FILE} |
|
|
62 | or die "$ENV{PANGO_RC_FILE}: $!"; |
|
|
63 | print $fh "[Pango]\nModuleFiles = $tmp/pango-modules\n"; |
|
|
64 | } |
|
|
65 | |
|
|
66 | unshift @INC, $tmp; |
22 | } |
67 | } |
23 | } |
68 | } |
24 | |
69 | |
25 | # need to do it again because that pile of garbage called PAR nukes it before main |
70 | # need to do it again because that pile of garbage called PAR nukes it before main |
26 | unshift @INC, $ENV{PAR_TEMP} |
71 | unshift @INC, $ENV{PAR_TEMP} |
27 | if %PAR::LibCache; |
72 | if %PAR::LibCache; |
28 | |
73 | |
29 | use Time::HiRes 'time'; |
74 | use Time::HiRes 'time'; |
30 | use Pod::POM; |
|
|
31 | use Event; |
75 | use Event; |
|
|
76 | use List::Util qw(max min); |
32 | |
77 | |
33 | use Crossfire; |
78 | use Crossfire; |
34 | use Crossfire::Protocol::Constants; |
79 | use Crossfire::Protocol::Constants; |
35 | |
80 | |
36 | use Compress::LZF; |
81 | use Compress::LZF; |
37 | |
82 | |
38 | use CFClient; |
83 | use CFPlus; |
39 | use CFClient::OpenGL (); |
84 | use CFPlus::OpenGL (); |
40 | use CFClient::Protocol; |
85 | use CFPlus::Protocol; |
|
|
86 | use CFPlus::DB; |
41 | use CFClient::UI; |
87 | use CFPlus::UI; |
|
|
88 | use CFPlus::UI::Canvas; |
|
|
89 | use CFPlus::UI::Inventory; |
|
|
90 | use CFPlus::UI::SpellList; |
|
|
91 | use CFPlus::UI::MessageWindow; |
|
|
92 | use CFPlus::Pod; |
42 | use CFClient::MapWidget; |
93 | use CFPlus::MapWidget; |
|
|
94 | use CFPlus::Macro; |
43 | |
95 | |
|
|
96 | $SIG{QUIT} = sub { Carp::cluck "QUIT" }; |
|
|
97 | $SIG{PIPE} = 'IGNORE'; |
|
|
98 | |
|
|
99 | $Event::Eval = 1; |
44 | $Event::DIED = sub { |
100 | $Event::DIED = sub { |
45 | # TODO: display dialog box or so |
101 | CFPlus::fatal Carp::longmess $_[1] |
46 | Carp::confess $_[1];#d#TODO: remove when stable |
|
|
47 | CFClient::error $_[1]; |
|
|
48 | }; |
102 | }; |
49 | |
|
|
50 | #$SIG{__WARN__} = sub { Carp::cluck $_[0] };#d# |
|
|
51 | |
|
|
52 | our $VERSION = '0.1'; |
|
|
53 | |
103 | |
54 | my $MAX_FPS = 60; |
104 | my $MAX_FPS = 60; |
55 | my $MIN_FPS = 5; # unused as of yet |
105 | my $MIN_FPS = 5; # unused as of yet |
56 | |
106 | |
57 | our $META_SERVER = "crossfire.real-time.com:13326"; |
107 | our $META_SERVER = "http://metaserver.schmorp.de/current.json"; |
58 | |
108 | |
59 | our $LAST_REFRESH; |
109 | our $LAST_REFRESH; |
60 | our $NOW; |
110 | our $NOW; |
61 | |
111 | |
62 | our $CFG; |
112 | our $CFG; |
63 | our $CONN; |
113 | our $CONN; |
|
|
114 | our $PROFILE; # current profile |
64 | our $FAST; # fast, low-quality mode, possibly useful for software-rendering |
115 | our $FAST; # fast, low-quality mode, possibly useful for software-rendering |
65 | |
116 | |
66 | our $WANT_REFRESH; |
117 | our $WANT_REFRESH; |
67 | our $CAN_REFRESH; |
|
|
68 | |
118 | |
69 | our @SDL_MODES; |
119 | our @SDL_MODES; |
70 | our $WIDTH; |
120 | our $WIDTH; |
71 | our $HEIGHT; |
121 | our $HEIGHT; |
72 | our $FULLSCREEN; |
122 | our $FULLSCREEN; |
… | |
… | |
77 | |
127 | |
78 | our $MAP; |
128 | our $MAP; |
79 | our $MAPMAP; |
129 | our $MAPMAP; |
80 | our $MAPWIDGET; |
130 | our $MAPWIDGET; |
81 | our $BUTTONBAR; |
131 | our $BUTTONBAR; |
82 | our $LOGVIEW; |
|
|
83 | our $CONSOLE; |
|
|
84 | our $METASERVER; |
132 | our $METASERVER; |
85 | our $LOGIN_BUTTON; |
133 | our $LOGIN_BUTTON; |
86 | our $QUIT_DIALOG; |
134 | our $QUIT_DIALOG; |
87 | our $HOST_ENTRY; |
135 | our $HOST_ENTRY; |
|
|
136 | our $FULLSCREEN_ENABLE; |
|
|
137 | our $PICKUP_ENABLE; |
|
|
138 | our $SERVER_INFO; |
88 | |
139 | |
89 | our $SETUP_DIALOG; |
140 | our $SETUP_DIALOG; |
90 | our $SETUP_NOTEBOOK; |
141 | our $SETUP_NOTEBOOK; |
91 | our $SETUP_SERVER; |
142 | our $SETUP_SERVER; |
92 | our $SETUP_KEYBOARD; |
143 | our $SETUP_KEYBOARD; |
93 | our $SETUP_SPELLS; |
|
|
94 | |
144 | |
|
|
145 | our $PL_NOTEBOOK; |
|
|
146 | our $PL_WINDOW; |
|
|
147 | |
|
|
148 | our $MUSIC_PLAYING_WIDGET; |
|
|
149 | our $LICENSE_WIDGET; |
|
|
150 | |
|
|
151 | our $INVENTORY_PAGE; |
|
|
152 | our $STATS_PAGE; |
|
|
153 | our $SKILL_PAGE; |
|
|
154 | our $SPELL_PAGE; |
|
|
155 | our $SPELL_LIST; |
|
|
156 | |
|
|
157 | our $HELP_WINDOW; |
|
|
158 | our $MESSAGE_WINDOW; |
95 | our $FLOORBOX; |
159 | our $FLOORBOX; |
96 | our $GAUGES; |
160 | our $GAUGES; |
97 | our $STATWIDS; |
161 | our $STATWIDS; |
98 | |
162 | |
99 | our $SDL_ACTIVE; |
163 | our $SDL_ACTIVE; |
100 | our %SDL_CB; |
164 | our %SDL_CB; |
101 | |
165 | |
102 | our $SDL_MIXER; |
|
|
103 | our @SOUNDS; # event => file mapping |
|
|
104 | our %AUDIO_CHUNKS; # audio files |
|
|
105 | |
|
|
106 | our $ALT_ENTER_MESSAGE; |
166 | our $ALT_ENTER_MESSAGE; |
107 | our $STATUSBOX; |
167 | our $STATUSBOX; |
108 | our $DEBUG_STATUS; |
168 | our $DEBUG_STATUS; |
109 | |
169 | |
110 | our $INV_WINDOW; |
|
|
111 | our $INV; |
170 | our $INV; |
112 | our $INVR; |
171 | our $INVR; |
113 | our $INV_RIGHT_HB; |
172 | our $INV_RIGHT_HB; |
114 | |
173 | |
115 | our $BIND_EDITOR; |
|
|
116 | |
|
|
117 | our $PICKUP_CFG; |
174 | our $PICKUP_CFG; |
118 | |
175 | |
|
|
176 | ############################################################################# |
|
|
177 | |
119 | sub status { |
178 | sub status { |
120 | $STATUSBOX->add (CFClient::UI::Label::escape $_[0], pri => -10, group => "status", timeout => 10, fg => [1, 1, 0, 1]); |
179 | $STATUSBOX->add (CFPlus::asxml $_[0], pri => -10, group => "status", timeout => 10, fg => [1, 1, 0, 1]); |
121 | } |
180 | } |
122 | |
181 | |
123 | sub debug { |
182 | sub debug { |
124 | $DEBUG_STATUS->set_text ($_[0]); |
183 | $DEBUG_STATUS->set_text ($_[0]); |
125 | } |
184 | } |
126 | |
185 | |
|
|
186 | sub message { |
|
|
187 | my ($para) = @_; |
|
|
188 | $MESSAGE_WINDOW->message ($para); |
|
|
189 | } |
|
|
190 | |
|
|
191 | ############################################################################# |
|
|
192 | #TODO: maybe move into own audio module... |
|
|
193 | |
|
|
194 | our $SDL_MIXER; |
|
|
195 | |
|
|
196 | our $MUSIC_DEFAULT = "in_a_heartbeat.ogg"; |
|
|
197 | our $MUSIC_WANT; # arryref of ambient music we want to play |
|
|
198 | our @MUSIC_HAVE; # ambient music we have on disk |
|
|
199 | our $MUSIC_START; |
|
|
200 | our @MUSIC_JINGLE; # which jingles to play next |
|
|
201 | our $MUSIC_PLAYING_DATA; |
|
|
202 | our $MUSIC_PLAYING_META; |
|
|
203 | our $MUSIC_PLAYER; |
|
|
204 | our $MUSIC_RESUME = 30; # resume music when played less than these many seconds before |
|
|
205 | |
|
|
206 | our %AUDIO_CHUNK; # audio "files" |
|
|
207 | our %AUDIO_PLAY; # which audio faces should be played |
|
|
208 | |
|
|
209 | sub audio_channel_finished { |
|
|
210 | my ($channel) = @_; |
|
|
211 | |
|
|
212 | # warn "channel $channel finished\n";#d# |
|
|
213 | } |
|
|
214 | |
|
|
215 | sub audio_sound_push($) { |
|
|
216 | my ($face) = @_; |
|
|
217 | |
|
|
218 | $CFG->{effects_enable} |
|
|
219 | or return; |
|
|
220 | |
|
|
221 | $AUDIO_PLAY{$face} |
|
|
222 | or return; |
|
|
223 | |
|
|
224 | if (my $chunk = $AUDIO_CHUNK{$face}) { |
|
|
225 | for (grep $_->[0] >= Event::time, @{(delete $AUDIO_PLAY{$face}) || []}) { |
|
|
226 | my (undef, $dx, $dy, $vol) = @$_; |
|
|
227 | |
|
|
228 | my $channel = CFPlus::Channel::find; |
|
|
229 | $channel->volume ($vol * $CFG->{effects_volume} * 128 / 255); |
|
|
230 | $channel->set_position_r ($dx, $dy, 20); |
|
|
231 | $chunk->play ($channel); |
|
|
232 | } |
|
|
233 | } else { |
|
|
234 | # sound_meta not set means data is in flight either way |
|
|
235 | my $meta = $CONN->{sound_meta}{$face} |
|
|
236 | or return; |
|
|
237 | |
|
|
238 | # if its a jingle, play it as ambient music |
|
|
239 | if ($meta->{meta}{jingle}) { |
|
|
240 | if (delete $AUDIO_PLAY{$face}) { # take the jingle out of the sound queue |
|
|
241 | push @MUSIC_JINGLE, $meta; # push it oto the music/jingle queue |
|
|
242 | &audio_music_push ($face); |
|
|
243 | } |
|
|
244 | } else { |
|
|
245 | # fetch from database |
|
|
246 | CFPlus::DB::get res_data => $meta->{name}, sub { |
|
|
247 | my $rwops = new CFPlus::RW $_[0]; |
|
|
248 | my $chunk = new CFPlus::MixChunk $rwops |
|
|
249 | or Carp::confess "sound face " . (JSON::XS::to_json $meta) . " unloadable: " . CFPlus::Mix_GetError; |
|
|
250 | $chunk->volume (($meta->{meta}{volume} || 1) * 128); |
|
|
251 | $AUDIO_CHUNK{$face} = $chunk; |
|
|
252 | |
|
|
253 | audio_sound_push ($face); |
|
|
254 | }; |
|
|
255 | } |
|
|
256 | } |
|
|
257 | } |
|
|
258 | |
|
|
259 | sub audio_sound_play { |
|
|
260 | my ($face, $dx, $dy, $vol) = @_; |
|
|
261 | |
|
|
262 | $SDL_MIXER |
|
|
263 | or return; |
|
|
264 | $CFG->{effects_enable} |
|
|
265 | or return; |
|
|
266 | |
|
|
267 | my $queue = $AUDIO_PLAY{$face} ||= []; |
|
|
268 | push @$queue, [Event::time + 0.6, $dx, $dy, $vol]; # do not play sound for outdated events |
|
|
269 | audio_sound_push $face |
|
|
270 | unless @$queue > 1; |
|
|
271 | } |
|
|
272 | |
|
|
273 | sub audio_music_set_meta { |
|
|
274 | my ($meta) = @_; |
|
|
275 | |
|
|
276 | $MUSIC_PLAYING_META = $meta; |
|
|
277 | $MUSIC_PLAYING_WIDGET->set_markup ( |
|
|
278 | "<b>Name</b>: " . (CFPlus::asxml $meta->{meta}{name}) . "\n" |
|
|
279 | . "<b>Author</b>: " . (CFPlus::asxml $meta->{meta}{author}) . "\n" |
|
|
280 | . "<b>Source</b>: " . (CFPlus::asxml $meta->{meta}{source}) . "\n" |
|
|
281 | . "<b>License</b>: " . (CFPlus::asxml $meta->{meta}{license}) |
|
|
282 | ); |
|
|
283 | } |
|
|
284 | |
|
|
285 | sub audio_music_update_volume { |
|
|
286 | return unless $MUSIC_PLAYING_META; |
|
|
287 | my $volume = $MUSIC_PLAYING_META->{meta}{volume} || 1; |
|
|
288 | my $base = $MUSIC_PLAYING_META->{meta}{jingle} ? 1 : $CFG->{bgm_volume}; |
|
|
289 | CFPlus::MixMusic::volume $base * $volume * 128; |
|
|
290 | } |
|
|
291 | |
|
|
292 | sub audio_music_start { |
|
|
293 | my $meta = $MUSIC_PLAYING_META; |
|
|
294 | |
|
|
295 | CFPlus::DB::get res_data => $meta->{name}, sub { |
|
|
296 | return unless $SDL_MIXER; |
|
|
297 | |
|
|
298 | # music might have changed... |
|
|
299 | $meta eq $MUSIC_PLAYING_META |
|
|
300 | or return &audio_music_start (); |
|
|
301 | |
|
|
302 | audio_music_update_volume; |
|
|
303 | |
|
|
304 | $MUSIC_PLAYING_DATA = \$_[0]; |
|
|
305 | |
|
|
306 | my $rwops = $meta->{path} |
|
|
307 | ? new_from_file CFPlus::RW $meta->{path} |
|
|
308 | : new CFPlus::RW $$MUSIC_PLAYING_DATA; |
|
|
309 | |
|
|
310 | $MUSIC_PLAYER = new CFPlus::MixMusic $rwops |
|
|
311 | or Carp::confess "music face $meta->{face} unloadable: " . CFPlus::Mix_GetError; |
|
|
312 | |
|
|
313 | my $NOW = time; |
|
|
314 | |
|
|
315 | if ($MUSIC_PLAYING_META->{stop_time} > $NOW - $MUSIC_RESUME) { |
|
|
316 | my $pos = $MUSIC_PLAYING_META->{stop_pos}; |
|
|
317 | $MUSIC_PLAYER->fade_in_pos (0, 1000, $pos); |
|
|
318 | $MUSIC_START = time - $pos; |
|
|
319 | } else { |
|
|
320 | $MUSIC_PLAYER->play (0); |
|
|
321 | $MUSIC_START = time; |
|
|
322 | } |
|
|
323 | |
|
|
324 | delete $meta->{stop_time}; |
|
|
325 | delete $meta->{stop_pos}; |
|
|
326 | } |
|
|
327 | } |
|
|
328 | |
|
|
329 | sub audio_music_push { |
|
|
330 | return unless $SDL_MIXER; |
|
|
331 | |
|
|
332 | my $fade_out; |
|
|
333 | |
|
|
334 | if (@MUSIC_JINGLE) { |
|
|
335 | @MUSIC_HAVE = $MUSIC_JINGLE[0]; |
|
|
336 | $fade_out = 333; |
|
|
337 | } else { |
|
|
338 | return unless $CFG->{bgm_enable}; |
|
|
339 | |
|
|
340 | my @have = |
|
|
341 | grep $_, |
|
|
342 | map $CONN->{music_meta}{$_}, |
|
|
343 | @$MUSIC_WANT; |
|
|
344 | |
|
|
345 | @MUSIC_HAVE = @have |
|
|
346 | if @have; |
|
|
347 | |
|
|
348 | # default MUSIC_HAVE == MUSIC_DEFAULT |
|
|
349 | @MUSIC_HAVE = { path => CFPlus::find_rcfile "music/$MUSIC_DEFAULT" } unless @MUSIC_HAVE; |
|
|
350 | $fade_out = 1000; |
|
|
351 | } |
|
|
352 | |
|
|
353 | # if the currently playing song is acceptable, let it continue |
|
|
354 | return if grep $MUSIC_PLAYING_META == $_, @MUSIC_HAVE; |
|
|
355 | |
|
|
356 | my $NOW = time; |
|
|
357 | |
|
|
358 | if ($MUSIC_PLAYING_META) { |
|
|
359 | $MUSIC_PLAYING_META->{stop_time} = $NOW; |
|
|
360 | $MUSIC_PLAYING_META->{stop_pos} = $NOW - $MUSIC_START; |
|
|
361 | CFPlus::MixMusic::fade_out $fade_out; |
|
|
362 | } else { |
|
|
363 | # sort by stop time, oldest first |
|
|
364 | @MUSIC_HAVE = sort { $a->{stop_time} <=> $b->{stop_time} } @MUSIC_HAVE; |
|
|
365 | |
|
|
366 | # if the most recently-played piece played very recently, |
|
|
367 | # resume it, else choose the oldest piece for rotation. |
|
|
368 | audio_music_set_meta |
|
|
369 | $MUSIC_HAVE[-1]{stop_time} > $NOW - $MUSIC_RESUME |
|
|
370 | ? $MUSIC_HAVE[-1] |
|
|
371 | : $MUSIC_HAVE[0]; |
|
|
372 | |
|
|
373 | audio_music_start; |
|
|
374 | } |
|
|
375 | } |
|
|
376 | |
|
|
377 | sub audio_music_set_ambient { |
|
|
378 | my ($songs) = @_; |
|
|
379 | |
|
|
380 | $MUSIC_WANT = $songs; |
|
|
381 | audio_music_push; |
|
|
382 | } |
|
|
383 | |
|
|
384 | sub audio_music_finished { |
|
|
385 | # we compress multiple jingles of the same type |
|
|
386 | shift @MUSIC_JINGLE |
|
|
387 | while @MUSIC_JINGLE && $MUSIC_PLAYING_META == $MUSIC_JINGLE[0]; |
|
|
388 | |
|
|
389 | $MUSIC_PLAYING_WIDGET->clear; |
|
|
390 | |
|
|
391 | undef $MUSIC_PLAYER; |
|
|
392 | undef $MUSIC_PLAYING_META; |
|
|
393 | undef $MUSIC_PLAYING_DATA; |
|
|
394 | |
|
|
395 | audio_music_push; |
|
|
396 | } |
|
|
397 | |
|
|
398 | sub audio_init { |
|
|
399 | if ($CFG->{audio_enable}) { |
|
|
400 | $ENV{MIX_EFFECTSMAXSPEED} = 1; |
|
|
401 | $SDL_MIXER = !CFPlus::Mix_OpenAudio; |
|
|
402 | |
|
|
403 | unless ($SDL_MIXER) { |
|
|
404 | status "Unable to open sound device: there will be no sound"; |
|
|
405 | return; |
|
|
406 | } |
|
|
407 | |
|
|
408 | CFPlus::Mix_AllocateChannels 16; |
|
|
409 | |
|
|
410 | audio_music_finished; |
|
|
411 | } else { |
|
|
412 | undef $SDL_MIXER; |
|
|
413 | } |
|
|
414 | } |
|
|
415 | |
|
|
416 | sub audio_shutdown { |
|
|
417 | undef $MUSIC_PLAYER; |
|
|
418 | undef $MUSIC_PLAYING_META; |
|
|
419 | undef $MUSIC_PLAYING_DATA; |
|
|
420 | |
|
|
421 | $MUSIC_WANT = []; |
|
|
422 | @MUSIC_JINGLE = (); |
|
|
423 | %AUDIO_PLAY = (); |
|
|
424 | %AUDIO_CHUNK = (); |
|
|
425 | |
|
|
426 | CFPlus::Mix_CloseAudio if $SDL_MIXER; |
|
|
427 | undef $SDL_MIXER; |
|
|
428 | } |
|
|
429 | |
|
|
430 | ############################################################################# |
|
|
431 | |
|
|
432 | sub destroy_query_dialog { |
|
|
433 | (delete $_[0]{query_dialog})->destroy |
|
|
434 | if $_[0]{query_dialog}; |
|
|
435 | } |
|
|
436 | |
|
|
437 | # FIXME: a very ugly hack to wait for stat update look below! #d# |
|
|
438 | our $QUERY_TIMER; #d# |
|
|
439 | |
|
|
440 | # server query dialog |
|
|
441 | sub server_query { |
|
|
442 | my ($conn, $flags, $prompt) = @_; |
|
|
443 | |
|
|
444 | # FIXME: a very ugly hack to wait for stat update #d# |
|
|
445 | if ($prompt =~ /roll new stats/ and not $conn->{stat_change_with}) { |
|
|
446 | unless ($QUERY_TIMER) { |
|
|
447 | $QUERY_TIMER = |
|
|
448 | Event->timer ( |
|
|
449 | after => 1, |
|
|
450 | cb => sub { |
|
|
451 | server_query ($conn, $flags, $prompt, 1); |
|
|
452 | $QUERY_TIMER = undef |
|
|
453 | } |
|
|
454 | ); |
|
|
455 | return; |
|
|
456 | } |
|
|
457 | } |
|
|
458 | |
|
|
459 | $conn->{query_dialog} = my $dialog = new CFPlus::UI::Toplevel |
|
|
460 | x => "center", |
|
|
461 | y => "center", |
|
|
462 | title => "Server Query", |
|
|
463 | child => my $vbox = new CFPlus::UI::VBox, |
|
|
464 | ; |
|
|
465 | |
|
|
466 | my @dialog = my $label = new CFPlus::UI::Label |
|
|
467 | max_w => $::WIDTH * 0.8, |
|
|
468 | ellipsise => 0, |
|
|
469 | text => $prompt; |
|
|
470 | |
|
|
471 | if ($flags & CS_QUERY_YESNO) { |
|
|
472 | push @dialog, my $hbox = new CFPlus::UI::HBox; |
|
|
473 | |
|
|
474 | $hbox->add (new CFPlus::UI::Button |
|
|
475 | text => "No", |
|
|
476 | on_activate => sub { |
|
|
477 | $conn->send ("reply n"); |
|
|
478 | $dialog->destroy; |
|
|
479 | 0 |
|
|
480 | } |
|
|
481 | ); |
|
|
482 | $hbox->add (new CFPlus::UI::Button |
|
|
483 | text => "Yes", |
|
|
484 | on_activate => sub { |
|
|
485 | $conn->send ("reply y"); |
|
|
486 | destroy_query_dialog $conn; |
|
|
487 | 0 |
|
|
488 | }, |
|
|
489 | ); |
|
|
490 | |
|
|
491 | $dialog->grab_focus; |
|
|
492 | |
|
|
493 | } elsif ($flags & CS_QUERY_SINGLECHAR) { |
|
|
494 | if ($prompt =~ /Now choose a character|Press any key for the next race/i) { |
|
|
495 | $dialog->{tooltip} = "#charcreation_focus"; |
|
|
496 | |
|
|
497 | unshift @dialog, new CFPlus::UI::Label |
|
|
498 | max_w => $::WIDTH * 0.8, |
|
|
499 | ellipsise => 0, |
|
|
500 | markup => "\nOr use your keyboard and the text entry below:\n"; |
|
|
501 | |
|
|
502 | unshift @dialog, my $table = new CFPlus::UI::Table; |
|
|
503 | |
|
|
504 | $table->add_at (0, 0, new CFPlus::UI::Button |
|
|
505 | text => "Next Race", |
|
|
506 | on_activate => sub { |
|
|
507 | $conn->send ("reply n"); |
|
|
508 | destroy_query_dialog $conn; |
|
|
509 | 0 |
|
|
510 | }, |
|
|
511 | ); |
|
|
512 | $table->add_at (2, 0, new CFPlus::UI::Button |
|
|
513 | text => "Accept", |
|
|
514 | on_activate => sub { |
|
|
515 | $conn->send ("reply d"); |
|
|
516 | destroy_query_dialog $conn; |
|
|
517 | 0 |
|
|
518 | }, |
|
|
519 | ); |
|
|
520 | |
|
|
521 | if ($conn->{chargen_race_description}) { |
|
|
522 | unshift @dialog, new CFPlus::UI::Label |
|
|
523 | max_w => $::WIDTH * 0.8, |
|
|
524 | ellipsise => 0, |
|
|
525 | markup => "<span foreground='#ccccff'>$conn->{chargen_race_description}</span>", |
|
|
526 | ; |
|
|
527 | } |
|
|
528 | |
|
|
529 | unshift @dialog, new CFPlus::UI::Face |
|
|
530 | face => $conn->{player}{face}, |
|
|
531 | bg => [.2, .2, .2, 1], |
|
|
532 | min_w => 64, |
|
|
533 | min_h => 64, |
|
|
534 | ; |
|
|
535 | |
|
|
536 | if ($conn->{chargen_race_title}) { |
|
|
537 | unshift @dialog, new CFPlus::UI::Label |
|
|
538 | allign => 1, |
|
|
539 | ellipsise => 0, |
|
|
540 | markup => "<span foreground='#ccccff' size='large'>Race: $conn->{chargen_race_title}</span>", |
|
|
541 | ; |
|
|
542 | } |
|
|
543 | |
|
|
544 | unshift @dialog, new CFPlus::UI::Label |
|
|
545 | max_w => $::WIDTH * 0.4, |
|
|
546 | ellipsise => 0, |
|
|
547 | markup => (CFPlus::Pod::section_label ui => "chargen_race"), |
|
|
548 | ; |
|
|
549 | |
|
|
550 | } elsif ($prompt =~ /roll new stats/) { |
|
|
551 | if (my $stat = delete $conn->{stat_change_with}) { |
|
|
552 | $conn->send ("reply $stat"); |
|
|
553 | destroy_query_dialog $conn; |
|
|
554 | return; |
|
|
555 | } |
|
|
556 | |
|
|
557 | unshift @dialog, new CFPlus::UI::Label |
|
|
558 | max_w => $::WIDTH * 0.4, |
|
|
559 | ellipsise => 0, |
|
|
560 | markup => "\nOr use your keyboard and the text entry below:\n"; |
|
|
561 | |
|
|
562 | unshift @dialog, my $table = new CFPlus::UI::Table; |
|
|
563 | |
|
|
564 | # left: re-roll |
|
|
565 | $table->add_at (0, 0, new CFPlus::UI::Button |
|
|
566 | text => "Roll Again", |
|
|
567 | on_activate => sub { |
|
|
568 | $conn->send ("reply y"); |
|
|
569 | destroy_query_dialog $conn; |
|
|
570 | 0 |
|
|
571 | }, |
|
|
572 | ); |
|
|
573 | |
|
|
574 | # center: swap stats |
|
|
575 | my ($sw1, $sw2) = map +(new CFPlus::UI::Selector |
|
|
576 | expand => 1, |
|
|
577 | value => $_, |
|
|
578 | options => [ |
|
|
579 | [1 => "Str", "Strength ($conn->{stat}{+CS_STAT_STR})"], |
|
|
580 | [2 => "Dex", "Dexterity ($conn->{stat}{+CS_STAT_DEX})"], |
|
|
581 | [3 => "Con", "Constitution ($conn->{stat}{+CS_STAT_CON})"], |
|
|
582 | [4 => "Int", "Intelligence ($conn->{stat}{+CS_STAT_INT})"], |
|
|
583 | [5 => "Wis", "Wisdom ($conn->{stat}{+CS_STAT_WIS})"], |
|
|
584 | [6 => "Pow", "Power ($conn->{stat}{+CS_STAT_POW})"], |
|
|
585 | [7 => "Cha", "Charisma ($conn->{stat}{+CS_STAT_CHA})"], |
|
|
586 | ], |
|
|
587 | ), 1 .. 2; |
|
|
588 | |
|
|
589 | $table->add_at (2, 0, new CFPlus::UI::Button |
|
|
590 | text => "Swap Stats", |
|
|
591 | on_activate => sub { |
|
|
592 | $conn->{stat_change_with} = $sw2->{value}; |
|
|
593 | $conn->send ("reply $sw1->{value}"); |
|
|
594 | destroy_query_dialog $conn; |
|
|
595 | 0 |
|
|
596 | }, |
|
|
597 | ); |
|
|
598 | $table->add_at (2, 1, new CFPlus::UI::HBox children => [$sw1, $sw2]); |
|
|
599 | |
|
|
600 | # right: accept |
|
|
601 | $table->add_at (4, 0, new CFPlus::UI::Button |
|
|
602 | text => "Accept", |
|
|
603 | on_activate => sub { |
|
|
604 | $conn->send ("reply n"); |
|
|
605 | $STATS_PAGE->hide; |
|
|
606 | destroy_query_dialog $conn; |
|
|
607 | 0 |
|
|
608 | }, |
|
|
609 | ); |
|
|
610 | |
|
|
611 | unshift @dialog, my $hbox = new CFPlus::UI::HBox; |
|
|
612 | for ( |
|
|
613 | [Str => CS_STAT_STR], |
|
|
614 | [Dex => CS_STAT_DEX], |
|
|
615 | [Con => CS_STAT_CON], |
|
|
616 | [Int => CS_STAT_INT], |
|
|
617 | [Wis => CS_STAT_WIS], |
|
|
618 | [Pow => CS_STAT_POW], |
|
|
619 | [Cha => CS_STAT_CHA], |
|
|
620 | ) { |
|
|
621 | my ($name, $id) = @$_; |
|
|
622 | $hbox->add (new CFPlus::UI::Label |
|
|
623 | markup => "$conn->{stat}{$id} <span foreground='yellow'>$name</span>", |
|
|
624 | align => 0, |
|
|
625 | expand => 1, |
|
|
626 | can_events => 1, |
|
|
627 | can_hover => 1, |
|
|
628 | tooltip => "#stat_$name", |
|
|
629 | ); |
|
|
630 | } |
|
|
631 | |
|
|
632 | unshift @dialog, new CFPlus::UI::Label |
|
|
633 | max_w => $::WIDTH * 0.4, |
|
|
634 | ellipsise => 0, |
|
|
635 | markup => (CFPlus::Pod::section_label ui => "chargen_stats"), |
|
|
636 | ; |
|
|
637 | } |
|
|
638 | |
|
|
639 | push @dialog, my $entry = new CFPlus::UI::Entry |
|
|
640 | on_changed => sub { |
|
|
641 | $conn->send ("reply $_[1]"); |
|
|
642 | destroy_query_dialog $conn; |
|
|
643 | 0 |
|
|
644 | }, |
|
|
645 | ; |
|
|
646 | |
|
|
647 | $entry->grab_focus; |
|
|
648 | |
|
|
649 | } else { |
|
|
650 | $dialog->{tooltip} = "Enter the reply and press return (click on the entry to make sure it has keyboard focus)"; |
|
|
651 | |
|
|
652 | push @dialog, my $entry = new CFPlus::UI::Entry |
|
|
653 | $flags & CS_QUERY_HIDEINPUT ? (hidden => "*") : (), |
|
|
654 | on_activate => sub { |
|
|
655 | $conn->send ("reply $_[1]"); |
|
|
656 | destroy_query_dialog $conn; |
|
|
657 | 0 |
|
|
658 | }, |
|
|
659 | ; |
|
|
660 | |
|
|
661 | $entry->grab_focus; |
|
|
662 | } |
|
|
663 | |
|
|
664 | $vbox->add (@dialog); |
|
|
665 | $dialog->show; |
|
|
666 | } |
|
|
667 | |
127 | sub start_game { |
668 | sub start_game { |
128 | status "logging in..."; |
669 | status "logging in..."; |
129 | |
670 | |
130 | $LOGIN_BUTTON->set_text ("Logout"); |
671 | $LOGIN_BUTTON->set_text ("Logout"); |
131 | $SETUP_DIALOG->hide; |
672 | $SETUP_DIALOG->hide; |
132 | |
673 | |
133 | my $mapsize = List::Util::min 32, List::Util::max 11, int $WIDTH * $CFG->{mapsize} * 0.01 / 32; |
674 | my $mapsize = List::Util::min 32, List::Util::max 11, int $WIDTH * $CFG->{mapsize} * 0.01 / 32; |
134 | |
675 | |
135 | my ($host, $port) = split /:/, $CFG->{host}; |
676 | my ($host, $port) = split /:/, $PROFILE->{host}; |
136 | |
677 | |
137 | $MAP = new CFClient::Map $mapsize, $mapsize; |
678 | $MAP = new CFPlus::Map; |
138 | |
679 | |
139 | $CONN = eval { |
680 | $CONN = eval { |
140 | new CFClient::Protocol |
681 | new CFPlus::Protocol |
141 | host => $host, |
682 | host => $host, |
142 | port => $port || 13327, |
683 | port => $port || 13327, |
143 | user => $CFG->{user}, |
684 | user => $PROFILE->{user}, |
144 | pass => $CFG->{password}, |
685 | pass => $PROFILE->{password}, |
145 | mapw => $mapsize, |
686 | mapw => $mapsize, |
146 | maph => $mapsize, |
687 | maph => $mapsize, |
147 | |
688 | |
|
|
689 | client => "cfplus $CFPlus::VERSION $] $^O", |
|
|
690 | |
148 | map_widget => $MAPWIDGET, |
691 | map_widget => $MAPWIDGET, |
149 | logview => $LOGVIEW, |
|
|
150 | statusbox => $STATUSBOX, |
692 | statusbox => $STATUSBOX, |
151 | map => $MAP, |
693 | map => $MAP, |
152 | mapmap => $MAPMAP, |
694 | mapmap => $MAPMAP, |
|
|
695 | query => \&server_query, |
153 | |
696 | |
154 | sound_play => sub { |
697 | setup_req => { |
155 | my ($x, $y, $soundnum, $type) = @_; |
698 | smoothing => $CFG->{map_smoothing}*1, |
156 | |
|
|
157 | $SDL_MIXER |
|
|
158 | or return; |
|
|
159 | |
|
|
160 | my $chunk = $AUDIO_CHUNKS{$SOUNDS[$soundnum]} |
|
|
161 | or return; |
|
|
162 | |
|
|
163 | $chunk->play; |
|
|
164 | }, |
699 | }, |
165 | }; |
700 | }; |
166 | |
701 | |
167 | if ($CONN) { |
702 | if ($CONN) { |
168 | CFClient::lowdelay fileno $CONN->{fh}; |
703 | CFPlus::lowdelay fileno $CONN->{fh}; |
169 | |
704 | |
170 | status "login successful"; |
705 | status "login successful"; |
171 | } else { |
706 | } else { |
172 | status "unable to connect"; |
707 | status "unable to connect"; |
173 | stop_game(); |
708 | stop_game(); |
174 | } |
709 | } |
175 | } |
710 | } |
176 | |
711 | |
177 | sub stop_game { |
712 | sub stop_game { |
178 | $LOGIN_BUTTON->set_text ("Login"); |
713 | $LOGIN_BUTTON->set_text ("Login"); |
|
|
714 | $SETUP_NOTEBOOK->set_current_page ($SETUP_SERVER); |
179 | $SETUP_DIALOG->show; |
715 | $SETUP_DIALOG->show; |
180 | $INV_WINDOW->hide; |
716 | $PL_WINDOW->hide; |
181 | $LOGVIEW->hide; |
717 | $SPELL_LIST->clear_spells; |
|
|
718 | $CFPlus::UI::ROOT->emit (stop_game => ! ! $CONN); |
|
|
719 | |
|
|
720 | &audio_music_set_ambient ([]); |
182 | |
721 | |
183 | return unless $CONN; |
722 | return unless $CONN; |
184 | |
723 | |
185 | status "connection closed"; |
724 | status "connection closed"; |
186 | |
725 | |
|
|
726 | destroy_query_dialog $CONN; |
187 | $CONN->destroy; |
727 | $CONN->destroy; |
188 | $CONN = 0; # false, does not autovivify |
728 | $CONN = 0; # false, does not autovivify |
|
|
729 | |
|
|
730 | undef $MAP; |
189 | } |
731 | } |
190 | |
732 | |
191 | sub graphics_setup { |
733 | sub graphics_setup { |
192 | my $vbox = new CFClient::UI::VBox; |
734 | my $vbox = new CFPlus::UI::VBox; |
193 | |
735 | |
194 | $vbox->add (my $table = new CFClient::UI::Table expand => 1, col_expand => [0, 1]); |
736 | $vbox->add (my $table = new CFPlus::UI::Table expand => 1, col_expand => [0, 1]); |
195 | |
737 | |
|
|
738 | my $row = 0; |
|
|
739 | |
|
|
740 | $table->add_at (0, $row, new CFPlus::UI::Label valign => 0, align => 1, text => "OpenGL Info"); |
|
|
741 | $table->add_at (1, $row++, new CFPlus::UI::Label valign => 0, fontsize => 0.8, text => CFPlus::OpenGL::gl_vendor . ", " . CFPlus::OpenGL::gl_version, |
|
|
742 | can_events => 1, |
|
|
743 | tooltip => "<tt><span size='8192'>" . (CFPlus::OpenGL::gl_extensions) . "</span></tt>"); |
|
|
744 | |
|
|
745 | my $vidmode_tooltip = |
|
|
746 | "<b>Video Mode.</b> The video mode to use for fullscreen (and the window size for windowed operation). " |
|
|
747 | . "The format is <i>width</i> x <i>height</i> \@ <i>depth-per-channel</i> + <i>alpha-channel</i>."; |
|
|
748 | |
196 | $table->add (0, 0, new CFClient::UI::Label valign => 0, align => 1, text => "Video Mode"); |
749 | $table->add_at (0, $row, new CFPlus::UI::Label valign => 0, align => 1, text => "Video Mode"); |
197 | $table->add (1, 0, my $hbox = new CFClient::UI::HBox); |
750 | $table->add_at (1, $row++, my $hbox = new CFPlus::UI::HBox); |
198 | |
751 | |
|
|
752 | $hbox->add (my $mode_slider = new CFPlus::UI::Slider |
199 | $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]); |
753 | force_w => $WIDTH * 0.1, expand => 1, range => [$CFG->{sdl_mode}, 0, $#SDL_MODES, 0, 1], |
200 | $hbox->add (my $mode_label = new CFClient::UI::Label align => 0, valign => 0, height => 0.8, template => "9999x9999"); |
754 | tooltip => $vidmode_tooltip); |
|
|
755 | $hbox->add (my $mode_label = new CFPlus::UI::Label |
|
|
756 | align => 0, valign => 0, height => 0.8, template => "9999x9999@9+9", |
|
|
757 | can_events => 1, tooltip => $vidmode_tooltip); |
201 | |
758 | |
202 | $mode_slider->connect (changed => sub { |
759 | $mode_slider->connect (changed => sub { |
203 | my ($self, $value) = @_; |
760 | my ($self, $value) = @_; |
204 | |
761 | |
205 | $CFG->{sdl_mode} = $self->{range}[0] = $value = int $value; |
762 | $CFG->{sdl_mode} = $self->{range}[0] = $value = int $value; |
206 | $mode_label->set_text (sprintf "%dx%d", @{$SDL_MODES[$value]}); |
763 | $mode_label->set_text (sprintf '%dx%d@%d+%d', @{$SDL_MODES[$value]}); |
207 | }); |
764 | }); |
208 | $mode_slider->emit (changed => $mode_slider->{range}[0]); |
765 | $mode_slider->emit (changed => $mode_slider->{range}[0]); |
209 | |
766 | |
210 | my $row = 1; |
|
|
211 | |
|
|
212 | $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Fullscreen"); |
767 | $table->add_at (0, $row, new CFPlus::UI::Label valign => 0, align => 1, text => "Fullscreen"); |
213 | $table->add (1, $row++, new CFClient::UI::CheckBox |
768 | $table->add_at (1, $row++, $FULLSCREEN_ENABLE = new CFPlus::UI::CheckBox |
214 | state => $CFG->{fullscreen}, |
769 | state => $CFG->{fullscreen}, |
215 | tooltip => "Bring the client into fullscreen mode.", |
770 | tooltip => "Bring the client into fullscreen mode.", |
216 | on_changed => sub { |
771 | on_changed => sub { my ($self, $value) = @_; $CFG->{fullscreen} = $value; 0 } |
217 | my ($self, $value) = @_; |
|
|
218 | $CFG->{fullscreen} = $value; |
|
|
219 | } |
|
|
220 | ); |
772 | ); |
221 | |
773 | |
|
|
774 | $table->add_at (0, $row, new CFPlus::UI::Label valign => 0, align => 1, text => "Force OpenGL 1.1"); |
|
|
775 | $table->add_at (1, $row++, new CFPlus::UI::CheckBox |
|
|
776 | state => $CFG->{opengl11}, |
|
|
777 | tooltip => "Limit CFPlus to use OpenGL 1.1 features only. This will normally result in " |
|
|
778 | . "higher memory usage and slower performance. It will, however, help tremendously on " |
|
|
779 | . "cards that claim to support a feature but fall back to software rendering. " |
|
|
780 | . "Nvidia Geforce FX cards are known to claim features the hardware doesn't support, " |
|
|
781 | . "but cards and drivers from other vendors (ATI) are often just as bad. <b>If you " |
|
|
782 | . "experience extremely low framerates and your card should do better, try this option.</b>", |
|
|
783 | on_changed => sub { my ($self, $value) = @_; $CFG->{opengl11} = $value; 0 } |
|
|
784 | ); |
|
|
785 | |
222 | $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Fast & Ugly"); |
786 | $table->add_at (0, $row, new CFPlus::UI::Label valign => 0, align => 1, text => "Fast & Ugly"); |
223 | $table->add (1, $row++, new CFClient::UI::CheckBox |
787 | $table->add_at (1, $row++, new CFPlus::UI::CheckBox |
224 | state => $CFG->{fast}, |
788 | state => $CFG->{fast}, |
225 | tooltip => "Lower the visual quality considerably to speed up rendering.", |
789 | tooltip => "Lower the visual quality considerably to speed up rendering.", |
|
|
790 | on_changed => sub { my ($self, $value) = @_; $CFG->{fast} = $value; 0 } |
|
|
791 | ); |
|
|
792 | |
|
|
793 | $table->add_at (0, $row, new CFPlus::UI::Label valign => 0, align => 1, text => "GUI Fontsize"); |
|
|
794 | $table->add_at (1, $row++, new CFPlus::UI::Slider |
|
|
795 | range => [$CFG->{gui_fontsize}, 0.5, 2, 0, 0.1], |
|
|
796 | tooltip => "The base font size used by most GUI elements that do not have their own setting.", |
|
|
797 | on_changed => sub { $CFG->{gui_fontsize} = $_[1]; 0 }, |
|
|
798 | ); |
|
|
799 | |
|
|
800 | $table->add_at (1, $row++, new CFPlus::UI::Button |
|
|
801 | expand => 1, align => 0, text => "Apply", |
|
|
802 | tooltip => "Apply the video settings above.", |
226 | on_changed => sub { |
803 | on_activate => sub { |
227 | my ($self, $value) = @_; |
804 | video_shutdown (); |
228 | $CFG->{fast} = $value; |
805 | video_init (); |
|
|
806 | 0 |
229 | } |
807 | } |
230 | ); |
808 | ); |
231 | |
809 | |
232 | $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Map Scale"); |
810 | $table->add_at (0, $row, new CFPlus::UI::Label valign => 0, align => 1, text => "Map Scale"); |
233 | $table->add (1, $row++, new CFClient::UI::Slider |
811 | $table->add_at (1, $row++, new CFPlus::UI::Slider |
234 | range => [(log $CFG->{map_scale}) / (log 2), -3, 1, 0, 1], |
812 | range => [(log $CFG->{map_scale}) / (log 2), -3, 1, 0, 1], |
235 | tooltip => "Enlarge or shrink the displayed map. Changes are instant.", |
813 | tooltip => "Enlarge or shrink the displayed map. Changes are instant.", |
236 | on_changed => sub { |
814 | on_changed => sub { my ($self, $value) = @_; $CFG->{map_scale} = 2 ** $value; 0 } |
237 | my ($self, $value) = @_; |
|
|
238 | $CFG->{map_scale} = 2 ** $value; |
|
|
239 | } |
|
|
240 | ); |
815 | ); |
241 | |
816 | |
|
|
817 | $table->add_at (0, $row, new CFPlus::UI::Label valign => 0, align => 1, text => "Map Smoothing"); |
|
|
818 | $table->add_at (1, $row++, new CFPlus::UI::CheckBox |
|
|
819 | state => $CFG->{map_smoothing}, |
|
|
820 | tooltip => "<b>Map Smoothing</b> tries to make tile borders less square. " |
|
|
821 | . "This increases load on the graphics subsystem and works only with TRT servers. " |
|
|
822 | . "Changes take effect at next login only.", |
|
|
823 | on_changed => sub { my ($self, $value) = @_; $CFG->{map_smoothing} = $value; 0 } |
|
|
824 | ); |
|
|
825 | |
242 | $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Fog of War"); |
826 | $table->add_at (0, $row, new CFPlus::UI::Label valign => 0, align => 1, text => "Fog of War"); |
243 | $table->add (1, $row++, new CFClient::UI::CheckBox |
827 | $table->add_at (1, $row++, new CFPlus::UI::CheckBox |
244 | state => $CFG->{fow_enable}, |
828 | state => $CFG->{fow_enable}, |
245 | tooltip => "<b>Fog-of-War</b> marks areas that cannot be seen by the player. Changes are instant.", |
829 | tooltip => "<b>Fog-of-War</b> marks areas that cannot be seen by the player. Changes are instant.", |
246 | on_changed => sub { |
830 | on_changed => sub { my ($self, $value) = @_; $CFG->{fow_enable} = $value; 0 } |
247 | my ($self, $value) = @_; |
|
|
248 | $CFG->{fow_enable} = $value; |
|
|
249 | } |
|
|
250 | ); |
831 | ); |
251 | |
832 | |
252 | $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "FoW Intensity"); |
833 | $table->add_at (0, $row, new CFPlus::UI::Label valign => 0, align => 1, text => "FoW Intensity"); |
253 | $table->add (1, $row++, new CFClient::UI::Slider |
834 | $table->add_at (1, $row++, new CFPlus::UI::Slider |
254 | range => [$CFG->{fow_intensity}, 0, 1, 0, 1 / 256], |
835 | range => [$CFG->{fow_intensity}, 0, 1, 0, 1 / 256], |
255 | tooltip => "<b>Fog of War Lightness.</b> The higher the intensity, the lighter the Fog-of-War color. Changes are instant.", |
836 | tooltip => "<b>Fog of War Lightness.</b> The higher the intensity, the lighter the Fog-of-War color. Changes are instant.", |
256 | on_changed => sub { |
837 | on_changed => sub { my ($self, $value) = @_; $CFG->{fow_intensity} = $value; 0 } |
257 | my ($self, $value) = @_; |
|
|
258 | $CFG->{fow_intensity} = $value; |
|
|
259 | } |
|
|
260 | ); |
838 | ); |
261 | |
839 | |
262 | $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "FoW Smooth"); |
|
|
263 | $table->add (1, $row++, new CFClient::UI::CheckBox |
|
|
264 | state => $CFG->{fow_smooth}, |
|
|
265 | tooltip => "Smooth the Fog-of-War a bit to make it more realistic. Changes are instant.", |
|
|
266 | on_changed => sub { |
|
|
267 | my ($self, $value) = @_; |
|
|
268 | $CFG->{fow_smooth} = $value; |
|
|
269 | status "Fog of War smoothing requires OpenGL 1.2 or higher" if $CFClient::OpenGL::GL_VERSION < 1.2; |
|
|
270 | } |
|
|
271 | ); |
|
|
272 | |
|
|
273 | $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "GUI Fontsize"); |
|
|
274 | $table->add (1, $row++, new CFClient::UI::Slider |
|
|
275 | range => [$CFG->{gui_fontsize}, 0.5, 2, 0, 0.1], |
|
|
276 | tooltip => "The base font size used by most GUI elements that do not have their own setting.", |
|
|
277 | on_changed => sub { $CFG->{gui_fontsize} = $_[1] }, |
|
|
278 | ); |
|
|
279 | |
|
|
280 | $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Message Fontsize"); |
840 | $table->add_at (0, $row, new CFPlus::UI::Label valign => 0, align => 1, text => "Message Fontsize"); |
281 | $table->add (1, $row++, new CFClient::UI::Slider |
841 | $table->add_at (1, $row++, new CFPlus::UI::Slider |
282 | range => [$CFG->{log_fontsize}, 0.5, 2, 0, 0.1], |
842 | range => [$CFG->{log_fontsize}, 0.5, 2, 0, 0.1], |
283 | tooltip => "The font size used by the <b>message/server log</b> window only. Changes are instant.", |
843 | tooltip => "The font size used by the <b>message/server log</b> window only. Changes are instant, " |
|
|
844 | . "but you still need to press apply to correctly re-layout the widget.", |
284 | on_changed => sub { $LOGVIEW->set_fontsize ($CFG->{log_fontsize} = $_[1]) }, |
845 | on_changed => sub { $MESSAGE_WINDOW->set_fontsize ($CFG->{log_fontsize} = $_[1]); 0 }, |
285 | ); |
846 | ); |
286 | |
847 | |
287 | $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Stats Fontsize"); |
|
|
288 | |
|
|
289 | $table->add (1, $row++, new CFClient::UI::Slider |
|
|
290 | range => [$CFG->{stat_fontsize}, 0.5, 2, 0, 0.1], |
|
|
291 | tooltip => "The font size used by the <b>statistics window</b> only. Changes are instant.", |
|
|
292 | on_changed => sub { |
|
|
293 | $CFG->{stat_fontsize} = $_[1]; |
|
|
294 | &set_stats_window_fontsize; |
|
|
295 | } |
|
|
296 | ); |
|
|
297 | |
|
|
298 | $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Gauge fontsize"); |
848 | $table->add_at (0, $row, new CFPlus::UI::Label valign => 0, align => 1, text => "Gauge fontsize"); |
299 | $table->add (1, $row++, new CFClient::UI::Slider |
849 | $table->add_at (1, $row++, new CFPlus::UI::Slider |
300 | range => [$CFG->{gauge_fontsize}, 0.5, 2, 0, 0.1], |
850 | range => [$CFG->{gauge_fontsize}, 0.5, 2, 0, 0.1], |
301 | tooltip => "Adjusts the fontsize of the gauges at the bottom right. Changes are instant.", |
851 | tooltip => "Adjusts the fontsize of the gauges at the bottom right. Changes are instant.", |
302 | on_changed => sub { |
852 | on_changed => sub { |
303 | $CFG->{gauge_fontsize} = $_[1]; |
853 | $CFG->{gauge_fontsize} = $_[1]; |
304 | &set_gauge_window_fontsize; |
854 | &set_gauge_window_fontsize; |
|
|
855 | 0 |
305 | } |
856 | } |
306 | ); |
857 | ); |
307 | |
858 | |
308 | $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Gauge size"); |
859 | $table->add_at (0, $row, new CFPlus::UI::Label valign => 0, align => 1, text => "Gauge size"); |
309 | $table->add (1, $row++, new CFClient::UI::Slider |
860 | $table->add_at (1, $row++, new CFPlus::UI::Slider |
310 | range => [$CFG->{gauge_size}, 0.2, 0.8], |
861 | range => [$CFG->{gauge_size}, 0.2, 0.8], |
311 | tooltip => "Adjust the size of the stats gauges at the bottom right. Changes are instant.", |
862 | tooltip => "Adjust the size of the stats gauges at the bottom right. Changes are instant.", |
312 | on_changed => sub { |
863 | on_changed => sub { |
313 | $CFG->{gauge_size} = $_[1]; |
864 | $CFG->{gauge_size} = $_[1]; |
314 | $GAUGES->{win}->set_size ($WIDTH, int $HEIGHT * $CFG->{gauge_size}); |
865 | $GAUGES->{win}->set_size ($WIDTH, int $HEIGHT * $CFG->{gauge_size}); |
315 | } |
866 | 0 |
316 | ); |
|
|
317 | |
|
|
318 | $table->add (1, $row++, new CFClient::UI::Button |
|
|
319 | expand => 1, align => 0, text => "Apply", |
|
|
320 | tooltip => "Apply the video settings", |
|
|
321 | on_activate => sub { |
|
|
322 | video_shutdown (); |
|
|
323 | video_init (); |
|
|
324 | } |
867 | } |
325 | ); |
868 | ); |
326 | |
869 | |
327 | $vbox |
870 | $vbox |
328 | } |
871 | } |
329 | |
872 | |
330 | sub audio_setup { |
873 | sub audio_setup { |
331 | my $vbox = new CFClient::UI::VBox; |
874 | my $vbox = new CFPlus::UI::VBox; |
332 | |
875 | |
333 | $vbox->add (my $table = new CFClient::UI::Table expand => 1, col_expand => [0, 1]); |
876 | $vbox->add (my $table = new CFPlus::UI::Table expand => 1, col_expand => [0, 0, 1]); |
334 | |
877 | |
335 | my $row = 0; |
878 | my $row = 0; |
336 | |
879 | |
337 | $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Audio Enable"); |
880 | $table->add_at (0, $row, new CFPlus::UI::Label valign => 0, align => 1, text => "Audio Enable"); |
338 | $table->add (1, $row++, new CFClient::UI::CheckBox |
881 | $table->add_at (1, $row++, new CFPlus::UI::CheckBox |
339 | state => $CFG->{audio_enable}, |
882 | state => $CFG->{audio_enable}, |
340 | 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.", |
883 | 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.", |
341 | on_changed => sub { |
884 | on_changed => sub { $CFG->{audio_enable} = $_[1]; 0 } |
342 | $CFG->{audio_enable} = $_[1]; |
|
|
343 | } |
|
|
344 | ); |
885 | ); |
345 | # $table->add (0, 9, new CFClient::UI::Label valign => 0, align => 1, text => "Effects Volume"); |
886 | # $table->add_at (0, 9, new CFPlus::UI::Label valign => 0, align => 1, text => "Effects Volume"); |
346 | # $table->add (1, 8, new CFClient::UI::Slider range => [$CFG->{effects_volume}, 0, 128, 1], on_changed => sub { |
887 | # $table->add_at (1, 8, new CFPlus::UI::Slider range => [$CFG->{effects_volume}, 0, 128, 1], on_changed => sub { |
347 | # $CFG->{effects_volume} = $_[1]; |
888 | # $CFG->{effects_volume} = $_[1]; |
348 | # }); |
889 | # }); |
|
|
890 | |
|
|
891 | $table->add_at (0, $row, new CFPlus::UI::Label valign => 0, align => 1, text => "Sound Effects"); |
|
|
892 | $table->add_at (1, $row, new CFPlus::UI::CheckBox |
|
|
893 | expand => 1, state => $CFG->{effects_enable}, |
|
|
894 | tooltip => "If enabled, sound effects are enabled. If disabled, no sound effects will be played.", |
|
|
895 | on_changed => sub { |
|
|
896 | $CFG->{effects_enable} = $_[1]; |
|
|
897 | $CONN->update_fx_want if $CONN; |
|
|
898 | 0 |
|
|
899 | } |
|
|
900 | ); |
|
|
901 | $table->add_at (2, $row++, new CFPlus::UI::Slider |
|
|
902 | expand => 1, range => [$CFG->{effects_volume}, 0, 1, 0, 1/128], |
|
|
903 | tooltip => "The relative volume of sound effects. Best audio quality is achieved if this " |
|
|
904 | . "is set highest (rightmost) and you use your operating system volume setting. Changes are instant.", |
|
|
905 | on_changed => sub { $CFG->{effects_volume} = $_[1]; 0 } |
|
|
906 | ); |
|
|
907 | |
349 | $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Background Music"); |
908 | $table->add_at (0, $row, new CFPlus::UI::Label valign => 0, align => 1, text => "Background Music"); |
350 | $table->add (1, $row++, my $hbox = new CFClient::UI::HBox); |
909 | $table->add_at (1, $row, new CFPlus::UI::CheckBox |
351 | $hbox->add (new CFClient::UI::CheckBox |
|
|
352 | expand => 1, state => $CFG->{bgm_enable}, |
910 | expand => 1, state => $CFG->{bgm_enable}, |
353 | tooltip => "If enabled, playing of background music is enabled. If disabled, no background music will be played.", |
911 | tooltip => "If enabled, playing of background music is enabled. If disabled, no background music will be played.", |
354 | on_changed => sub { |
912 | on_changed => sub { |
355 | $CFG->{bgm_enable} = $_[1]; |
913 | $CFG->{bgm_enable} = $_[1]; |
|
|
914 | $CONN->update_fx_want if $CONN; |
|
|
915 | 0 |
356 | } |
916 | } |
357 | ); |
917 | ); |
358 | $hbox->add (new CFClient::UI::Slider |
918 | $table->add_at (2, $row++, new CFPlus::UI::Slider |
359 | expand => 1, range => [$CFG->{bgm_volume}, 0, 1, 0, 1/128], |
919 | expand => 1, range => [$CFG->{bgm_volume}, 0, 1, 0, 1/128], |
360 | tooltip => "The volume of the background music. Changes are instant.", |
920 | tooltip => "The volume of the background music. Changes are instant.", |
361 | on_changed => sub { |
921 | on_changed => sub { $CFG->{bgm_volume} = $_[1]; audio_music_update_volume; 0 } |
362 | $CFG->{bgm_volume} = $_[1]; |
|
|
363 | CFClient::MixMusic::volume $_[1] * 128; |
|
|
364 | } |
|
|
365 | ); |
922 | ); |
366 | |
923 | |
367 | $table->add (1, $row++, new CFClient::UI::Button |
924 | $table->add_at (1, $row++, new CFPlus::UI::Button |
368 | expand => 1, align => 0, text => "Apply", |
925 | c_colspan => 2, expand => 1, align => 0, text => "Apply", |
369 | tooltip => "Apply the audio settings", |
926 | tooltip => "Apply the audio settings", |
370 | on_activate => sub { |
927 | on_activate => sub { |
371 | audio_shutdown (); |
928 | audio_shutdown (); |
372 | audio_init (); |
929 | audio_init (); |
|
|
930 | 0 |
373 | } |
931 | } |
374 | ); |
932 | ); |
375 | |
933 | |
376 | $vbox |
934 | $vbox |
377 | } |
|
|
378 | |
|
|
379 | sub set_stats_window_fontsize { |
|
|
380 | for (values %{$STATWIDS}) { |
|
|
381 | $_->set_fontsize ($::CFG->{stat_fontsize}); |
|
|
382 | } |
|
|
383 | } |
935 | } |
384 | |
936 | |
385 | sub set_gauge_window_fontsize { |
937 | sub set_gauge_window_fontsize { |
386 | for (map { $GAUGES->{$_} } grep { $_ ne 'win' } keys %{$GAUGES}) { |
938 | for (map { $GAUGES->{$_} } grep { $_ ne 'win' } keys %{$GAUGES}) { |
387 | $_->set_fontsize ($::CFG->{gauge_fontsize}); |
939 | $_->set_fontsize ($::CFG->{gauge_fontsize}); |
… | |
… | |
389 | } |
941 | } |
390 | |
942 | |
391 | sub make_gauge_window { |
943 | sub make_gauge_window { |
392 | my $gh = int $HEIGHT * $CFG->{gauge_size}; |
944 | my $gh = int $HEIGHT * $CFG->{gauge_size}; |
393 | |
945 | |
394 | my $win = new CFClient::UI::Frame ( |
946 | my $win = new CFPlus::UI::Frame ( |
395 | force_x => 0, |
947 | force_x => 0, |
396 | force_y => "max", |
948 | force_y => "max", |
397 | force_w => $WIDTH, |
949 | force_w => $WIDTH, |
398 | force_h => $gh, |
950 | force_h => $gh, |
399 | ); |
951 | ); |
400 | |
952 | |
401 | $win->add (my $hbox = new CFClient::UI::HBox |
953 | $win->add (my $hbox = new CFPlus::UI::HBox |
402 | children => [ |
954 | children => [ |
403 | (new CFClient::UI::HBox expand => 1), |
955 | (new CFPlus::UI::HBox expand => 1), |
404 | (new CFClient::UI::VBox children => [ |
956 | (new CFPlus::UI::VBox children => [ |
405 | (new CFClient::UI::Empty expand => 1), |
957 | (new CFPlus::UI::Empty expand => 1), |
406 | (new CFClient::UI::Frame bg => [0, 0, 0, 0.4], child => ($FLOORBOX = new CFClient::UI::Table)), |
958 | (new CFPlus::UI::Frame bg => [0, 0, 0, 0.4], child => ($FLOORBOX = new CFPlus::UI::Table)), |
407 | ]), |
959 | ]), |
408 | (my $vbox = new CFClient::UI::VBox), |
960 | (my $vbox = new CFPlus::UI::VBox), |
409 | ], |
961 | ], |
410 | ); |
962 | ); |
411 | |
963 | |
412 | $vbox->add (new CFClient::UI::HBox |
964 | $vbox->add (new CFPlus::UI::HBox |
413 | expand => 1, |
965 | expand => 1, |
414 | children => [ |
966 | children => [ |
415 | (new CFClient::UI::Empty expand => 1), |
967 | (new CFPlus::UI::Empty expand => 1), |
416 | (my $hb = new CFClient::UI::HBox), |
968 | (my $hb = new CFPlus::UI::HBox), |
417 | ], |
969 | ], |
418 | ); |
970 | ); |
419 | |
971 | |
420 | $hb->add (my $hg = new CFClient::UI::Gauge type => 'hp', |
972 | $hb->add (my $hg = new CFPlus::UI::Gauge type => 'hp', tooltip => "#stat_health"); |
421 | tooltip => "<b>Health points</b>. Measures of how much damage you can take before dying. Hit points are determined from your level and are influenced by the value of your Con. Hp value may range between 1 to beyond 500 and higher values indicate a greater ability to withstand punishment."); |
|
|
422 | $hb->add (my $mg = new CFClient::UI::Gauge type => 'mana', |
973 | $hb->add (my $mg = new CFPlus::UI::Gauge type => 'mana', tooltip => "#stat_mana"); |
423 | tooltip => "<b>Spell points</b>. Measures of how much \"fuel\" you have for casting spells and incantations. Mana is calculated from your level and your Pow. Mana values can range between 1 to beyond 500 (glowing crystals can increase the current spell points beyond your normal maximum). Higher values indicate greater amounts of mana."); |
|
|
424 | $hb->add (my $gg = new CFClient::UI::Gauge type => 'grace', |
974 | $hb->add (my $gg = new CFPlus::UI::Gauge type => 'grace', tooltip => "#stat_grace"); |
425 | tooltip => "<b>Grace points</b> - how favored you are by your god. In game terms, how much divine magic you can cast. Your level, Wis and Pow effect what the value of grace is. Prayong on an altar of your god can increase this value beyond your normal maximum. Grace can take on large positive and negative values. Positive values indicate favor by the gods."); |
|
|
426 | $hb->add (my $fg = new CFClient::UI::Gauge type => 'food', |
975 | $hb->add (my $fg = new CFPlus::UI::Gauge type => 'food', tooltip => "#stat_food"); |
427 | tooltip => "<b>Food</b>. Ranges between 0 (starving) and 999 (satiated). At a value of 0 the character begins to die. Some magic can speed up or slow down the character digestion. Healing wounds will speed up digestion too."); |
|
|
428 | |
976 | |
429 | $vbox->add (my $exp = new CFClient::UI::Label valign => 0, align => 1, can_hover => 1, can_events => 1, |
977 | $vbox->add (my $exp = new CFPlus::UI::Label valign => 0, align => 1, can_hover => 1, can_events => 1, tooltip => "#stat_exp"); |
430 | tooltip => "<b>Experience points and overall level</b> - experience is increased as a reward for appropriate action (such as killing monsters) and may decrease as a result of a magical attack or dying. Level is directly derived from the experience value. As the level of the character increases, the character becomes able to succeed at more difficult tasks. A character's level starts at a value of 0 and may range up beyond 100."); |
|
|
431 | $vbox->add (my $rng = new CFClient::UI::Label valign => 0, align => 1, can_hover => 1, can_events => 1, |
978 | $vbox->add (my $rng = new CFPlus::UI::Label valign => 0, align => 1, can_hover => 1, can_events => 1, tooltip => "#stat_ranged"); |
432 | tooltip => "<b>Ranged attack</b> - how you attack when you press shift-cursor (spell, skill, weapon etc.)"); |
|
|
433 | |
979 | |
434 | $GAUGES = { |
980 | $GAUGES = { |
435 | exp => $exp, win => $win, range => $rng, |
981 | exp => $exp, win => $win, range => $rng, |
436 | food => $fg, mana => $mg, hp => $hg, grace => $gg |
982 | food => $fg, mana => $mg, hp => $hg, grace => $gg |
437 | }; |
983 | }; |
… | |
… | |
439 | &set_gauge_window_fontsize; |
985 | &set_gauge_window_fontsize; |
440 | |
986 | |
441 | $win |
987 | $win |
442 | } |
988 | } |
443 | |
989 | |
|
|
990 | sub debug_setup { |
|
|
991 | my $table = new CFPlus::UI::Table; |
444 | |
992 | |
|
|
993 | $table->add_at (0, 0, new CFPlus::UI::Label text => "Widget Borders"); |
|
|
994 | $table->add_at (1, 0, new CFPlus::UI::CheckBox on_changed => sub { $ENV{CFPLUS_DEBUG} ^= 1; 0 }); |
|
|
995 | $table->add_at (0, 1, new CFPlus::UI::Label text => "Tooltip Widget Info"); |
|
|
996 | $table->add_at (1, 1, new CFPlus::UI::CheckBox on_changed => sub { $ENV{CFPLUS_DEBUG} ^= 2; 0 }); |
|
|
997 | $table->add_at (0, 2, new CFPlus::UI::Label text => "Show FPS"); |
|
|
998 | $table->add_at (1, 2, new CFPlus::UI::CheckBox on_changed => sub { $ENV{CFPLUS_DEBUG} ^= 4; 0 }); |
|
|
999 | $table->add_at (0, 3, new CFPlus::UI::Label text => "Suppress Tooltips"); |
|
|
1000 | $table->add_at (1, 3, new CFPlus::UI::CheckBox on_changed => sub { $ENV{CFPLUS_DEBUG} ^= 8; 0 }); |
|
|
1001 | $table->add_at (0, 4, new CFPlus::UI::Button text => "die on click(tm)", on_activate => sub { &CFPlus::debug() } ); |
|
|
1002 | |
|
|
1003 | $table->add_at (0, 5, new CFPlus::UI::TextEdit text => "line1\0152\0153");#d# |
|
|
1004 | |
|
|
1005 | $table->add_at (7,7, my $t = new CFPlus::UI::Table expand => 0); |
|
|
1006 | $t->add_at (0,0, new CFPlus::UI::Label text => "a a a a", c_rowspan => 1, c_colspan => 2); |
|
|
1007 | $t->add_at (2,0, new CFPlus::UI::Label text => "b\nb", c_rowspan => 2, c_colspan => 1); |
|
|
1008 | $t->add_at (1,2, new CFPlus::UI::Label text => "c c c c", c_rowspan => 1, c_colspan => 2); |
|
|
1009 | $t->add_at (0,1, new CFPlus::UI::Label text => "d\nd", c_rowspan => 2, c_colspan => 1); |
|
|
1010 | $t->add_at (1,1, new CFPlus::UI::Label text => "e"); |
|
|
1011 | |
|
|
1012 | $table->add_at (7, 6, my $c = new CFPlus::UI::Canvas); |
|
|
1013 | |
|
|
1014 | $c->add_items ({ |
|
|
1015 | type => "line_loop", |
|
|
1016 | color => [0, 1, 0], |
|
|
1017 | width => 9, |
|
|
1018 | coord_mode => "abs", |
|
|
1019 | coord => [[10, 5], [5, 50], [20, 5], [5, 60]], |
|
|
1020 | }); |
|
|
1021 | |
|
|
1022 | $c->add_items ({ |
|
|
1023 | type => "lines", |
|
|
1024 | color => [1, 1, 0], |
|
|
1025 | width => 2, |
|
|
1026 | coord_mode => "rel", |
|
|
1027 | coord => [[0,0], [1,1], [1,0], [0,1]], |
|
|
1028 | }); |
|
|
1029 | |
|
|
1030 | $c->add_items ({ |
|
|
1031 | type => "polygon", |
|
|
1032 | color => [0, 0.43, 0], |
|
|
1033 | width => 2, |
|
|
1034 | coord_mode => "rel", |
|
|
1035 | coord => [[0,0.2], [1,.4], [1,.6], [0,.8]], |
|
|
1036 | }); |
|
|
1037 | |
|
|
1038 | $table |
|
|
1039 | } |
|
|
1040 | |
445 | sub make_stats_window { |
1041 | sub stats_window { |
446 | my $tgw = new CFClient::UI::FancyFrame |
1042 | my $r = new CFPlus::UI::ScrolledWindow ( |
447 | y => $HEIGHT * (2/8), |
1043 | expand => 1, |
448 | x => "max", |
1044 | scroll_y => 1 |
449 | title => "Stats", |
1045 | ); |
450 | name => "stats_window"; |
1046 | $r->add (my $vb = new CFPlus::UI::VBox); |
451 | |
1047 | |
452 | $tgw->add (new CFClient::UI::Window child => my $vb = new CFClient::UI::VBox); |
1048 | $vb->add (new CFPlus::UI::FancyFrame |
|
|
1049 | label => "Player", |
|
|
1050 | child => (my $pi = new CFPlus::UI::VBox), |
|
|
1051 | ); |
|
|
1052 | |
453 | $vb->add ($STATWIDS->{title} = new CFClient::UI::Label valign => 0, align => -1, text => "Title:", expand => 1, |
1053 | $pi->add ($STATWIDS->{title} = new CFPlus::UI::Label valign => 0, align => -1, text => "Title:", expand => 1, |
454 | can_hover => 1, can_events => 1, |
1054 | can_hover => 1, can_events => 1, |
455 | tooltip => "Your name and title. You can change your title by using the <b>title</b> command, if supported by the server."); |
1055 | tooltip => "Your name and title. You can change your title by using the <b>title</b> command, if supported by the server."); |
456 | $vb->add ($STATWIDS->{map} = new CFClient::UI::Label valign => 0, align => -1, text => "Map:", expand => 1, |
1056 | $pi->add ($STATWIDS->{map} = new CFPlus::UI::Label valign => 0, align => -1, text => "Map:", expand => 1, |
457 | can_hover => 1, can_events => 1, |
1057 | can_hover => 1, can_events => 1, |
458 | tooltip => "The map you are currently on (if supported by the server)."); |
1058 | tooltip => "The map you are currently on (if supported by the server)."); |
459 | |
1059 | |
460 | $vb->add (my $hb0 = new CFClient::UI::HBox); |
1060 | $pi->add (my $hb0 = new CFPlus::UI::HBox); |
461 | $hb0->add ($STATWIDS->{weight} = new CFClient::UI::Label valign => 0, align => -1, text => "Weight:", expand => 1, |
1061 | $hb0->add ($STATWIDS->{weight} = new CFPlus::UI::Label valign => 0, align => -1, text => "Weight:", expand => 1, |
462 | can_hover => 1, can_events => 1, |
1062 | can_hover => 1, can_events => 1, |
463 | tooltip => "The weight of the player including all inventory items."); |
1063 | tooltip => "The weight of the player including all inventory items."); |
464 | $hb0->add ($STATWIDS->{m_weight} = new CFClient::UI::Label valign => 0, align => -1, text => "Max weight:", expand => 1, |
1064 | $hb0->add ($STATWIDS->{m_weight} = new CFPlus::UI::Label valign => 0, align => -1, text => "Max weight:", expand => 1, |
465 | can_hover => 1, can_events => 1, |
1065 | can_hover => 1, can_events => 1, |
466 | tooltip => "The weight limit: you cannot carry more than this."); |
1066 | tooltip => "The weight limit: you cannot carry more than this."); |
467 | |
1067 | |
468 | |
1068 | $vb->add (new CFPlus::UI::FancyFrame |
|
|
1069 | label => "Primary/Secondary Statistics", |
469 | $vb->add (my $hb = new CFClient::UI::HBox expand => 1); |
1070 | child => (my $hb = new CFPlus::UI::HBox expand => 1), |
|
|
1071 | ); |
470 | $hb->add (my $tbl = new CFClient::UI::Table expand => 1); |
1072 | $hb->add (my $tbl = new CFPlus::UI::Table expand => 1); |
471 | |
1073 | |
472 | my $color2 = [1, 1, 0]; |
1074 | my $color2 = [1, 1, 0]; |
473 | |
1075 | |
474 | for ( |
1076 | for ( |
475 | [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"], |
1077 | [0, 0, st_str => "Str", 30], |
476 | [0, 1, st_dex => "Dex", 30, "<b>Dexterity</b>, your physical agility. Determines chance of being hit and affects armor class and speed"], |
1078 | [0, 1, st_dex => "Dex", 30], |
477 | [0, 2, st_con => "Con", 30, "<b>Constitution</b>, physical health and toughness. Determines how many healthpoints you can have"], |
1079 | [0, 2, st_con => "Con", 30], |
478 | [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"], |
1080 | [0, 3, st_int => "Int", 30], |
479 | [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"], |
1081 | [0, 4, st_wis => "Wis", 30], |
480 | [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"], |
1082 | [0, 5, st_pow => "Pow", 30], |
481 | [0, 6, st_cha => "Cha", 30, "<b>Charisma</b>, how well you are received by NPCs. Affects buying and selling prices in shops."], |
1083 | [0, 6, st_cha => "Cha", 30], |
482 | |
1084 | |
483 | [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."], |
1085 | [2, 0, st_wc => "Wc", -120], |
484 | [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."], |
1086 | [2, 1, st_ac => "Ac", -120], |
485 | [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."], |
1087 | [2, 2, st_dam => "Dam", 120], |
486 | [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."], |
1088 | [2, 3, st_arm => "Arm", 120], |
487 | [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."], |
1089 | [2, 4, st_spd => "Spd", 10.54], |
488 | [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."], |
1090 | [2, 5, st_wspd => "WSp", 10.54], |
489 | ) { |
1091 | ) { |
490 | my ($col, $row, $id, $label, $template, $tooltip) = @$_; |
1092 | my ($col, $row, $id, $label, $template) = @$_; |
491 | |
1093 | |
492 | $tbl->add ($col , $row, $STATWIDS->{$id} = new CFClient::UI::Label |
1094 | $tbl->add_at ($col , $row, $STATWIDS->{$id} = new CFPlus::UI::Label |
493 | font => $FONT_FIXED, can_hover => 1, can_events => 1, valign => 0, align => +1, template => $template, tooltip => $tooltip); |
1095 | font => $FONT_FIXED, can_hover => 1, can_events => 1, valign => 0, |
|
|
1096 | align => +1, template => $template, tooltip => "#stat_$label"); |
494 | $tbl->add ($col + 1, $row, $STATWIDS->{"$id\_lbl"} = new CFClient::UI::Label |
1097 | $tbl->add_at ($col + 1, $row, $STATWIDS->{"$id\_lbl"} = new CFPlus::UI::Label |
495 | font => $FONT_FIXED, can_hover => 1, can_events => 1, fg => $color2, valign => 0, align => -1, text => $label, tooltip => $tooltip); |
1098 | font => $FONT_FIXED, can_hover => 1, can_events => 1, fg => $color2, valign => 0, |
|
|
1099 | align => -1, text => $label, tooltip => "#stat_$label"); |
496 | } |
1100 | } |
497 | |
1101 | |
|
|
1102 | $vb->add (new CFPlus::UI::FancyFrame |
|
|
1103 | label => "Resistancies", |
498 | $hb->add (my $tbl2 = new CFClient::UI::Table expand => 1); |
1104 | child => (my $tbl2 = new CFPlus::UI::Table expand => 1), |
|
|
1105 | ); |
499 | |
1106 | |
500 | my $row = 0; |
1107 | my $row = 0; |
501 | my $col = 0; |
1108 | my $col = 0; |
502 | |
1109 | |
503 | my %resist_names = ( |
1110 | my %resist_names = ( |
|
|
1111 | slow => ["Slow", |
504 | 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.)", |
1112 | "<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.)"], |
|
|
1113 | holyw => ["Holy Word", |
505 | 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.)", |
1114 | "<b>Holy Word</b> (resistance you against getting the fear when someone whose god doesn't like you spells the holy word on you.)"], |
|
|
1115 | conf => ["Confusion", |
506 | conf => "<b>Confusion</b> (If you are hit by confusion you will move into random directions, and likely into monsters.)", |
1116 | "<b>Confusion</b> (If you are hit by confusion you will move into random directions, and likely into monsters.)"], |
|
|
1117 | fire => ["Fire", |
507 | fire => "<b>Fire</b> (just your resistance to fire spells like burning hands, dragonbreath, meteor swarm fire, ...)", |
1118 | "<b>Fire</b> (just your resistance to fire spells like burning hands, dragonbreath, meteor swarm fire, ...)"], |
|
|
1119 | depl => ["Depletion", |
508 | depl => "<b>Depletion</b> (some monsters and other effects can cause stats depletion)", |
1120 | "<b>Depletion</b> (some monsters and other effects can cause stats depletion)"], |
|
|
1121 | magic => ["Magic", |
509 | magic => "<b>Magic</b> (resistance to magic spells like magic missile or similar)", |
1122 | "<b>Magic</b> (resistance to magic spells like magic missile or similar)"], |
|
|
1123 | drain => ["Draining", |
510 | drain => "<b>Draining</b> (some monsters (e.g. vampires) and other effects can steal experience)", |
1124 | "<b>Draining</b> (some monsters (e.g. vampires) and other effects can steal experience)"], |
|
|
1125 | acid => ["Acid", |
511 | acid => "<b>Acid</b> (resistance to acid, acid hurts pretty much and also corrodes your weapons)", |
1126 | "<b>Acid</b> (resistance to acid, acid hurts pretty much and also corrodes your weapons)"], |
|
|
1127 | pois => ["Poison", |
512 | pois => "<b>Poison</b> (resistance to getting poisoned)", |
1128 | "<b>Poison</b> (resistance to getting poisoned)"], |
|
|
1129 | para => ["Paralysation", |
513 | para => "<b>Paralysation</b> (this resistance affects the chance you get paralysed)", |
1130 | "<b>Paralysation</b> (this resistance affects the chance you get paralysed)"], |
|
|
1131 | deat => ["Death", |
514 | deat => "<b>Death</b> (resistance against death spells)", |
1132 | "<b>Death</b> (resistance against death spells)"], |
515 | phys => "<b>Physical</b> (this is the resistance against physical attacks, like when a monster hit you in melee combat)", |
1133 | phys => ["Physical", |
|
|
1134 | "<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.)"], |
|
|
1135 | blind => ["Blind", |
516 | blind => "<b>Blind</b> (blind resistance affects the chance of a successful blinding attack)", |
1136 | "<b>Blind</b> (blind resistance affects the chance of a successful blinding attack)"], |
|
|
1137 | fear => ["Fear", |
517 | 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)", |
1138 | "<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)"], |
|
|
1139 | tund => ["Turn undead", |
518 | tund => "<b>Turn undead</b> (affects your resistancy to various forms of 'turn undead' spells. Only relevant when you are, in fact, undead...", |
1140 | "<b>Turn undead</b> (affects your resistancy to various forms of 'turn undead' spells. Only relevant when you are, in fact, undead..."], |
|
|
1141 | elec => ["Electricity", |
519 | elec => "<b>Electricity</b> (resistance against electricity, spells like large lightning, small lightning, ...)", |
1142 | "<b>Electricity</b> (resistance against electricity, spells like large lightning, small lightning, ...)"], |
|
|
1143 | cold => ["Cold", |
520 | cold => "<b>Cold</b> (this is your resistance against cold spells like icestorm, snowstorm, ...)", |
1144 | "<b>Cold</b> (this is your resistance against cold spells like icestorm, snowstorm, ...)"], |
|
|
1145 | ghit => ["Ghost hit", |
521 | ghit => "<b>Ghost hit</b> (special attack used by ghosts and ghost-like beings)", |
1146 | "<b>Ghost hit</b> (special attack used by ghosts and ghost-like beings)"], |
522 | ); |
1147 | ); |
|
|
1148 | |
523 | for (qw/slow holyw conf fire depl magic |
1149 | for (qw/slow holyw conf fire depl magic |
524 | drain acid pois para deat phys |
1150 | drain acid pois para deat phys |
525 | blind fear tund elec cold ghit/) |
1151 | blind fear tund elec cold ghit/) |
526 | { |
1152 | { |
527 | $tbl2->add ($col, $row, |
1153 | $tbl2->add_at ($col, $row, |
528 | $STATWIDS->{"res_$_"} = |
1154 | $STATWIDS->{"res_$_"} = |
529 | new CFClient::UI::Label |
1155 | new CFPlus::UI::Label |
530 | font => $FONT_FIXED, |
1156 | font => $FONT_FIXED, |
531 | template => "-100%", |
1157 | template => "-100%", |
532 | align => +1, |
1158 | align => +1, |
533 | valign => 0, |
1159 | valign => 0, |
534 | can_events => 1, |
1160 | can_events => 1, |
535 | can_hover => 1, |
1161 | can_hover => 1, |
536 | tooltip => $resist_names{$_}, |
1162 | tooltip => $resist_names{$_}->[1], |
537 | ); |
1163 | ); |
538 | $tbl2->add ($col + 1, $row, new CFClient::UI::Image |
1164 | $tbl2->add_at ($col + 1, $row, new CFPlus::UI::Image |
539 | font => $FONT_FIXED, |
1165 | font => $FONT_FIXED, |
540 | can_hover => 1, |
1166 | can_hover => 1, |
541 | can_events => 1, |
1167 | can_events => 1, |
542 | image => "ui/resist/resist_$_.png", |
1168 | path => "ui/resist/resist_$_.png", |
543 | tooltip => $resist_names{$_}, |
1169 | tooltip => $resist_names{$_}->[1], |
|
|
1170 | ); |
|
|
1171 | $tbl2->add_at ($col + 2, $row, new CFPlus::UI::Label |
|
|
1172 | text => $resist_names{$_}->[0], |
|
|
1173 | font => $FONT_FIXED, |
|
|
1174 | can_hover => 1, |
|
|
1175 | can_events => 1, |
|
|
1176 | tooltip => $resist_names{$_}->[1], |
544 | ); |
1177 | ); |
545 | |
1178 | |
546 | $row++; |
1179 | $row++; |
547 | if ($row % 6 == 0) { |
1180 | if ($row % 6 == 0) { |
548 | $col += 2; |
1181 | $col += 3; |
549 | $row = 0; |
1182 | $row = 0; |
550 | } |
1183 | } |
551 | } |
1184 | } |
552 | |
1185 | |
553 | &set_stats_window_fontsize; |
|
|
554 | update_stats_window ({}); |
1186 | #update_stats_window ({}); |
555 | |
1187 | |
|
|
1188 | $r |
|
|
1189 | } |
|
|
1190 | |
|
|
1191 | sub skill_window { |
|
|
1192 | my $sw = new CFPlus::UI::ScrolledWindow (expand => 1); |
|
|
1193 | $sw->add ($STATWIDS->{skill_tbl} = new CFPlus::UI::Table expand => 1, col_expand => [0, 0, 1, 0, 0, 1]); |
556 | $tgw |
1194 | $sw |
557 | } |
1195 | } |
558 | |
1196 | |
559 | sub formsep($) { |
1197 | sub formsep($) { |
560 | scalar reverse join ",", unpack "(A3)*", reverse $_[0] * 1 |
1198 | scalar reverse join ",", unpack "(A3)*", reverse $_[0] * 1 |
561 | } |
1199 | } |
562 | |
1200 | |
563 | sub update_stats_window { |
|
|
564 | my ($stats) = @_; |
|
|
565 | |
|
|
566 | # I love text protocols... |
|
|
567 | |
|
|
568 | my $hp = $stats->{+CS_STAT_HP} * 1; |
|
|
569 | my $hp_m = $stats->{+CS_STAT_MAXHP} * 1; |
|
|
570 | my $sp = $stats->{+CS_STAT_SP} * 1; |
|
|
571 | my $sp_m = $stats->{+CS_STAT_MAXSP} * 1; |
|
|
572 | my $fo = $stats->{+CS_STAT_FOOD} * 1; |
|
|
573 | my $fo_m = 999; |
|
|
574 | my $gr = $stats->{+CS_STAT_GRACE} * 1; |
|
|
575 | my $gr_m = $stats->{+CS_STAT_MAXGRACE} * 1; |
|
|
576 | |
|
|
577 | $GAUGES->{hp} ->set_value ($hp, $hp_m); |
|
|
578 | $GAUGES->{mana} ->set_value ($sp, $sp_m); |
|
|
579 | $GAUGES->{food} ->set_value ($fo, $fo_m); |
|
|
580 | $GAUGES->{grace} ->set_value ($gr, $gr_m); |
|
|
581 | $GAUGES->{exp} ->set_text ("Exp: " . (formsep $stats->{+CS_STAT_EXP64}) |
|
|
582 | . " (lvl " . ($stats->{+CS_STAT_LEVEL} * 1) . ")"); |
|
|
583 | my $rng = $stats->{+CS_STAT_RANGE}; |
|
|
584 | $rng =~ s/^Range: //; # thank you so much dear server |
|
|
585 | $GAUGES->{range} ->set_text ("Rng: " . $rng); |
|
|
586 | my $title = $stats->{+CS_STAT_TITLE}; |
|
|
587 | $title =~ s/^Player: //; |
|
|
588 | $STATWIDS->{title} ->set_text ("Title: " . $title); |
|
|
589 | |
|
|
590 | $STATWIDS->{st_str} ->set_text (sprintf "%d" , $stats->{+CS_STAT_STR}); |
|
|
591 | $STATWIDS->{st_dex} ->set_text (sprintf "%d" , $stats->{+CS_STAT_DEX}); |
|
|
592 | $STATWIDS->{st_con} ->set_text (sprintf "%d" , $stats->{+CS_STAT_CON}); |
|
|
593 | $STATWIDS->{st_int} ->set_text (sprintf "%d" , $stats->{+CS_STAT_INT}); |
|
|
594 | $STATWIDS->{st_wis} ->set_text (sprintf "%d" , $stats->{+CS_STAT_WIS}); |
|
|
595 | $STATWIDS->{st_pow} ->set_text (sprintf "%d" , $stats->{+CS_STAT_POW}); |
|
|
596 | $STATWIDS->{st_cha} ->set_text (sprintf "%d" , $stats->{+CS_STAT_CHA}); |
|
|
597 | $STATWIDS->{st_wc} ->set_text (sprintf "%d" , $stats->{+CS_STAT_WC}); |
|
|
598 | $STATWIDS->{st_ac} ->set_text (sprintf "%d" , $stats->{+CS_STAT_AC}); |
|
|
599 | $STATWIDS->{st_dam} ->set_text (sprintf "%d" , $stats->{+CS_STAT_DAM}); |
|
|
600 | $STATWIDS->{st_arm} ->set_text (sprintf "%d" , $stats->{+CS_STAT_ARMOUR}); |
|
|
601 | $STATWIDS->{st_spd} ->set_text (sprintf "%.1f", $stats->{+CS_STAT_SPEED}); |
|
|
602 | $STATWIDS->{st_wspd}->set_text (sprintf "%.1f", $stats->{+CS_STAT_WEAP_SP}); |
|
|
603 | |
|
|
604 | $STATWIDS->{m_weight}->set_text (sprintf "Max weight: %.1fkg", $stats->{+CS_STAT_WEIGHT_LIM} / 1000); |
|
|
605 | |
|
|
606 | # TODO: replace by CS_STAT_RES_xxx constants |
|
|
607 | my %tbl = ( |
|
|
608 | phys => 100, |
|
|
609 | magic => 101, |
|
|
610 | fire => 102, |
|
|
611 | elec => 103, |
|
|
612 | cold => 104, |
|
|
613 | conf => 105, |
|
|
614 | acid => 106, |
|
|
615 | drain => 107, |
|
|
616 | ghit => 108, |
|
|
617 | pois => 109, |
|
|
618 | slow => 110, |
|
|
619 | para => 111, |
|
|
620 | tund => 112, |
|
|
621 | fear => 113, |
|
|
622 | depl => 113, |
|
|
623 | deat => 115, |
|
|
624 | holyw => 116, |
|
|
625 | blind => 117, |
|
|
626 | ); |
|
|
627 | |
|
|
628 | $STATWIDS->{"res_$_"}->set_text (sprintf "%d%", $stats->{$tbl{$_}}) |
|
|
629 | for keys %tbl; |
|
|
630 | } |
|
|
631 | |
|
|
632 | my $METASERVER_ATIME; |
1201 | my $METASERVER_ATIME; |
633 | |
1202 | |
634 | sub update_metaserver { |
1203 | sub update_metaserver { |
|
|
1204 | my ($metaserver_dialog) = @_; |
|
|
1205 | |
|
|
1206 | $METASERVER = $metaserver_dialog |
|
|
1207 | if defined $metaserver_dialog; |
|
|
1208 | |
635 | return if $METASERVER_ATIME > time; |
1209 | return if $METASERVER_ATIME > time; |
636 | $METASERVER_ATIME = time + 60; |
1210 | $METASERVER_ATIME = time + 60; |
637 | |
1211 | |
638 | my $table = $METASERVER->{table}; |
1212 | my $table = $METASERVER->{table}; |
639 | $table->clear; |
1213 | $table->clear; |
640 | $table->add (0, 0, my $label = new CFClient::UI::Label max_w => $WIDTH * 0.8, text => "fetching server list..."); |
1214 | $table->add_at (0, 0, my $label = new CFPlus::UI::Label max_w => $WIDTH * 0.8, text => "fetching server list..."); |
641 | |
1215 | |
642 | my $buf; |
1216 | my $ok = 0; |
643 | |
1217 | |
644 | my $fh = new IO::Socket::INET PeerHost => $META_SERVER, Blocking => 0; |
1218 | CFPlus::background { |
|
|
1219 | my $ua = CFPlus::lwp_useragent; |
645 | |
1220 | |
646 | unless ($fh) { |
1221 | CFPlus::background_msg CFPlus::from_json +(CFPlus::lwp_check $ua->get ($META_SERVER))->decoded_content; |
647 | $label->set_text ("unable to contact metaserver: $!"); |
1222 | } sub { |
648 | return; |
1223 | my ($msg) = @_; |
649 | } |
1224 | if ($msg) { |
650 | |
|
|
651 | Event->io (fd => $fh, poll => 'r', cb => sub { |
|
|
652 | my $res = sysread $fh, $buf, 8192, length $buf; |
|
|
653 | |
|
|
654 | if (!defined $res) { |
|
|
655 | $_[0]->w->cancel; |
|
|
656 | $label->set_text ("error while retrieving server list: $!"); |
|
|
657 | } elsif ($res == 0) { |
|
|
658 | $_[0]->w->cancel; |
|
|
659 | status "server list retrieved"; |
|
|
660 | |
|
|
661 | utf8::decode $buf if utf8::valid $buf; |
|
|
662 | |
|
|
663 | $table->clear; |
1225 | $table->clear; |
664 | |
1226 | |
|
|
1227 | my @tip = ( |
|
|
1228 | "The current number of users logged in on the server.", |
|
|
1229 | "The hostname of the server.", |
|
|
1230 | "The time this server has been running without being restarted.", |
|
|
1231 | "The server software version - a '+' indicates a Crossfire+ server.", |
|
|
1232 | "Short information about this server provided by its admins.", |
|
|
1233 | ); |
665 | my @col = qw(Use #Users Host Uptime Version Description); |
1234 | my @col = qw(#Users Host Uptime Version Description); |
666 | $table->add ($_, 0, new CFClient::UI::Label align => 0, fg => [1, 1, 0], text => $col[$_]) |
1235 | $table->add_at ($_, 0, new CFPlus::UI::Label |
|
|
1236 | can_hover => 1, can_events => 1, |
|
|
1237 | align => 0, fg => [1, 1, 0], |
|
|
1238 | text => $col[$_], tooltip => $tip[$_]) |
667 | for 0 .. $#col; |
1239 | for 0 .. $#col; |
668 | |
1240 | |
669 | my @align = qw(1 0 1 1 -1); |
1241 | my @align = qw(1 0 1 1 -1); |
670 | |
1242 | |
671 | my $y = 0; |
1243 | my $y = 0; |
672 | for my $m (sort { $b->[3] <=> $a->[3] } map [split /\|/], split /\015?\012/, $buf) { |
1244 | for my $m (@{ $msg->{servers} }) { |
673 | my ($ip, $last, $host, $users, $version, $desc, $ibytes, $obytes, $uptime) = @$m; |
1245 | my ($ip, $last, $host, $users, $version, $desc, $ibytes, $obytes, $uptime, $highlight) = |
|
|
1246 | @$m{qw(ip age hostname users version description ibytes obytes uptime highlight)}; |
674 | |
1247 | |
675 | for ($desc) { |
1248 | for ($desc) { |
676 | s/<br>/\n/gi; |
1249 | s/<br>/\n/gi; |
677 | s/<li>/\n· /gi; |
1250 | s/<li>/\n· /gi; |
678 | s/<.*?>//sgi; |
1251 | s/<.*?>//sgi; |
679 | s/&/&/g; |
1252 | s/&/&/g; |
680 | s/</</g; |
1253 | s/</</g; |
681 | s/>/>/g; |
1254 | s/>/>/g; |
682 | } |
1255 | } |
683 | |
1256 | |
684 | $uptime = sprintf "%dd %02d:%02d:%02d", |
1257 | $uptime = sprintf "%dd %02d:%02d:%02d", |
685 | (int $m->[8] / 86400), |
1258 | (int $uptime / 86400), |
686 | (int $m->[8] / 3600) % 24, |
1259 | (int $uptime / 3600) % 24, |
687 | (int $m->[8] / 60) % 60, |
1260 | (int $uptime / 60) % 60, |
688 | $m->[8] % 60; |
1261 | $uptime % 60; |
689 | |
1262 | |
690 | $m = [$users, $host, $uptime, $version, $desc]; |
1263 | $m = [$users, $host, $uptime, $version, $desc]; |
691 | |
1264 | |
692 | $y++; |
1265 | $y++; |
693 | |
1266 | |
694 | $table->add (0, $y, new CFClient::UI::VBox children => [ |
1267 | $table->add_at (scalar @$m, $y, new CFPlus::UI::VBox children => [ |
695 | (new CFClient::UI::Button text => "Use", on_activate => sub { |
1268 | (new CFPlus::UI::Button |
|
|
1269 | text => "Use", |
|
|
1270 | tooltip => "Put this server into the <b>Host:Port</b> field", |
|
|
1271 | on_activate => sub { |
696 | $HOST_ENTRY->set_text ($CFG->{host} = $host); |
1272 | $HOST_ENTRY->set_text ($CFG->{profile}{default}{host} = $host); |
697 | $METASERVER->toggle_visibility; |
1273 | $METASERVER->hide; |
|
|
1274 | 0 |
|
|
1275 | }, |
698 | }), |
1276 | ), |
699 | (new CFClient::UI::Empty expand => 1), |
1277 | (new CFPlus::UI::Empty expand => 1), |
700 | ]); |
1278 | ]); |
701 | |
1279 | |
702 | $table->add ($_ + 1, $y, new CFClient::UI::Label |
1280 | $table->add_at ($_, $y, new CFPlus::UI::Label |
703 | ellipsise => 0, align => $align[$_], text => $m->[$_], fontsize => 0.8) |
1281 | max_w => $::WIDTH * 0.4, |
|
|
1282 | ellipsise => 0, |
|
|
1283 | align => $align[$_], |
|
|
1284 | text => $m->[$_], |
|
|
1285 | tooltip => $tip[$_], |
|
|
1286 | fg => ($highlight ? [1, 1, 1] : [.7, .7, .7]), |
|
|
1287 | can_hover => 1, |
|
|
1288 | can_events => 1, |
|
|
1289 | fontsize => 0.8) |
704 | for 0 .. $#$m; |
1290 | for 0 .. $#$m; |
705 | } |
1291 | } |
|
|
1292 | } else { |
|
|
1293 | $ok or $label->set_text ("error while contacting metaserver"); |
706 | } |
1294 | } |
707 | }); |
1295 | }; |
|
|
1296 | |
708 | } |
1297 | } |
709 | |
1298 | |
710 | sub metaserver_dialog { |
1299 | sub metaserver_dialog { |
711 | my $dialog = new CFClient::UI::FancyFrame |
1300 | my $vbox = new CFPlus::UI::VBox; |
|
|
1301 | my $table = new CFPlus::UI::Table; |
|
|
1302 | $vbox->add (new CFPlus::UI::ScrolledWindow expand => 1, child => $table); |
|
|
1303 | |
|
|
1304 | my $dialog = new CFPlus::UI::Toplevel |
712 | title => "Server List", |
1305 | title => "Server List", |
713 | name => 'metaserver_dialog', |
1306 | name => 'metaserver_dialog', |
714 | x => 'center', |
1307 | x => 'center', |
715 | y => 'center', |
1308 | y => 'center', |
716 | child => (my $vbox = new CFClient::UI::VBox), |
1309 | z => 3, |
|
|
1310 | force_w => $::WIDTH * 0.9, |
|
|
1311 | force_h => $::HEIGHT * 0.7, |
|
|
1312 | child => $vbox, |
|
|
1313 | has_close_button => 1, |
|
|
1314 | table => $table, |
717 | on_visibility_change => sub { |
1315 | on_visibility_change => sub { |
718 | update_metaserver if $_[1]; |
1316 | update_metaserver ($_[0]) if $_[1]; |
|
|
1317 | 0 |
719 | }, |
1318 | }, |
720 | ; |
1319 | ; |
721 | |
1320 | |
722 | $vbox->add ($dialog->{table} = new CFClient::UI::Table); |
|
|
723 | |
|
|
724 | $dialog |
1321 | $dialog |
725 | } |
1322 | } |
726 | |
1323 | |
727 | sub server_setup { |
1324 | sub server_setup { |
728 | my $vbox = new CFClient::UI::VBox; |
1325 | my $vbox = new CFPlus::UI::VBox; |
729 | |
1326 | |
|
|
1327 | $vbox->add (new CFPlus::UI::FancyFrame |
|
|
1328 | label => "Connection Settings", |
730 | $vbox->add (my $table = new CFClient::UI::Table expand => 1, col_expand => [0, 1]); |
1329 | child => (my $table = new CFPlus::UI::Table expand => 1, col_expand => [0, 1]), |
|
|
1330 | ); |
731 | $table->add (0, 2, new CFClient::UI::Label valign => 0, align => 1, text => "Host:Port"); |
1331 | $table->add_at (0, 2, new CFPlus::UI::Label valign => 0, align => 1, text => "Host:Port"); |
732 | |
1332 | |
733 | { |
1333 | { |
734 | $table->add (1, 2, my $vbox = new CFClient::UI::VBox); |
1334 | $table->add_at (1, 2, my $vbox = new CFPlus::UI::VBox); |
735 | |
1335 | |
736 | $vbox->add ( |
1336 | $vbox->add ( |
737 | $HOST_ENTRY = new CFClient::UI::Entry |
1337 | $HOST_ENTRY = new CFPlus::UI::Entry |
738 | expand => 1, |
1338 | expand => 1, |
739 | text => $CFG->{host}, |
1339 | text => $CFG->{profile}{default}{host}, |
740 | tooltip => "The hostname or ip address of the Crossfire(+) server to connect to", |
1340 | tooltip => "The hostname or ip address of the Crossfire(+) server to connect to", |
741 | on_changed => sub { |
1341 | on_changed => sub { |
742 | my ($self, $value) = @_; |
1342 | my ($self, $value) = @_; |
743 | $CFG->{host} = $value; |
1343 | $CFG->{profile}{default}{host} = $value; |
|
|
1344 | 1 |
744 | } |
1345 | } |
745 | ); |
1346 | ); |
746 | |
1347 | |
747 | $METASERVER = metaserver_dialog; |
|
|
748 | |
|
|
749 | $vbox->add (new CFClient::UI::Button |
1348 | $vbox->add (new CFPlus::UI::Button |
750 | expand => 1, |
1349 | expand => 1, |
751 | text => "Server List", |
1350 | text => "Server List", |
752 | other => $METASERVER, |
1351 | other => $METASERVER, |
753 | tooltip => "Show a list of available crossfire servers", |
1352 | tooltip => "Show a list of available crossfire servers", |
754 | on_activate => sub { $METASERVER->toggle_visibility }, |
1353 | on_activate => sub { $METASERVER->toggle_visibility; 0 }, |
|
|
1354 | on_visibility_change => sub { $METASERVER->hide unless $_[1]; 1 }, |
755 | ); |
1355 | ); |
756 | } |
1356 | } |
757 | |
1357 | |
758 | $table->add (0, 4, new CFClient::UI::Label valign => 0, align => 1, text => "Username"); |
1358 | $table->add_at (0, 4, new CFPlus::UI::Label valign => 0, align => 1, text => "Username"); |
759 | $table->add (1, 4, new CFClient::UI::Entry |
1359 | $table->add_at (1, 4, new CFPlus::UI::Entry |
760 | text => $CFG->{user}, |
1360 | text => $CFG->{profile}{default}{user}, |
761 | tooltip => "The name of your character on the server", |
1361 | tooltip => "The name of your character on the server", |
762 | on_changed => sub { |
1362 | on_changed => sub { my ($self, $value) = @_; $CFG->{profile}{default}{user} = $value; 1 } |
763 | my ($self, $value) = @_; |
|
|
764 | $CFG->{user} = $value; |
|
|
765 | } |
|
|
766 | ); |
1363 | ); |
767 | |
1364 | |
768 | $table->add (0, 5, new CFClient::UI::Label valign => 0, align => 1, text => "Password"); |
1365 | $table->add_at (0, 5, new CFPlus::UI::Label valign => 0, align => 1, text => "Password"); |
769 | $table->add (1, 5, new CFClient::UI::Entry |
1366 | $table->add_at (1, 5, new CFPlus::UI::Entry |
770 | text => $CFG->{password}, |
1367 | text => $CFG->{profile}{default}{password}, |
771 | hidden => 1, |
1368 | hidden => 1, |
772 | tooltip => "The password for your character", |
1369 | tooltip => "The password for your character", |
773 | on_changed => sub { |
1370 | on_changed => sub { my ($self, $value) = @_; $CFG->{profile}{default}{password} = $value; 1 } |
774 | my ($self, $value) = @_; |
|
|
775 | $CFG->{password} = $value; |
|
|
776 | } |
|
|
777 | ); |
1371 | ); |
778 | |
1372 | |
779 | $table->add (0, 7, new CFClient::UI::Label valign => 0, align => 1, text => "Map Size"); |
1373 | $table->add_at (0, 7, new CFPlus::UI::Label valign => 0, align => 1, text => "Map Size"); |
780 | $table->add (1, 7, new CFClient::UI::Slider |
1374 | $table->add_at (1, 7, new CFPlus::UI::Slider |
781 | force_w => 100, |
1375 | force_w => 100, |
782 | range => [$CFG->{mapsize}, 10, 100, 0, 1], |
1376 | range => [$CFG->{mapsize}, 10, 100, 0, 1], |
783 | tooltip => "This is the size of the portion of the map update the server sends you. " |
1377 | tooltip => "This is the size of the portion of the map update the server sends you. " |
784 | . "If you set this to a high value you will be able to see further, " |
1378 | . "If you set this to a high value you will be able to see further, " |
785 | . "but you also increase bandwidth requirements and latency. " |
1379 | . "but you also increase bandwidth requirements and latency. " |
786 | . "This option is only used once at log-in.", |
1380 | . "This option is only used once at log-in.", |
787 | on_changed => sub { |
1381 | on_changed => sub { my ($self, $value) = @_; $CFG->{mapsize} = $self->{range}[0] = $value = int $value; 1 }, |
788 | my ($self, $value) = @_; |
|
|
789 | |
|
|
790 | $CFG->{mapsize} = $self->{range}[0] = $value = int $value; |
|
|
791 | }, |
|
|
792 | ); |
1382 | ); |
793 | |
1383 | |
794 | $table->add (0, 8, new CFClient::UI::Label valign => 0, align => 1, text => "Face Prefetch"); |
1384 | $table->add_at (0, 8, new CFPlus::UI::Label valign => 0, align => 1, text => "Output-Rate"); |
795 | $table->add (1, 8, new CFClient::UI::CheckBox |
1385 | $table->add_at (1, 8, new CFPlus::UI::Entry |
796 | state => $CFG->{face_prefetch}, |
1386 | text => $CFG->{output_rate}, |
797 | tooltip => "<b>Background Image Prefetch</b>\n\n" |
1387 | tooltip => "The maximum bandwidth in bytes per second that the server should not exceed " |
798 | . "If enabled, the client automatically pre-fetches images from the server. " |
1388 | . "when sending data. When 0 or unset, the server " |
799 | . "This might increase or create lag, but increases the chances " |
1389 | . "default will be used, which is usually around 100kb/s. Most servers will " |
800 | . "of faces being ready for display when you encounter them. " |
1390 | . "dynamically find an optimal rate, so adjust this only when necessary.", |
801 | . "It also uses up server bandwidth on every connect, " |
|
|
802 | . "so only set it if you really need to prefetch images. " |
|
|
803 | . "This option can be set and unset any time.", |
|
|
804 | on_changed => sub { $CFG->{face_prefetch} = $_[1] }, |
1391 | on_changed => sub { $CFG->{output_rate} = $_[1]; 1 }, |
805 | ); |
1392 | ); |
806 | |
1393 | |
807 | $table->add (0, 9, new CFClient::UI::Label valign => 0, align => 1, text => "Output-Count"); |
1394 | $table->add_at (0, 9, new CFPlus::UI::Label valign => 0, align => 1, text => "Output-Count"); |
808 | $table->add (1, 9, new CFClient::UI::Entry |
1395 | $table->add_at (1, 9, new CFPlus::UI::Entry |
809 | text => $CFG->{output_count}, |
1396 | text => $CFG->{output_count}, |
810 | tooltip => "Should be set to 1 unless you know what you are doing. This option is only used once at log-in.", |
1397 | tooltip => "Should be set to 1 unless you know what you are doing. This option is only used once at log-in.", |
811 | on_changed => sub { $CFG->{output_count} = $_[1] }, |
1398 | on_changed => sub { $CFG->{output_count} = $_[1]; 1 }, |
812 | ); |
1399 | ); |
813 | |
1400 | |
814 | $table->add (0, 10, new CFClient::UI::Label valign => 0, align => 1, text => "Output-Sync"); |
1401 | $table->add_at (0, 10, new CFPlus::UI::Label valign => 0, align => 1, text => "Output-Sync"); |
815 | $table->add (1, 10, new CFClient::UI::Entry |
1402 | $table->add_at (1, 10, new CFPlus::UI::Entry |
816 | text => $CFG->{output_sync}, |
1403 | text => $CFG->{output_sync}, |
817 | tooltip => "Should be set to 1 unless you know what you are doing. This option is only used once at log-in.", |
1404 | tooltip => "Should be set to 1 unless you know what you are doing. This option is only used once at log-in.", |
818 | on_changed => sub { $CFG->{output_sync} = $_[1] }, |
1405 | on_changed => sub { $CFG->{output_sync} = $_[1]; 1 }, |
819 | ); |
1406 | ); |
820 | |
1407 | |
821 | $table->add (1, 11, $LOGIN_BUTTON = new CFClient::UI::Button |
1408 | $table->add_at (1, 11, $LOGIN_BUTTON = new CFPlus::UI::Button |
822 | expand => 1, |
1409 | expand => 1, |
823 | align => 0, |
1410 | align => 0, |
824 | text => "Login", |
1411 | text => "Login", |
825 | on_activate => sub { |
1412 | on_activate => sub { |
826 | $CONN ? stop_game |
1413 | $CONN ? stop_game |
827 | : start_game; |
1414 | : start_game; |
|
|
1415 | 1 |
828 | }, |
1416 | }, |
829 | ); |
1417 | ); |
830 | |
1418 | |
|
|
1419 | $vbox->add (new CFPlus::UI::FancyFrame |
|
|
1420 | label => "Server Info", |
|
|
1421 | child => ($SERVER_INFO = new CFPlus::UI::Label ellipsise => 0), |
|
|
1422 | ); |
|
|
1423 | |
|
|
1424 | $vbox |
|
|
1425 | } |
|
|
1426 | |
|
|
1427 | sub client_setup { |
|
|
1428 | my $table = new CFPlus::UI::Table expand => 1, col_expand => [0, 1]; |
|
|
1429 | |
|
|
1430 | my $row = 0; |
|
|
1431 | |
831 | $table->add (0, 12, new CFClient::UI::Label valign => 0, align => 1, text => "Chat Command"); |
1432 | $table->add_at (0, $row, new CFPlus::UI::Label valign => 0, align => 1, text => "Tip of the day"); |
832 | $table->add (1, 12, my $saycmd = new CFClient::UI::Entry |
1433 | $table->add_at (1, $row++, new CFPlus::UI::CheckBox |
833 | text => $CFG->{say_command}, |
1434 | state => $CFG->{show_tips}, |
834 | 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. " |
1435 | tooltip => "Show the <b>Tip of the day</b> window at startup?", |
835 | . "Usually you want to enter something like 'say' or 'shout' or 'gsay' here. " |
|
|
836 | . "But you could also set it to <b>tell <i>playername</i></b> to only chat with that user.", |
|
|
837 | on_changed => sub { |
1436 | on_changed => sub { |
838 | my ($self, $value) = @_; |
1437 | my ($self, $value) = @_; |
839 | $CFG->{say_command} = $value; |
1438 | $CFG->{show_tips} = $value; |
840 | } |
|
|
841 | ); |
|
|
842 | |
|
|
843 | $vbox |
|
|
844 | } |
|
|
845 | |
|
|
846 | sub message_window { |
|
|
847 | my $window = new CFClient::UI::FancyFrame |
|
|
848 | name => "message_window", |
|
|
849 | title => "Messages", |
|
|
850 | border_bg => [1, 1, 1, 1], |
|
|
851 | bg => [0, 0, 0, 0.75], |
|
|
852 | x => "max", |
|
|
853 | y => 0, |
|
|
854 | force_w => $::WIDTH / 3, |
|
|
855 | force_h => $::HEIGHT / 5, |
|
|
856 | child => (my $vbox = new CFClient::UI::VBox); |
|
|
857 | |
|
|
858 | $vbox->add ($LOGVIEW); |
|
|
859 | |
|
|
860 | $vbox->add (my $input = new CFClient::UI::Entry |
|
|
861 | tooltip => "<b>Chat Box</b>. If you enter a text and press return/enter here, the current <i>communication command</i> " |
|
|
862 | . "from the client setup will be prepended (e.g. <b>shout</b>, <b>chat</b>...). " |
|
|
863 | . "If you prepend a slash (/), you will submit a command instead (similar to IRC). " |
|
|
864 | . "A better way to submit commands (and the occasional chat command) is often the map command completer.", |
|
|
865 | on_focus_in => sub { |
|
|
866 | my ($input, $prev_focus) = @_; |
|
|
867 | |
|
|
868 | delete $input->{refocus_map}; |
|
|
869 | |
|
|
870 | if ($prev_focus == $MAPWIDGET && $input->{auto_activated}) { |
|
|
871 | $input->{refocus_map} = 1; |
|
|
872 | } |
1439 | 0 |
873 | delete $input->{auto_activated}; |
1440 | } |
|
|
1441 | ); |
|
|
1442 | |
|
|
1443 | $table->add_at (0, $row, new CFPlus::UI::Label valign => 0, align => 1, text => "Messages Window Size"); |
|
|
1444 | $table->add_at (1, $row++, my $saycmd = new CFPlus::UI::Entry |
|
|
1445 | text => $CFG->{logview_max_par}, |
|
|
1446 | tooltip => "This is maximum number of messages remembered in the <b>Messages</b> window. If the server " |
|
|
1447 | . "sends more messages than this number, older messages get removed to save memory and " |
|
|
1448 | . "computing time. A value of <b>0</b> disables this feature, but that is not recommended.", |
|
|
1449 | on_changed => sub { |
|
|
1450 | my ($self, $value) = @_; |
|
|
1451 | $MESSAGE_WINDOW->set_max_para ($CFG->{logview_max_par} = $value*1); |
|
|
1452 | 0 |
874 | }, |
1453 | }, |
|
|
1454 | ); |
|
|
1455 | |
|
|
1456 | $table |
|
|
1457 | } |
|
|
1458 | |
|
|
1459 | sub autopickup_setup { |
|
|
1460 | my $r = new CFPlus::UI::ScrolledWindow ( |
|
|
1461 | expand => 1, |
|
|
1462 | scroll_y => 1 |
|
|
1463 | ); |
|
|
1464 | $r->add (my $table = new CFPlus::UI::Table |
|
|
1465 | row_expand => [0], |
|
|
1466 | col_expand => [0, 1, 0, 1], |
|
|
1467 | ); |
|
|
1468 | |
|
|
1469 | for ( |
|
|
1470 | ["General", 0, 0, |
|
|
1471 | ["Enable autopickup" => PICKUP_NEWMODE, \$PICKUP_ENABLE], |
|
|
1472 | ["Inhibit autopickup" => PICKUP_INHIBIT], |
|
|
1473 | ["Stop before pickup" => PICKUP_STOP], |
|
|
1474 | ["Debug autopickup" => PICKUP_DEBUG], |
|
|
1475 | ], |
|
|
1476 | ["Weapons", 0, 6, |
|
|
1477 | ["All weapons" => PICKUP_ALLWEAPON], |
|
|
1478 | ["Missile weapons" => PICKUP_MISSILEWEAPON], |
|
|
1479 | ["Bows" => PICKUP_BOW], |
|
|
1480 | ["Arrows" => PICKUP_ARROW], |
|
|
1481 | ], |
|
|
1482 | ["Armour", 0, 12, |
|
|
1483 | ["Helmets" => PICKUP_HELMET], |
|
|
1484 | ["Shields" => PICKUP_SHIELD], |
|
|
1485 | ["Body Armour" => PICKUP_ARMOUR], |
|
|
1486 | ["Boots" => PICKUP_BOOTS], |
|
|
1487 | ["Gloves" => PICKUP_GLOVES], |
|
|
1488 | ["Cloaks" => PICKUP_CLOAK], |
|
|
1489 | ], |
|
|
1490 | |
|
|
1491 | ["Readables", 2, 0, |
|
|
1492 | ["Spellbooks" => PICKUP_SPELLBOOK], |
|
|
1493 | ["Skillscrolls" => PICKUP_SKILLSCROLL], |
|
|
1494 | ["Normal Books/Scrolls" => PICKUP_READABLES], |
|
|
1495 | ], |
|
|
1496 | ["Misc", 2, 5, |
|
|
1497 | ["Food" => PICKUP_FOOD], |
|
|
1498 | ["Drinks" => PICKUP_DRINK], |
|
|
1499 | ["Valuables (Money, Gems)" => PICKUP_VALUABLES], |
|
|
1500 | ["Keys" => PICKUP_KEY], |
|
|
1501 | ["Magical Items" => PICKUP_MAGICAL], |
|
|
1502 | ["Potions" => PICKUP_POTION], |
|
|
1503 | ["Magic Devices" => PICKUP_MAGIC_DEVICE], |
|
|
1504 | ["Ignore cursed" => PICKUP_NOT_CURSED], |
|
|
1505 | ["Jewelery" => PICKUP_JEWELS], |
|
|
1506 | ["Flesh" => PICKUP_FLESH], |
|
|
1507 | ], |
|
|
1508 | ["Weight/Value ratio", 2, 17] |
|
|
1509 | ) |
|
|
1510 | { |
|
|
1511 | my ($title, $x, $y, @bits) = @$_; |
|
|
1512 | $table->add_at ($x, $y, new CFPlus::UI::Label text => $title, align => 1, fg => [1, 1, 0]); |
|
|
1513 | |
|
|
1514 | for (@bits) { |
|
|
1515 | ++$y; |
|
|
1516 | |
|
|
1517 | my $mask = $_->[1]; |
|
|
1518 | $table->add_at ($x , $y, new CFPlus::UI::Label text => $_->[0], align => 1, expand => 1); |
|
|
1519 | $table->add_at ($x+1, $y, my $checkbox = new CFPlus::UI::CheckBox |
|
|
1520 | state => $::CFG->{pickup} & $mask, |
|
|
1521 | on_changed => sub { |
|
|
1522 | my ($box, $value) = @_; |
|
|
1523 | |
|
|
1524 | if ($value) { |
|
|
1525 | $::CFG->{pickup} |= $mask; |
|
|
1526 | } else { |
|
|
1527 | $::CFG->{pickup} &= ~$mask; |
|
|
1528 | } |
|
|
1529 | |
|
|
1530 | $::CONN->send_command ("pickup $::CFG->{pickup}") |
|
|
1531 | if defined $::CONN; |
|
|
1532 | |
|
|
1533 | 0 |
|
|
1534 | }); |
|
|
1535 | |
|
|
1536 | ${$_->[2]} = $checkbox if $_->[2]; |
|
|
1537 | } |
|
|
1538 | } |
|
|
1539 | |
|
|
1540 | $table->add_at (2, 18, new CFPlus::UI::ValSlider |
|
|
1541 | range => [$::CFG->{pickup} & 0xF, 0, 16, 1, 1], |
|
|
1542 | template => ">= 99", |
|
|
1543 | to_value => sub { ">= " . 5 * $_[0] }, |
|
|
1544 | on_changed => sub { |
|
|
1545 | my ($slider, $value) = @_; |
|
|
1546 | |
|
|
1547 | $::CFG->{pickup} &= ~0xF; |
|
|
1548 | $::CFG->{pickup} |= int $value |
|
|
1549 | if $value; |
|
|
1550 | 1; |
|
|
1551 | }); |
|
|
1552 | |
|
|
1553 | $table->add_at (3, 18, new CFPlus::UI::Button |
|
|
1554 | text => "set", |
875 | on_activate => sub { |
1555 | on_activate => sub { |
876 | my ($input, $text) = @_; |
1556 | $::CONN->send_command ("pickup $::CFG->{pickup}") |
877 | $input->set_text (''); |
1557 | if defined $::CONN; |
878 | |
|
|
879 | if ($text =~ /^\/(.*)/) { |
|
|
880 | $::CONN->user_send ($1); |
|
|
881 | } else { |
|
|
882 | my $say_cmd = $::CFG->{say_command} || 'say'; |
|
|
883 | $::CONN->user_send ("$say_cmd $text"); |
|
|
884 | } |
1558 | 0 |
885 | if ($input->{refocus_map}) { |
1559 | }); |
886 | delete $input->{refocus_map}; |
1560 | |
887 | $MAPWIDGET->focus_in |
1561 | $r |
|
|
1562 | } |
|
|
1563 | |
|
|
1564 | my %SORT_ORDER = ( |
|
|
1565 | type => undef, |
|
|
1566 | mtime => sub { |
|
|
1567 | my $NOW = time; |
|
|
1568 | sort { |
|
|
1569 | my $atime = $a->{mtime} - $NOW; $atime = $atime < 5 * 60 ? int $atime / 60 : 6; |
|
|
1570 | my $btime = $b->{mtime} - $NOW; $btime = $btime < 5 * 60 ? int $btime / 60 : 6; |
|
|
1571 | |
|
|
1572 | ($a->{flags} & F_LOCKED) <=> ($b->{flags} & F_LOCKED) |
|
|
1573 | or $btime <=> $atime |
|
|
1574 | or $a->{type} <=> $b->{type} |
|
|
1575 | } @_ |
888 | } |
1576 | }, |
|
|
1577 | weight => sub { sort { |
|
|
1578 | $a->{weight} * ($a->{nrof} || 1) <=> $b->{weight} * ($b->{nrof} || 1) |
|
|
1579 | or $a->{type} <=> $b->{type} |
|
|
1580 | } @_ }, |
|
|
1581 | ); |
|
|
1582 | |
|
|
1583 | sub inventory_widget { |
|
|
1584 | my $hb = new CFPlus::UI::HBox homogeneous => 1; |
|
|
1585 | |
|
|
1586 | $hb->add (my $vb1 = new CFPlus::UI::VBox); |
|
|
1587 | $vb1->add (new CFPlus::UI::Label align => 0, text => "Player"); |
|
|
1588 | |
|
|
1589 | $vb1->add (my $hb1 = new CFPlus::UI::HBox); |
|
|
1590 | |
|
|
1591 | use sort 'stable'; |
|
|
1592 | |
|
|
1593 | $hb1->add (new CFPlus::UI::Selector |
|
|
1594 | value => $::CFG->{inv_sort}, |
|
|
1595 | options => [ |
|
|
1596 | [type => "Type/Name"], |
|
|
1597 | [mtime => "Recent/Normal/Locked"], |
|
|
1598 | [weight => "Weight/Type"], |
|
|
1599 | ], |
|
|
1600 | on_changed => sub { |
|
|
1601 | $::CFG->{inv_sort} = $_[1]; |
|
|
1602 | $INV->set_sort_order ($SORT_ORDER{$_[1]}); |
889 | }, |
1603 | }, |
890 | on_escape => sub { |
1604 | ); |
891 | $MAPWIDGET->focus_in |
1605 | $hb1->add (new CFPlus::UI::Label text => "Weight: ", align => 1, expand => 1); |
|
|
1606 | #TODO# update to weigh/maxweight |
|
|
1607 | $hb1->add ($STATWIDS->{i_weight} = new CFPlus::UI::Label align => -1); |
|
|
1608 | |
|
|
1609 | $vb1->add (my $sw1 = new CFPlus::UI::ScrolledWindow expand => 1, scroll_y => 1); |
|
|
1610 | $sw1->add ($INV = new CFPlus::UI::Inventory); |
|
|
1611 | $INV->set_sort_order ($SORT_ORDER{$::CFG->{inv_sort}}); |
|
|
1612 | |
|
|
1613 | $hb->add (my $vb2 = new CFPlus::UI::VBox); |
|
|
1614 | |
|
|
1615 | $vb2->add ($INV_RIGHT_HB = new CFPlus::UI::HBox); |
|
|
1616 | |
|
|
1617 | $vb2->add (my $sw2 = new CFPlus::UI::ScrolledWindow expand => 1, scroll_y => 1); |
|
|
1618 | $sw2->add ($INVR = new CFPlus::UI::Inventory); |
|
|
1619 | |
|
|
1620 | # XXX: Call after $INVR = ... because set_opencont sets the items |
|
|
1621 | CFPlus::Protocol::set_opencont ($::CONN, 0, "Floor"); |
|
|
1622 | |
|
|
1623 | $hb |
|
|
1624 | } |
|
|
1625 | |
|
|
1626 | sub media_window { |
|
|
1627 | my $vb = new CFPlus::UI::VBox; |
|
|
1628 | |
|
|
1629 | $vb->add (new CFPlus::UI::FancyFrame |
|
|
1630 | label => "Currently playing music", |
|
|
1631 | child => new CFPlus::UI::ScrolledWindow scroll_x => 1, scroll_y => 0, |
|
|
1632 | child => ($MUSIC_PLAYING_WIDGET = new CFPlus::UI::Label ellipsise => 0, fontsize => 0.8), |
|
|
1633 | ); |
|
|
1634 | |
|
|
1635 | $vb->add (new CFPlus::UI::FancyFrame |
|
|
1636 | label => "Other media used in this session", |
|
|
1637 | expand => 1, |
|
|
1638 | child => ($LICENSE_WIDGET = new CFPlus::UI::TextScroller |
|
|
1639 | expand => 1, fontsize => 0.8, padding_x => 4, padding_y => 4), |
|
|
1640 | ); |
|
|
1641 | |
|
|
1642 | $vb |
|
|
1643 | } |
|
|
1644 | |
|
|
1645 | sub add_license { |
|
|
1646 | my ($meta) = @_; |
|
|
1647 | |
|
|
1648 | $meta = $meta->{meta} |
|
|
1649 | or return; |
|
|
1650 | |
|
|
1651 | $meta->{license} || $meta->{author} || $meta->{source} |
|
|
1652 | or return; |
|
|
1653 | |
|
|
1654 | $LICENSE_WIDGET->add_paragraph ({ |
|
|
1655 | fg => [1, 1, 1, 1], |
|
|
1656 | markup => "<small>" |
|
|
1657 | . "<b>Name:</b> " . (CFPlus::asxml $meta->{name}) . "\n" |
|
|
1658 | . "<b>Author:</b> " . (CFPlus::asxml $meta->{author}) . "\n" |
|
|
1659 | . "<b>Source:</b> " . (CFPlus::asxml $meta->{source}) . "\n" |
|
|
1660 | . "<b>License:</b> " . (CFPlus::asxml $meta->{license}) . "\n" |
|
|
1661 | . "</small>", |
|
|
1662 | }); |
|
|
1663 | $LICENSE_WIDGET->scroll_to_bottom; |
|
|
1664 | } |
|
|
1665 | |
|
|
1666 | sub toggle_player_page { |
|
|
1667 | my ($widget) = @_; |
|
|
1668 | |
|
|
1669 | if ($PL_WINDOW->{visible} && $PL_NOTEBOOK->get_current_page == $widget) { |
|
|
1670 | $PL_WINDOW->hide; |
|
|
1671 | } else { |
|
|
1672 | $PL_NOTEBOOK->set_current_page ($widget); |
|
|
1673 | $PL_WINDOW->show; |
|
|
1674 | } |
|
|
1675 | } |
|
|
1676 | |
|
|
1677 | sub player_window { |
|
|
1678 | my $plwin = $PL_WINDOW = new CFPlus::UI::Toplevel |
|
|
1679 | x => "center", |
|
|
1680 | y => "center", |
|
|
1681 | force_w => $WIDTH * 9/10, |
|
|
1682 | force_h => $HEIGHT * 9/10, |
|
|
1683 | title => "Player", |
|
|
1684 | name => "playerbook", |
|
|
1685 | has_close_button => 1 |
|
|
1686 | ; |
|
|
1687 | |
|
|
1688 | my $ntb = |
|
|
1689 | $PL_NOTEBOOK = |
|
|
1690 | new CFPlus::UI::Notebook expand => 1; |
|
|
1691 | |
|
|
1692 | $ntb->add_tab ( |
|
|
1693 | "Statistics (F2)" => $STATS_PAGE = stats_window, |
|
|
1694 | "Shows statistics, where all your Stats and Resistances are shown." |
|
|
1695 | ); |
|
|
1696 | $ntb->add_tab ( |
|
|
1697 | "Skills (F3)" => $SKILL_PAGE = skill_window, |
|
|
1698 | "Shows all your Skills." |
|
|
1699 | ); |
|
|
1700 | |
|
|
1701 | my $spellsw = $SPELL_PAGE = new CFPlus::UI::ScrolledWindow (expand => 1, scroll_y => 1); |
|
|
1702 | $spellsw->add ($SPELL_LIST = new CFPlus::UI::SpellList); |
|
|
1703 | $ntb->add_tab ( |
|
|
1704 | "Spellbook (F4)" => $spellsw, |
|
|
1705 | "Displays all spells you have and lets you edit keyboard shortcuts for them." |
|
|
1706 | ); |
|
|
1707 | $ntb->add_tab ( |
|
|
1708 | "Inventory (F5)" => $INVENTORY_PAGE = inventory_widget, |
|
|
1709 | "Toggles the inventory window, where you can manage your loot (or treasures :). " |
|
|
1710 | . "You can also hit the <b>Tab</b>-key to show/hide the Inventory." |
|
|
1711 | ); |
|
|
1712 | $ntb->add_tab (Pickup => autopickup_setup, |
|
|
1713 | "Configure autopickup settings, i.e. which items you will pick up automatically when walking (or running) over them."); |
|
|
1714 | |
|
|
1715 | $ntb->add_tab (Media => media_window, |
|
|
1716 | "License, Author and Source info for media sent by the server."); |
|
|
1717 | |
|
|
1718 | $ntb->set_current_page ($INVENTORY_PAGE); |
|
|
1719 | |
|
|
1720 | $plwin->add ($ntb); |
|
|
1721 | $plwin |
|
|
1722 | } |
|
|
1723 | |
|
|
1724 | sub keyboard_setup { |
|
|
1725 | CFPlus::Macro::keyboard_setup |
|
|
1726 | } |
|
|
1727 | |
|
|
1728 | sub help_window { |
|
|
1729 | my $win = new CFPlus::UI::Toplevel |
|
|
1730 | x => 'center', |
|
|
1731 | y => 'center', |
|
|
1732 | z => 4, |
|
|
1733 | name => 'doc_browser', |
|
|
1734 | force_w => int $WIDTH * 7/8, |
|
|
1735 | force_h => int $HEIGHT * 7/8, |
|
|
1736 | title => "Help Browser", |
|
|
1737 | has_close_button => 1; |
|
|
1738 | |
|
|
1739 | $win->add (my $vbox = new CFPlus::UI::VBox); |
|
|
1740 | |
|
|
1741 | $vbox->add (new CFPlus::UI::FancyFrame |
|
|
1742 | label => "Navigation", |
|
|
1743 | child => (my $buttons = new CFPlus::UI::HBox), |
|
|
1744 | ); |
|
|
1745 | $vbox->add (my $viewer = new CFPlus::UI::TextScroller |
|
|
1746 | expand => 1, fontsize => 0.8, padding_x => 4, padding_y => 4); |
|
|
1747 | |
|
|
1748 | my @history; |
|
|
1749 | my @future; |
|
|
1750 | my $curnode; |
|
|
1751 | |
|
|
1752 | my $load_node; $load_node = sub { |
|
|
1753 | my ($node, $para) = @_; |
|
|
1754 | |
|
|
1755 | $buttons->clear; |
|
|
1756 | |
|
|
1757 | $buttons->add (new CFPlus::UI::Button |
|
|
1758 | text => "⇤", |
|
|
1759 | tooltip => "back to the starting page", |
|
|
1760 | on_activate => sub { |
|
|
1761 | unshift @future, [$curnode, $viewer->current_paragraph] if $curnode; |
|
|
1762 | unshift @future, @history; |
|
|
1763 | @history = (); |
|
|
1764 | $load_node->(@{shift @future}); |
|
|
1765 | }, |
|
|
1766 | ); |
|
|
1767 | |
|
|
1768 | if (@history) { |
|
|
1769 | $buttons->add (new CFPlus::UI::Button |
|
|
1770 | text => "⋘", |
|
|
1771 | tooltip => "back to <i>" . (CFPlus::asxml CFPlus::Pod::full_path $history[-1][0]) . "</i>", |
|
|
1772 | on_activate => sub { |
|
|
1773 | unshift @future, [$curnode, $viewer->current_paragraph] if $curnode; |
|
|
1774 | $load_node->(@{pop @history}); |
|
|
1775 | }, |
|
|
1776 | ); |
892 | }, |
1777 | } |
893 | ); |
|
|
894 | |
1778 | |
895 | $CONSOLE = { |
1779 | if (@future) { |
896 | window => $window, |
1780 | $buttons->add (new CFPlus::UI::Button |
897 | input => $input, |
1781 | text => "⋙", |
|
|
1782 | tooltip => "forward to <i>" . (CFPlus::asxml CFPlus::Pod::full_path $future[0][0]) . "</i>", |
|
|
1783 | on_activate => sub { |
|
|
1784 | push @history, [$curnode, $viewer->current_paragraph]; |
|
|
1785 | $load_node->(@{shift @future}); |
|
|
1786 | }, |
|
|
1787 | ); |
|
|
1788 | } |
|
|
1789 | |
|
|
1790 | $buttons->add (new CFPlus::UI::Label text => " "); |
|
|
1791 | |
|
|
1792 | my @path = CFPlus::Pod::full_path_of $node; |
|
|
1793 | pop @path; # drop current node |
|
|
1794 | |
|
|
1795 | for my $node (@path) { |
|
|
1796 | $buttons->add (new CFPlus::UI::Button |
|
|
1797 | text => $node->{kw}[0], |
|
|
1798 | tooltip => "go to <i>" . (CFPlus::asxml CFPlus::Pod::full_path $node) . "</i>", |
|
|
1799 | on_activate => sub { |
|
|
1800 | push @history, [$curnode, $viewer->current_paragraph] if $curnode; @future = (); |
|
|
1801 | $load_node->($node); |
|
|
1802 | }, |
|
|
1803 | ); |
|
|
1804 | $buttons->add (new CFPlus::UI::Label text => "/"); |
|
|
1805 | } |
|
|
1806 | |
|
|
1807 | $buttons->add (new CFPlus::UI::Label text => $node->{kw}[0], padding_x => 4, padding_y => 4); |
|
|
1808 | |
|
|
1809 | $curnode = $node; |
|
|
1810 | |
|
|
1811 | $viewer->clear; |
|
|
1812 | $viewer->add_paragraph (CFPlus::Pod::as_paragraphs CFPlus::Pod::section_of $curnode); |
|
|
1813 | $viewer->scroll_to ($para); |
898 | }; |
1814 | }; |
899 | |
1815 | |
|
|
1816 | $load_node->(CFPlus::Pod::find pod => "mainpage"); |
|
|
1817 | |
|
|
1818 | $CFPlus::Pod::goto_document = sub { |
|
|
1819 | my (@path) = @_; |
|
|
1820 | |
|
|
1821 | push @history, [$curnode, $viewer->current_paragraph] if $curnode; @future = (); |
|
|
1822 | |
|
|
1823 | $load_node->((CFPlus::Pod::find @path)[0]); |
|
|
1824 | $win->show; |
|
|
1825 | }; |
|
|
1826 | |
900 | $window |
1827 | $win |
|
|
1828 | } |
|
|
1829 | |
|
|
1830 | sub open_string_query { |
|
|
1831 | my ($title, $cb, $txt, $tooltip) = @_; |
|
|
1832 | my $dialog = new CFPlus::UI::Toplevel |
|
|
1833 | x => "center", |
|
|
1834 | y => "center", |
|
|
1835 | z => 50, |
|
|
1836 | force_w => $WIDTH * 4/5, |
|
|
1837 | title => $title; |
|
|
1838 | |
|
|
1839 | $dialog->add ( |
|
|
1840 | my $e = new CFPlus::UI::Entry |
|
|
1841 | on_activate => sub { $cb->(@_); $dialog->hide; 0 }, |
|
|
1842 | on_key_down => sub { $_[1]->{sym} == 27 and $dialog->hide; 0 }, |
|
|
1843 | tooltip => $tooltip |
|
|
1844 | ); |
|
|
1845 | |
|
|
1846 | $e->grab_focus; |
|
|
1847 | $e->set_text ($txt) if $txt; |
|
|
1848 | $dialog->show; |
901 | } |
1849 | } |
902 | |
1850 | |
903 | sub open_quit_dialog { |
1851 | sub open_quit_dialog { |
904 | unless ($QUIT_DIALOG) { |
1852 | unless ($QUIT_DIALOG) { |
905 | $QUIT_DIALOG = new CFClient::UI::FancyFrame |
1853 | $QUIT_DIALOG = new CFPlus::UI::Toplevel |
906 | x => "center", |
1854 | x => "center", |
907 | y => "center", |
1855 | y => "center", |
|
|
1856 | z => 50, |
908 | title => "Really Quit?", |
1857 | title => "Really Quit?", |
|
|
1858 | on_key_down => sub { |
|
|
1859 | my ($dialog, $ev) = @_; |
|
|
1860 | $ev->{sym} == 27 and $dialog->hide; |
|
|
1861 | } |
909 | ; |
1862 | ; |
910 | |
1863 | |
911 | $QUIT_DIALOG->add (my $vb = new CFClient::UI::VBox expand => 1); |
1864 | $QUIT_DIALOG->add (my $vb = new CFPlus::UI::VBox expand => 1); |
912 | |
1865 | |
913 | $vb->add (new CFClient::UI::Label |
1866 | $vb->add (new CFPlus::UI::Label |
914 | text => "You should find a savebed and apply it first!", |
1867 | text => "You should find a savebed and apply it first!", |
915 | max_w => $WIDTH * 0.25, |
1868 | max_w => $WIDTH * 0.25, |
916 | ellipsize => 0, |
1869 | ellipsize => 0, |
917 | ); |
1870 | ); |
918 | $vb->add (my $hb = new CFClient::UI::HBox expand => 1); |
1871 | $vb->add (my $hb = new CFPlus::UI::HBox expand => 1); |
919 | $hb->add (new CFClient::UI::Button |
1872 | $hb->add (new CFPlus::UI::Button |
920 | text => "Ok", |
1873 | text => "Ok", |
921 | expand => 1, |
1874 | expand => 1, |
922 | on_activate => sub { $QUIT_DIALOG->hide }, |
1875 | on_activate => sub { $QUIT_DIALOG->hide; 0 }, |
923 | ); |
1876 | ); |
924 | $hb->add (new CFClient::UI::Button |
1877 | $hb->add (new CFPlus::UI::Button |
925 | text => "Quit anyway", |
1878 | text => "Quit anyway", |
926 | expand => 1, |
1879 | expand => 1, |
927 | on_activate => sub { exit }, |
1880 | on_activate => sub { Event::unloop_all }, |
928 | ); |
1881 | ); |
929 | } |
1882 | } |
930 | |
1883 | |
931 | $QUIT_DIALOG->show; |
1884 | $QUIT_DIALOG->show; |
|
|
1885 | $QUIT_DIALOG->grab_focus; |
932 | } |
1886 | } |
933 | |
1887 | |
934 | sub autopickup_setup { |
1888 | sub show_tip_of_the_day { |
935 | my $table = new CFClient::UI::Table; |
1889 | # find all tips |
|
|
1890 | my @tod = CFPlus::Pod::find tip_of_the_day => "*"; |
936 | |
1891 | |
937 | for ( |
1892 | CFPlus::DB::get state => "tip_of_the_day", sub { |
938 | ["General", 0, 0, |
1893 | my ($todindex) = @_; |
939 | ["Enable autopickup" => CFClient::Pickup::PU_NEWMODE], |
1894 | $todindex = 0 if $todindex >= @tod; |
940 | ["Inhibit autopickup" => CFClient::Pickup::PU_INHIBIT], |
1895 | CFPlus::DB::put state => tip_of_the_day => $todindex + 1, sub { }; |
941 | ["Stop before pickup" => CFClient::Pickup::PU_STOP], |
|
|
942 | ["Debug autopickup" => CFClient::Pickup::PU_DEBUG], |
|
|
943 | ], |
|
|
944 | ["Weapons", 0, 6, |
|
|
945 | ["All weapons" => CFClient::Pickup::PU_ALLWEAPON], |
|
|
946 | ["Missile weapons" => CFClient::Pickup::PU_MISSILEWEAPON], |
|
|
947 | ["Bows" => CFClient::Pickup::PU_BOW], |
|
|
948 | ["Arrows" => CFClient::Pickup::PU_ARROW], |
|
|
949 | ], |
|
|
950 | ["Armour", 0, 12, |
|
|
951 | ["Helmets" => CFClient::Pickup::PU_HELMET], |
|
|
952 | ["Shields" => CFClient::Pickup::PU_SHIELD], |
|
|
953 | ["Body Armour" => CFClient::Pickup::PU_ARMOUR], |
|
|
954 | ["Boots" => CFClient::Pickup::PU_BOOTS], |
|
|
955 | ["Gloves" => CFClient::Pickup::PU_GLOVES], |
|
|
956 | ["Cloaks" => CFClient::Pickup::PU_CLOAK], |
|
|
957 | ], |
|
|
958 | |
1896 | |
959 | ["Readables", 2, 2, |
1897 | # create dialog |
960 | ["Spellbooks" => CFClient::Pickup::PU_SPELLBOOK], |
1898 | my $dialog; |
961 | ["Skillscrolls" => CFClient::Pickup::PU_SKILLSCROLL], |
|
|
962 | ["Normal Books/Scrolls" => CFClient::Pickup::PU_READABLES], |
|
|
963 | ], |
|
|
964 | ["Misc", 2, 7, |
|
|
965 | ["Food" => CFClient::Pickup::PU_FOOD], |
|
|
966 | ["Drinks" => CFClient::Pickup::PU_DRINK], |
|
|
967 | ["Valuables (Money, Gems)" => CFClient::Pickup::PU_VALUABLES], |
|
|
968 | ["Keys" => CFClient::Pickup::PU_KEY], |
|
|
969 | ["Magical Items" => CFClient::Pickup::PU_MAGICAL], |
|
|
970 | ["Potions" => CFClient::Pickup::PU_POTION], |
|
|
971 | ["Magic Devices" => CFClient::Pickup::PU_MAGIC_DEVICE], |
|
|
972 | ["Ignore cursed" => CFClient::Pickup::PU_NOT_CURSED], |
|
|
973 | ["Jewelery" => CFClient::Pickup::PU_JEWELS], |
|
|
974 | ], |
|
|
975 | ) |
|
|
976 | { |
|
|
977 | my ($title, $x, $y, @bits) = @$_; |
|
|
978 | $table->add ($x, $y, new CFClient::UI::Label text => $title, align => 1, fg => [1, 1, 0]); |
|
|
979 | |
1899 | |
980 | for (@bits) { |
1900 | my $close = sub { |
981 | ++$y; |
1901 | $dialog->destroy; |
982 | |
|
|
983 | my $mask = $_->[1]; |
|
|
984 | $table->add ($x , $y, new CFClient::UI::Label text => $_->[0], align => 1, expand => 1); |
|
|
985 | $table->add ($x+1, $y, new CFClient::UI::CheckBox |
|
|
986 | state => $CFG->{pickup} & $mask, |
|
|
987 | on_changed => sub { |
|
|
988 | my ($box, $value) = @_; |
|
|
989 | if ($value) { |
|
|
990 | $::CFG->{pickup} |= $mask; |
|
|
991 | } else { |
|
|
992 | $::CFG->{pickup} = $::CFG->{pickup} & ~$mask; |
|
|
993 | } |
|
|
994 | $::CONN->send (sprintf "command pickup %u", $::CFG->{pickup}) |
|
|
995 | if defined $::CONN; |
|
|
996 | }); |
|
|
997 | } |
1902 | }; |
998 | } |
|
|
999 | |
1903 | |
1000 | $table |
1904 | $dialog = new CFPlus::UI::Toplevel |
1001 | } |
|
|
1002 | |
|
|
1003 | sub make_inventory_window { |
|
|
1004 | my $invwin = $INV_WINDOW = new CFClient::UI::FancyFrame |
|
|
1005 | x => "center", |
1905 | x => "center", |
1006 | y => "center", |
1906 | y => "center", |
1007 | force_w => $WIDTH * 9/10, |
1907 | z => 3, |
1008 | force_h => $HEIGHT * 9/10, |
1908 | name => 'tip_of_the_day', |
1009 | title => "Inventory", |
|
|
1010 | ; |
|
|
1011 | |
|
|
1012 | $invwin->add (my $hb = new CFClient::UI::HBox homogeneous => 1); |
|
|
1013 | |
|
|
1014 | $hb->add (my $vb1 = new CFClient::UI::VBox); |
|
|
1015 | $vb1->add (new CFClient::UI::Label align => 0, text => "Player"); |
|
|
1016 | $vb1->add ($INV = new CFClient::UI::Inventory expand => 1); |
|
|
1017 | |
|
|
1018 | $hb->add (my $vb2 = new CFClient::UI::VBox); |
|
|
1019 | |
|
|
1020 | $vb2->add ($INV_RIGHT_HB = new CFClient::UI::HBox); |
|
|
1021 | |
|
|
1022 | $vb2->add ($INVR = new CFClient::UI::Inventory expand => 1); |
|
|
1023 | |
|
|
1024 | # XXX: Call after $INVR = ... because set_opencont sets the items |
|
|
1025 | CFClient::Protocol::set_opencont ($::CONN, 0, "Floor"); |
|
|
1026 | |
|
|
1027 | $invwin |
|
|
1028 | } |
|
|
1029 | |
|
|
1030 | sub spell_setup { |
|
|
1031 | new CFClient::UI::SpellList |
|
|
1032 | } |
|
|
1033 | |
|
|
1034 | sub keyboard_setup { |
|
|
1035 | my $binding_list = new CFClient::UI::VBox; |
|
|
1036 | |
|
|
1037 | my $refresh; |
|
|
1038 | $refresh = sub { |
|
|
1039 | $binding_list->clear (); |
|
|
1040 | |
|
|
1041 | for my $mod (keys %{$::CFG->{bindings}}) { |
|
|
1042 | for my $sym (keys %{$::CFG->{bindings}->{$mod}}) { |
|
|
1043 | my $cmds = $::CFG->{bindings}->{$mod}->{$sym}; |
|
|
1044 | next unless ref $cmds eq 'ARRAY' and @$cmds > 0; |
|
|
1045 | |
|
|
1046 | my $lbl = join "; ", @$cmds; |
|
|
1047 | my $nam = CFClient::Binder::keycombo_to_name ($mod, $sym); |
|
|
1048 | $binding_list->add (my $hb = new CFClient::UI::HBox); |
|
|
1049 | $hb->add (new CFClient::UI::Button |
|
|
1050 | text => "delete", |
|
|
1051 | tooltip => "Deletes the binding", |
|
|
1052 | on_activate => sub { |
|
|
1053 | $binding_list->remove ($hb); |
|
|
1054 | delete $::CFG->{bindings}->{$mod}->{$sym}; |
|
|
1055 | }); |
|
|
1056 | |
|
|
1057 | $hb->add (new CFClient::UI::Button |
|
|
1058 | text => "edit", |
|
|
1059 | tooltip => "Edits the binding", |
|
|
1060 | on_activate => sub { |
|
|
1061 | $::BIND_EDITOR->set_binding ( |
|
|
1062 | $mod, $sym, $::CFG->{bindings}->{$mod}->{$sym}, |
|
|
1063 | sub { |
|
|
1064 | my ($nmod, $nsym, $ncmds) = @_; |
|
|
1065 | delete $::CFG->{bindings}->{$mod}->{$sym}; |
|
|
1066 | $::CFG->{bindings}->{$nmod}->{$nsym} = $ncmds; |
|
|
1067 | $refresh->(); |
|
|
1068 | $SETUP_NOTEBOOK->set_current_page ($SETUP_KEYBOARD); |
|
|
1069 | $SETUP_DIALOG->show; |
|
|
1070 | }, |
|
|
1071 | sub { |
|
|
1072 | $SETUP_NOTEBOOK->set_current_page ($SETUP_KEYBOARD); |
|
|
1073 | $SETUP_DIALOG->show; |
|
|
1074 | }); |
|
|
1075 | $::BIND_EDITOR->show; |
|
|
1076 | $SETUP_DIALOG->hide; |
|
|
1077 | }); |
|
|
1078 | |
|
|
1079 | $hb->add (new CFClient::UI::Label text => "(Key: $nam)"); |
|
|
1080 | $hb->add (new CFClient::UI::Label text => $lbl, expand => 1); |
|
|
1081 | } |
|
|
1082 | } |
|
|
1083 | }; |
|
|
1084 | |
|
|
1085 | my $vb = new CFClient::UI::VBox; |
|
|
1086 | $vb->add ($binding_list); |
|
|
1087 | $vb->add (my $hb = new CFClient::UI::HBox); |
|
|
1088 | |
|
|
1089 | $hb->add (new CFClient::UI::Button |
|
|
1090 | text => "record new", |
|
|
1091 | expand => 1, |
|
|
1092 | tooltip => "This button opens the binding editor with an empty binding.", |
|
|
1093 | on_activate => sub { |
|
|
1094 | $::BIND_EDITOR->set_binding (undef, undef, [], |
|
|
1095 | sub { |
|
|
1096 | my ($mod, $sym, $cmds) = @_; |
|
|
1097 | $::CFG->{bindings}->{$mod}->{$sym} = $cmds; |
|
|
1098 | $refresh->(); |
|
|
1099 | $SETUP_NOTEBOOK->set_current_page ($SETUP_KEYBOARD); |
|
|
1100 | $SETUP_DIALOG->show; |
|
|
1101 | }, |
|
|
1102 | sub { |
|
|
1103 | $SETUP_NOTEBOOK->set_current_page ($SETUP_KEYBOARD); |
|
|
1104 | $SETUP_DIALOG->show; |
|
|
1105 | }); |
|
|
1106 | $SETUP_DIALOG->hide; |
|
|
1107 | $::BIND_EDITOR->show; |
|
|
1108 | }, |
|
|
1109 | ); |
|
|
1110 | |
|
|
1111 | $hb->add (new CFClient::UI::Button |
|
|
1112 | text => "close", |
|
|
1113 | tooltip => "Closes the binding window", |
|
|
1114 | expand => 1, |
|
|
1115 | on_activate => sub { |
|
|
1116 | $SETUP_DIALOG->hide; |
|
|
1117 | } |
|
|
1118 | ); |
|
|
1119 | |
|
|
1120 | $refresh->(); |
|
|
1121 | |
|
|
1122 | $vb |
|
|
1123 | } |
|
|
1124 | |
|
|
1125 | sub make_help_window { |
|
|
1126 | my $win = new CFClient::UI::FancyFrame |
|
|
1127 | x => 'center', |
|
|
1128 | y => 'center', |
|
|
1129 | name => 'doc_browser', |
|
|
1130 | force_w => int $WIDTH * 7/8, |
1909 | force_w => int $WIDTH * 4/9, |
1131 | force_h => int $HEIGHT * 7/8, |
1910 | force_h => int $WIDTH * 2/9, |
1132 | title => "Documentation"; |
1911 | title => "Tip of the day #" . (1 + $todindex), |
|
|
1912 | child => my $vbox = new CFPlus::UI::VBox, |
|
|
1913 | has_close_button => 1, |
|
|
1914 | on_delete => $close, |
|
|
1915 | ; |
1133 | |
1916 | |
1134 | $win->add (my $vbox = new CFClient::UI::VBox); |
1917 | $vbox->add (my $viewer = new CFPlus::UI::TextScroller |
|
|
1918 | expand => 1, fontsize => 0.8, padding_x => 4, padding_y => 4); |
|
|
1919 | $viewer->add_paragraph (CFPlus::Pod::as_paragraphs CFPlus::Pod::section_of $tod[$todindex]); |
1135 | |
1920 | |
1136 | $vbox->add (my $buttons = new CFClient::UI::HBox); |
1921 | $vbox->add (my $table = new CFPlus::UI::Table col_expand => [0, 1]); |
1137 | $vbox->add (my $viewer = new CFClient::UI::TextView expand => 1, fontsize => 0.8); |
|
|
1138 | |
1922 | |
1139 | for ( |
1923 | $table->add_at (0, 0, new CFPlus::UI::Button |
1140 | [intro => "Introduction"], |
|
|
1141 | [manual => "Manual"], |
|
|
1142 | [command_help => "Commands"], |
|
|
1143 | [skill_help => "Skills"], |
|
|
1144 | ) { |
|
|
1145 | my ($pod, $label) = @$_; |
|
|
1146 | |
|
|
1147 | $buttons->add (new CFClient::UI::Button |
|
|
1148 | text => $label, |
1924 | text => "Close", |
|
|
1925 | tooltip => "Close the tip of the day window. To never see it again, disable the tip of the day in the <b>Server Setup</b>.", |
|
|
1926 | on_activate => $close, |
|
|
1927 | ); |
|
|
1928 | |
|
|
1929 | $table->add_at (2, 0, new CFPlus::UI::Button |
|
|
1930 | text => "Next", |
|
|
1931 | tooltip => "Show the next <b>Tip of the day</b>.", |
1149 | on_activate => sub { |
1932 | on_activate => sub { |
1150 | my $parser = new Pod::POM; |
1933 | $close->(); |
1151 | my $pom = $parser->parse_file (CFClient::find_rcfile "pod/$pod.pod"); |
1934 | &show_tip_of_the_day; |
1152 | |
|
|
1153 | $viewer->clear; |
|
|
1154 | |
|
|
1155 | $viewer->add_paragraph ([1, 1, 1, 1], $_->[1], $_->[0]) |
|
|
1156 | for @{ CFClient::pod_to_pango_list $pom }; |
|
|
1157 | |
|
|
1158 | $viewer->set_offset (0); |
|
|
1159 | }, |
1935 | }, |
1160 | ); |
1936 | ); |
|
|
1937 | |
|
|
1938 | $dialog->show; |
1161 | } |
1939 | }; |
1162 | |
|
|
1163 | $viewer->add_paragraph ([1, 1, 0, 1], "<big>Use one of the buttons above to display a document.</big>"); |
|
|
1164 | |
|
|
1165 | $win |
|
|
1166 | } |
1940 | } |
1167 | |
1941 | |
1168 | sub sdl_init { |
1942 | sub sdl_init { |
1169 | CFClient::SDL_Init |
1943 | CFPlus::SDL_Init |
1170 | and die "SDL::Init failed!\n"; |
1944 | and die "SDL::Init failed!\n"; |
1171 | } |
1945 | } |
1172 | |
1946 | |
1173 | sub video_init { |
1947 | sub video_init { |
1174 | sdl_init; |
|
|
1175 | |
|
|
1176 | $CFG->{sdl_mode} = 0 if $CFG->{sdl_mode} >= @SDL_MODES; |
1948 | $CFG->{sdl_mode} = 0 if $CFG->{sdl_mode} >= @SDL_MODES; |
1177 | |
1949 | |
1178 | my ($old_w, $old_h) = ($WIDTH, $HEIGHT); |
1950 | my ($old_w, $old_h) = ($WIDTH, $HEIGHT); |
1179 | |
1951 | |
1180 | ($WIDTH, $HEIGHT) = @{ $SDL_MODES[$CFG->{sdl_mode}] }; |
1952 | ($WIDTH, $HEIGHT, my ($rgb, $alpha)) = @{ $SDL_MODES[$CFG->{sdl_mode}] }; |
1181 | $FULLSCREEN = $CFG->{fullscreen}; |
1953 | $FULLSCREEN = $CFG->{fullscreen}; |
1182 | $FAST = $CFG->{fast}; |
1954 | $FAST = $CFG->{fast}; |
1183 | |
1955 | |
1184 | CFClient::SDL_SetVideoMode $WIDTH, $HEIGHT, $FULLSCREEN |
1956 | CFPlus::SDL_SetVideoMode $WIDTH, $HEIGHT, $rgb, $alpha, $FULLSCREEN |
1185 | or die "SDL_SetVideoMode failed: " . (CFClient::SDL_GetError) . "\n"; |
1957 | or die "SDL_SetVideoMode failed: " . (CFPlus::SDL_GetError) . "\n"; |
1186 | |
1958 | |
1187 | $SDL_ACTIVE = 1; |
1959 | $SDL_ACTIVE = 1; |
1188 | $LAST_REFRESH = time - 0.01; |
1960 | $LAST_REFRESH = time - 0.01; |
1189 | |
1961 | |
1190 | CFClient::OpenGL::init; |
1962 | CFPlus::OpenGL::init; |
1191 | |
1963 | |
1192 | $FONTSIZE = int $HEIGHT / 40 * $CFG->{gui_fontsize}; |
1964 | $FONTSIZE = int $HEIGHT / 40 * $CFG->{gui_fontsize}; |
1193 | |
1965 | |
1194 | $CFClient::UI::ROOT->configure (0, 0, $WIDTH, $HEIGHT);#d# |
1966 | $CFPlus::UI::ROOT->configure (0, 0, $WIDTH, $HEIGHT);#d# |
1195 | |
1967 | |
1196 | ############################################################################# |
1968 | ############################################################################# |
1197 | |
1969 | |
1198 | if ($DEBUG_STATUS) { |
1970 | if ($DEBUG_STATUS) { |
1199 | CFClient::UI::rescale_widgets $WIDTH / $old_w, $HEIGHT / $old_h; |
1971 | CFPlus::UI::rescale_widgets $WIDTH / $old_w, $HEIGHT / $old_h; |
1200 | } else { |
1972 | } else { |
1201 | # create the widgets |
1973 | # create the widgets |
1202 | |
1974 | |
1203 | $DEBUG_STATUS = new CFClient::UI::Label |
1975 | $DEBUG_STATUS = new CFPlus::UI::Label |
1204 | padding => 0, |
1976 | padding => 0, |
1205 | z => 100, |
1977 | z => 100, |
1206 | force_x => "max", |
1978 | force_x => "max", |
1207 | force_y => 0; |
1979 | force_y => 0; |
1208 | $DEBUG_STATUS->show; |
1980 | $DEBUG_STATUS->show; |
1209 | |
1981 | |
1210 | $BIND_EDITOR = new CFClient::UI::BindEditor (x => "max", y => 0); |
|
|
1211 | |
|
|
1212 | $STATUSBOX = new CFClient::UI::Statusbox; |
1982 | $STATUSBOX = new CFPlus::UI::Statusbox; |
1213 | $STATUSBOX->add ("Use <b>Alt-Enter</b> to toggle fullscreen mode", pri => -100, color => [1, 1, 1, 0.8]); |
1983 | $STATUSBOX->add ("Use <b>Alt-Enter</b> to toggle fullscreen mode", timeout => 864000, pri => -100, color => [1, 1, 1, 0.8]); |
1214 | |
1984 | |
1215 | (new CFClient::UI::Frame |
1985 | (new CFPlus::UI::Frame |
1216 | bg => [0, 0, 0, 0.4], |
1986 | bg => [0, 0, 0, 0.4], |
1217 | force_x => 0, |
1987 | force_x => 0, |
1218 | force_y => "max", |
1988 | force_y => "max", |
1219 | child => $STATUSBOX, |
1989 | child => $STATUSBOX, |
1220 | )->show; |
1990 | )->show; |
1221 | |
1991 | |
1222 | CFClient::UI::FancyFrame->new ( |
1992 | CFPlus::UI::Toplevel->new ( |
1223 | title => "Map", |
1993 | title => "Map", |
1224 | name => "mapmap", |
1994 | name => "mapmap", |
1225 | x => 0, |
1995 | x => 0, |
1226 | y => $FONTSIZE + 8, |
1996 | y => $FONTSIZE + 8, |
1227 | border_bg => [1, 1, 1, 192/255], |
1997 | border_bg => [1, 1, 1, 192/255], |
1228 | bg => [1, 1, 1, 0], |
1998 | bg => [1, 1, 1, 0], |
1229 | child => ($MAPMAP = new CFClient::MapWidget::MapMap |
1999 | child => ($MAPMAP = new CFPlus::MapWidget::MapMap |
1230 | tooltip => "<b>Map</b>. On servers that support this feature, this will display an overview of the surrounding areas.", |
2000 | tooltip => "<b>Map</b>. On servers that support this feature, this will display an overview of the surrounding areas.", |
1231 | ), |
2001 | ), |
1232 | )->show; |
2002 | )->show; |
1233 | |
2003 | |
1234 | $MAPWIDGET = new CFClient::MapWidget; |
2004 | $MAPWIDGET = new CFPlus::MapWidget; |
1235 | $MAPWIDGET->connect (activate_console => sub { |
2005 | $MAPWIDGET->connect (activate_console => sub { |
1236 | my ($mapwidget, $preset) = @_; |
2006 | my ($mapwidget, $preset) = @_; |
1237 | |
2007 | |
1238 | if ($CONSOLE) { |
2008 | $MESSAGE_WINDOW->activate_console ($preset) |
1239 | $CONSOLE->{input}->{auto_activated} = 1; |
2009 | if $MESSAGE_WINDOW; |
1240 | $CONSOLE->{input}->focus_in; |
|
|
1241 | |
|
|
1242 | if ($preset && $CONSOLE->{input}->get_text eq '') { |
|
|
1243 | $CONSOLE->{input}->set_text ($preset); |
|
|
1244 | } |
|
|
1245 | } |
|
|
1246 | }); |
2010 | }); |
1247 | $MAPWIDGET->show; |
2011 | $MAPWIDGET->show; |
1248 | $MAPWIDGET->focus_in; |
2012 | $MAPWIDGET->grab_focus; |
1249 | |
2013 | |
1250 | $LOGVIEW = new CFClient::UI::TextView |
|
|
1251 | expand => 1, |
|
|
1252 | font => $FONT_FIXED, |
|
|
1253 | fontsize => $::CFG->{log_fontsize}, |
|
|
1254 | can_hover => 1, |
|
|
1255 | can_events => 1, |
|
|
1256 | tooltip => "<b>Server Log</b>. This text viewer contains all the messages sent by the server.", |
|
|
1257 | ; |
|
|
1258 | |
|
|
1259 | $SETUP_DIALOG = new CFClient::UI::FancyFrame |
2014 | $SETUP_DIALOG = new CFPlus::UI::Toplevel |
1260 | title => "Setup", |
2015 | title => "Setup", |
1261 | name => "setup_dialog", |
2016 | name => "setup_dialog", |
1262 | x => 'center', |
2017 | x => 'center', |
1263 | y => 'center', |
2018 | y => 'center', |
|
|
2019 | z => 2, |
1264 | force_w => $::WIDTH * 0.6, |
2020 | force_w => $::WIDTH * 0.6, |
1265 | force_h => $::HEIGHT * 0.6, |
2021 | force_h => $::HEIGHT * 0.6, |
|
|
2022 | has_close_button => 1, |
1266 | ; |
2023 | ; |
1267 | |
2024 | |
|
|
2025 | $METASERVER = metaserver_dialog; |
|
|
2026 | $MESSAGE_WINDOW = new CFPlus::UI::MessageWindow; |
|
|
2027 | |
1268 | $SETUP_DIALOG->add ($SETUP_NOTEBOOK = new CFClient::UI::Notebook expand => 1, debug => 1, |
2028 | $SETUP_DIALOG->add ($SETUP_NOTEBOOK = new CFPlus::UI::Notebook expand => 1, debug => 1, |
1269 | filter => new CFClient::UI::ScrolledWindow xxx => 1, expand => 1, scroll_y => 1); |
2029 | filter => new CFPlus::UI::ScrolledWindow expand => 1, scroll_y => 1); |
1270 | |
2030 | |
1271 | $SETUP_NOTEBOOK->add (Server => $SETUP_SERVER = server_setup, |
2031 | $SETUP_NOTEBOOK->add_tab (Server => $SETUP_SERVER = server_setup, |
1272 | "Configure the server to play on, your username, password and other server-related options."); |
2032 | "Configure the server to play on, your username, password and other server-related options."); |
1273 | $SETUP_NOTEBOOK->add (Pickup => autopickup_setup, |
2033 | $SETUP_NOTEBOOK->add_tab (Client => client_setup, |
1274 | "Configure autopicking stetings, i.e. which items you will pick up automatically when walking over them."); |
2034 | "Configure various client-specific settings."); |
1275 | $SETUP_NOTEBOOK->add (Graphics => graphics_setup, |
2035 | $SETUP_NOTEBOOK->add_tab (Graphics => graphics_setup, |
1276 | "Configure the video mode, performance, fonts and other graphical aspects of the game."); |
2036 | "Configure the video mode, performance, fonts and other graphical aspects of the game."); |
1277 | $SETUP_NOTEBOOK->add (Audio => audio_setup, |
2037 | $SETUP_NOTEBOOK->add_tab (Audio => audio_setup, |
1278 | "Configure the use of audio, sound effects and background music."); |
2038 | "Configure the use of audio, sound effects and background music."); |
1279 | $SETUP_NOTEBOOK->add (Keyboard => $SETUP_KEYBOARD = keyboard_setup, |
2039 | $SETUP_NOTEBOOK->add_tab (Keyboard => $SETUP_KEYBOARD = keyboard_setup, |
1280 | "Lets you define, edit and delete bindings." |
2040 | "Lets you define, edit and delete key bindings." |
1281 | . "There is a shortcut for making bindings: Left Control + Insert opens the binding editor " |
2041 | . "There is a shortcut for making bindings: <b>Control-Insert</b> opens the binding editor " |
1282 | . "with nothing set and the recording started. After doing the actions you " |
2042 | . "with nothing set and the recording started. After doing the actions you " |
1283 | . "want to record press Insert and you will be asked to press a key-combo. " |
2043 | . "want to record press <b>Insert</b> and you will be asked to press a key-combo. " |
1284 | . "After pressing the combo the binding will be saved automatically and the " |
2044 | . "After pressing the combo the binding will be saved automatically and the " |
1285 | . "binding editor closes"); |
2045 | . "binding editor closes"); |
1286 | $SETUP_NOTEBOOK->add (Spells => $SETUP_SPELLS = spell_setup, |
2046 | $SETUP_NOTEBOOK->add_tab (Debug => debug_setup, |
1287 | "Displays all spells you have and lets you edit keyboard shortcuts for them."); |
2047 | "Some debuggin' options. Do not ask."); |
1288 | |
2048 | |
1289 | $BUTTONBAR = new CFClient::UI::HBox x => 0, y => 0; |
2049 | $BUTTONBAR = new CFPlus::UI::Buttonbar x => 0, y => 0, z => 200; # put on top |
1290 | |
2050 | |
1291 | $BUTTONBAR->add (new CFClient::UI::Flopper text => "Setup", other => $SETUP_DIALOG, |
2051 | $BUTTONBAR->add (new CFPlus::UI::Flopper text => "Setup", other => $SETUP_DIALOG, |
1292 | tooltip => "Toggles a dialog where you can configure all aspects of this client."); |
2052 | tooltip => "Toggles a dialog where you can configure all aspects of this client."); |
1293 | |
2053 | |
1294 | $BUTTONBAR->add (new CFClient::UI::Flopper text => "Message Window", other => message_window, |
2054 | $BUTTONBAR->add (new CFPlus::UI::Flopper text => "Message Window", other => $MESSAGE_WINDOW, |
1295 | tooltip => "Toggles the server message log, where the client collects <i>all</i> messages from the server."); |
2055 | tooltip => "Toggles the server message log, where the client collects <i>all</i> messages from the server."); |
1296 | |
2056 | |
1297 | 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 |
2057 | 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 |
1298 | |
2058 | |
1299 | $BUTTONBAR->add (new CFClient::UI::Flopper text => "Stats Window", other => make_stats_window, |
|
|
1300 | tooltip => "Toggles the statistics window, where all your Stats and Resistances are being displayed at all times."); |
|
|
1301 | $BUTTONBAR->add (new CFClient::UI::Flopper text => "Inventory", other => make_inventory_window, |
2059 | $BUTTONBAR->add (new CFPlus::UI::Flopper text => "Playerbook", other => player_window, |
1302 | tooltip => "Toggles the inventory window, where you can manage your loot (or treasures :). " |
2060 | tooltip => "Toggles the player view, where you can manage Inventory, Spells, Skills and see your Stats."); |
1303 | . "You can also hit the <b>Tab</b>-key to show/hide the Inventory."); |
|
|
1304 | |
2061 | |
1305 | $BUTTONBAR->add (new CFClient::UI::Button |
2062 | $BUTTONBAR->add (new CFPlus::UI::Button |
1306 | text => "Save Config", |
2063 | text => "Save Config", |
1307 | tooltip => "Saves the options chosen in the client setting, server settings and the window layout to be restored on later runs.", |
2064 | tooltip => "Saves the options chosen in the client setting, server settings and the window layout to be restored on later runs.", |
1308 | on_activate => sub { |
2065 | on_activate => sub { |
1309 | $::CFG->{layout} = CFClient::UI::get_layout; |
2066 | $::CFG->{layout} = CFPlus::UI::get_layout; |
1310 | CFClient::write_cfg "$Crossfire::VARDIR/cfplusrc"; |
2067 | CFPlus::write_cfg "$Crossfire::VARDIR/cfplusrc"; |
1311 | status "Configuration Saved"; |
2068 | status "Configuration Saved"; |
|
|
2069 | 0 |
1312 | }, |
2070 | }, |
1313 | ); |
2071 | ); |
1314 | |
2072 | |
1315 | $BUTTONBAR->add (new CFClient::UI::Flopper text => "Help!", other => make_help_window, |
2073 | $BUTTONBAR->add (new CFPlus::UI::Flopper text => "Help!", other => $HELP_WINDOW = help_window, |
1316 | tooltip => "View Documentation"); |
2074 | tooltip => "View Documentation"); |
1317 | |
2075 | |
|
|
2076 | |
1318 | $BUTTONBAR->add (new CFClient::UI::Button |
2077 | $BUTTONBAR->add (new CFPlus::UI::Button |
1319 | text => "Quit", |
2078 | text => "Quit", |
1320 | tooltip => "Terminates the program", |
2079 | tooltip => "Terminates the program", |
1321 | on_activate => sub { |
2080 | on_activate => sub { |
1322 | if ($CONN) { |
2081 | if ($CONN) { |
1323 | open_quit_dialog; |
2082 | open_quit_dialog; |
1324 | } else { |
2083 | } else { |
1325 | exit; |
2084 | Event::unloop_all; |
1326 | } |
2085 | } |
|
|
2086 | 0 |
1327 | }, |
2087 | }, |
1328 | ); |
2088 | ); |
1329 | |
2089 | |
1330 | $BUTTONBAR->show; |
2090 | $BUTTONBAR->show; |
1331 | $SETUP_DIALOG->show; |
2091 | $SETUP_DIALOG->show; |
|
|
2092 | $MESSAGE_WINDOW->show; |
1332 | } |
2093 | } |
1333 | |
2094 | |
1334 | $STATUSBOX->add ("Set video mode $WIDTH×$HEIGHT", timeout => 10, fg => [1, 1, 1, 0.5]); |
2095 | $STATUSBOX->add ("Set video mode $WIDTH×$HEIGHT", timeout => 10, fg => [1, 1, 1, 0.5]); |
1335 | } |
2096 | } |
1336 | |
2097 | |
1337 | sub video_shutdown { |
2098 | sub video_shutdown { |
|
|
2099 | CFPlus::OpenGL::shutdown; |
|
|
2100 | |
1338 | undef $SDL_ACTIVE; |
2101 | undef $SDL_ACTIVE; |
1339 | } |
|
|
1340 | |
|
|
1341 | my @bgmusic = qw(game1.ogg game2.ogg game3.ogg game5.ogg game6.ogg ross1.ogg ross2.ogg ross3.ogg ross4.ogg ross5.ogg); #d# |
|
|
1342 | my $bgmusic;#TODO#hack#d# |
|
|
1343 | |
|
|
1344 | sub audio_channel_finished { |
|
|
1345 | my ($channel) = @_; |
|
|
1346 | |
|
|
1347 | #warn "channel $channel finished\n";#d# |
|
|
1348 | } |
|
|
1349 | |
|
|
1350 | sub audio_music_finished { |
|
|
1351 | return unless $CFG->{bgm_enable}; |
|
|
1352 | |
|
|
1353 | # TODO: hack, do play loop and mood music |
|
|
1354 | $bgmusic = new_from_file CFClient::MixMusic CFClient::find_rcfile "music/$bgmusic[0]"; |
|
|
1355 | $bgmusic->play (0); |
|
|
1356 | |
|
|
1357 | push @bgmusic, shift @bgmusic; |
|
|
1358 | } |
|
|
1359 | |
|
|
1360 | sub audio_init { |
|
|
1361 | if ($CFG->{audio_enable}) { |
|
|
1362 | if (open my $fh, "<", CFClient::find_rcfile "sounds/config") { |
|
|
1363 | $SDL_MIXER = !CFClient::Mix_OpenAudio; |
|
|
1364 | |
|
|
1365 | unless ($SDL_MIXER) { |
|
|
1366 | status "Unable to open sound device: there will be no sound"; |
|
|
1367 | return; |
|
|
1368 | } |
|
|
1369 | |
|
|
1370 | CFClient::Mix_AllocateChannels 8; |
|
|
1371 | CFClient::MixMusic::volume $CFG->{bgm_volume} * 128; |
|
|
1372 | |
|
|
1373 | audio_music_finished; |
|
|
1374 | |
|
|
1375 | while (<$fh>) { |
|
|
1376 | next if /^\s*#/; |
|
|
1377 | next if /^\s*$/; |
|
|
1378 | |
|
|
1379 | my ($file, $volume, $event) = split /\s+/, $_, 3; |
|
|
1380 | |
|
|
1381 | push @SOUNDS, "$volume,$file"; |
|
|
1382 | |
|
|
1383 | $AUDIO_CHUNKS{"$volume,$file"} ||= do { |
|
|
1384 | my $chunk = new_from_file CFClient::MixChunk CFClient::find_rcfile "sounds/$file"; |
|
|
1385 | $chunk->volume ($volume * 128 / 100); |
|
|
1386 | $chunk |
|
|
1387 | }; |
|
|
1388 | } |
|
|
1389 | } else { |
|
|
1390 | status "unable to open sound config: $!"; |
|
|
1391 | } |
|
|
1392 | } |
|
|
1393 | } |
|
|
1394 | |
|
|
1395 | sub audio_shutdown { |
|
|
1396 | CFClient::Mix_CloseAudio if $SDL_MIXER; |
|
|
1397 | undef $SDL_MIXER; |
|
|
1398 | @SOUNDS = (); |
|
|
1399 | %AUDIO_CHUNKS = (); |
|
|
1400 | } |
2102 | } |
1401 | |
2103 | |
1402 | my %animate_object; |
2104 | my %animate_object; |
1403 | my $animate_timer; |
2105 | my $animate_timer; |
1404 | |
2106 | |
1405 | my $fps = 9; |
2107 | my $fps = 9; |
1406 | |
2108 | |
1407 | my %demo;#d# |
|
|
1408 | |
|
|
1409 | sub force_refresh { |
2109 | sub force_refresh { |
|
|
2110 | if ($ENV{CFPLUS_DEBUG} & 4) { |
1410 | $fps = $fps * 0.95 + 1 / (($NOW - $LAST_REFRESH) || 0.1) * 0.05; |
2111 | $fps = $fps * 0.98 + 1 / (($NOW - $LAST_REFRESH) || 0.1) * 0.02; |
1411 | debug sprintf "%3.2f", $fps if $ENV{CFPLUS_DEBUG} & 4; |
2112 | debug sprintf "%3.2f", $fps; |
|
|
2113 | } |
1412 | |
2114 | |
1413 | $CFClient::UI::ROOT->draw; |
2115 | $CFPlus::UI::ROOT->draw; |
1414 | |
2116 | CFPlus::SDL_GL_SwapBuffers; |
1415 | $WANT_REFRESH = 0; |
|
|
1416 | $CAN_REFRESH = 0; |
|
|
1417 | $LAST_REFRESH = $NOW; |
2117 | $LAST_REFRESH = $NOW; |
1418 | |
|
|
1419 | 0 && do { |
|
|
1420 | # some weird model-drawing code, just a joke right now |
|
|
1421 | use CFClient::OpenGL; |
|
|
1422 | |
|
|
1423 | $demo{t}{eye_auv} ||= new_from_file CFClient::Texture "eye2.png" or die; |
|
|
1424 | $demo{t}{body_auv} ||= new_from_file CFClient::Texture "body_auv3.png" or die; |
|
|
1425 | $demo{r} ||= do { |
|
|
1426 | my $mod = Compress::LZF::sthaw do { local $/; open my $fh, "<:raw:perlio", "dread.lz3"; <$fh> }; |
|
|
1427 | $mod->{v} = pack "f*", @{$mod->{v}}; |
|
|
1428 | $_ = [scalar @$_, pack "S!*", @$_] |
|
|
1429 | for values %{$mod->{g}}; |
|
|
1430 | $mod |
|
|
1431 | }; |
|
|
1432 | |
|
|
1433 | my $r = $demo{r} or die; |
|
|
1434 | |
|
|
1435 | glDepthMask 1; |
|
|
1436 | glClear GL_DEPTH_BUFFER_BIT; |
|
|
1437 | glEnable GL_TEXTURE_2D; |
|
|
1438 | glEnable GL_DEPTH_TEST; |
|
|
1439 | glEnable GL_CULL_FACE; |
|
|
1440 | glShadeModel $::FAST ? GL_FLAT : GL_SMOOTH; |
|
|
1441 | |
|
|
1442 | glMatrixMode GL_PROJECTION; |
|
|
1443 | glLoadIdentity; |
|
|
1444 | glFrustum -1 * ($::WIDTH / $::HEIGHT), 1 * ($::WIDTH / $::HEIGHT), 1, -1, 1, 10000; |
|
|
1445 | #glOrtho 0, $::WIDTH, 0, $::HEIGHT, -10000, 10000; |
|
|
1446 | glMatrixMode GL_MODELVIEW; |
|
|
1447 | glLoadIdentity; |
|
|
1448 | |
|
|
1449 | glPushMatrix; |
|
|
1450 | glTranslate 0, 0, -800; |
|
|
1451 | glScale 1, -1, 1; |
|
|
1452 | glRotate $NOW * 1000 % 36000 / 5, 0, 1, 0; |
|
|
1453 | glRotate $NOW * 1000 % 36000 / 6, 1, 0, 0; |
|
|
1454 | glRotate $NOW * 1000 % 36000 / 7, 0, 0, 1; |
|
|
1455 | glScale 50, 50, 50; |
|
|
1456 | |
|
|
1457 | glInterleavedArrays GL_T2F_N3F_V3F, 0, $r->{v}; |
|
|
1458 | while (my ($k, $v) = each %{$r->{g}}) { |
|
|
1459 | glBindTexture GL_TEXTURE_2D, ($demo{t}{$k}{name} or die); |
|
|
1460 | glDrawElements GL_TRIANGLES, $v->[0], GL_UNSIGNED_SHORT, $v->[1]; |
|
|
1461 | } |
|
|
1462 | |
|
|
1463 | glPopMatrix; |
|
|
1464 | |
|
|
1465 | glShadeModel GL_FLAT; |
|
|
1466 | glDisable GL_DEPTH_TEST; |
|
|
1467 | glDisable GL_TEXTURE_2D; |
|
|
1468 | glDepthMask 0; |
|
|
1469 | |
|
|
1470 | $WANT_REFRESH++; |
2118 | $WANT_REFRESH->stop; |
1471 | }; |
|
|
1472 | |
|
|
1473 | CFClient::SDL_GL_SwapBuffers; |
|
|
1474 | } |
2119 | } |
1475 | |
2120 | |
|
|
2121 | $WANT_REFRESH = Event->idle (min => 0.001, max => 0.06, parked => 1, cb => \&force_refresh); |
|
|
2122 | |
1476 | my $refresh_watcher = Event->timer (after => 0, hard => 0, interval => 1 / $MAX_FPS, cb => sub { |
2123 | my $input = Event->timer (after => 0, hard => 0, interval => 1 / 50, cb => sub { |
1477 | $NOW = time; |
2124 | $NOW = time; |
1478 | |
2125 | |
1479 | ($SDL_CB{$_->{type}} || sub { warn "unhandled event $_->{type}" })->($_) |
2126 | ($SDL_CB{$_->{type}} || sub { warn "unhandled event $_->{type}" })->($_) |
1480 | for CFClient::SDL_PollEvent; |
2127 | for CFPlus::poll_events; |
1481 | |
2128 | |
1482 | if (%animate_object) { |
2129 | if (%animate_object) { |
1483 | $_->animate ($LAST_REFRESH - $NOW) for values %animate_object; |
2130 | $_->animate ($LAST_REFRESH - $NOW) for values %animate_object; |
1484 | $WANT_REFRESH++; |
2131 | $WANT_REFRESH->start; |
1485 | } |
|
|
1486 | |
|
|
1487 | if ($WANT_REFRESH) { |
|
|
1488 | force_refresh; |
|
|
1489 | } else { |
|
|
1490 | $CAN_REFRESH = 1; |
|
|
1491 | } |
2132 | } |
1492 | }); |
2133 | }); |
1493 | |
2134 | |
1494 | sub animation_start { |
2135 | sub animation_start { |
1495 | my ($widget) = @_; |
2136 | my ($widget) = @_; |
… | |
… | |
1499 | sub animation_stop { |
2140 | sub animation_stop { |
1500 | my ($widget) = @_; |
2141 | my ($widget) = @_; |
1501 | delete $animate_object{$widget}; |
2142 | delete $animate_object{$widget}; |
1502 | } |
2143 | } |
1503 | |
2144 | |
1504 | # check once/second for faces that need to be prefetched |
|
|
1505 | # this should, of course, only run on demand, but |
|
|
1506 | # SDL forces worse things on us.... |
|
|
1507 | |
|
|
1508 | Event->timer (after => 1, interval => 0.25, cb => sub { |
|
|
1509 | $CONN->face_prefetch |
|
|
1510 | if $CONN; |
|
|
1511 | }); |
|
|
1512 | |
|
|
1513 | %SDL_CB = ( |
2145 | %SDL_CB = ( |
1514 | CFClient::SDL_QUIT => sub { |
2146 | CFPlus::SDL_QUIT => sub { |
1515 | Event::unloop -1; |
2147 | exit; |
1516 | }, |
2148 | }, |
1517 | CFClient::SDL_VIDEORESIZE => sub { |
2149 | CFPlus::SDL_VIDEORESIZE => sub { |
1518 | }, |
2150 | }, |
1519 | CFClient::SDL_VIDEOEXPOSE => sub { |
2151 | CFPlus::SDL_VIDEOEXPOSE => sub { |
1520 | CFClient::UI::full_refresh; |
2152 | CFPlus::UI::full_refresh; |
1521 | }, |
2153 | }, |
1522 | CFClient::SDL_ACTIVEEVENT => sub { |
2154 | CFPlus::SDL_ACTIVEEVENT => sub { |
1523 | # printf "active %x %x\n", $SDL_EV->active_gain, $SDL_EV->active_state;#d# |
2155 | # not useful, as APPACTIVE include sonly iconified state, not unmapped |
|
|
2156 | # printf "active %x %x %x\n", $_[0]{gain}, $_[0]{state}, CFPlus::SDL_GetAppState;#d# |
|
|
2157 | # printf "a %x\n", CFPlus::SDL_GetAppState & CFPlus::SDL_APPACTIVE;#d# |
|
|
2158 | # printf "A\n" if $_[0]{state} & CFPlus::SDL_APPACTIVE; |
|
|
2159 | # printf "K\n" if $_[0]{state} & CFPlus::SDL_APPINPUTFOCUS; |
|
|
2160 | # printf "M\n" if $_[0]{state} & CFPlus::SDL_APPMOUSEFOCUS; |
1524 | }, |
2161 | }, |
1525 | CFClient::SDL_KEYDOWN => sub { |
2162 | CFPlus::SDL_KEYDOWN => sub { |
1526 | if ($_[0]{mod} & CFClient::KMOD_ALT && $_[0]{sym} == 13) { |
2163 | if ($_[0]{mod} & CFPlus::KMOD_ALT && $_[0]{sym} == 13) { |
1527 | # alt-enter |
2164 | # alt-enter |
|
|
2165 | $FULLSCREEN_ENABLE->toggle; |
1528 | video_shutdown; |
2166 | video_shutdown; |
1529 | $CFG->{fullscreen} = !$CFG->{fullscreen}; |
|
|
1530 | video_init; |
2167 | video_init; |
1531 | } else { |
2168 | } else { |
1532 | CFClient::UI::feed_sdl_key_down_event ($_[0]); |
2169 | CFPlus::UI::feed_sdl_key_down_event ($_[0]); |
1533 | } |
2170 | } |
1534 | }, |
2171 | }, |
1535 | CFClient::SDL_KEYUP => \&CFClient::UI::feed_sdl_key_up_event, |
2172 | CFPlus::SDL_KEYUP => \&CFPlus::UI::feed_sdl_key_up_event, |
1536 | CFClient::SDL_MOUSEMOTION => \&CFClient::UI::feed_sdl_motion_event, |
2173 | CFPlus::SDL_MOUSEMOTION => \&CFPlus::UI::feed_sdl_motion_event, |
1537 | CFClient::SDL_MOUSEBUTTONDOWN => \&CFClient::UI::feed_sdl_button_down_event, |
2174 | CFPlus::SDL_MOUSEBUTTONDOWN => \&CFPlus::UI::feed_sdl_button_down_event, |
1538 | CFClient::SDL_MOUSEBUTTONUP => \&CFClient::UI::feed_sdl_button_up_event, |
2175 | CFPlus::SDL_MOUSEBUTTONUP => \&CFPlus::UI::feed_sdl_button_up_event, |
1539 | CFClient::SDL_USEREVENT => sub { |
2176 | CFPlus::SDL_USEREVENT => sub { |
1540 | if ($_[0]{code} == 1) { |
2177 | if ($_[0]{code} == 1) { |
1541 | audio_channel_finished $_[0]{data1}; |
2178 | audio_channel_finished $_[0]{data1}; |
1542 | } elsif ($_[0]{code} == 0) { |
2179 | } elsif ($_[0]{code} == 0) { |
1543 | audio_music_finished; |
2180 | audio_music_finished; |
1544 | } |
2181 | } |
… | |
… | |
1548 | ############################################################################# |
2185 | ############################################################################# |
1549 | |
2186 | |
1550 | $SIG{INT} = $SIG{TERM} = sub { exit }; |
2187 | $SIG{INT} = $SIG{TERM} = sub { exit }; |
1551 | |
2188 | |
1552 | { |
2189 | { |
1553 | local $SIG{__DIE__} = sub { |
|
|
1554 | return unless defined $^S && !$^S; |
|
|
1555 | Carp::confess $_[1];#d#TODO: remove when stable |
|
|
1556 | CFClient::fatal $_[0]; |
|
|
1557 | }; |
|
|
1558 | |
|
|
1559 | CFClient::read_cfg "$Crossfire::VARDIR/cfplusrc"; |
2190 | CFPlus::read_cfg "$Crossfire::VARDIR/cfplusrc"; |
|
|
2191 | CFPlus::DB::Server::run; |
|
|
2192 | |
1560 | CFClient::UI::set_layout ($::CFG->{layout}); |
2193 | CFPlus::UI::set_layout ($::CFG->{layout}); |
1561 | |
2194 | |
1562 | my %DEF_CFG = ( |
2195 | my %DEF_CFG = ( |
1563 | sdl_mode => 0, |
2196 | sdl_mode => 0, |
1564 | width => 640, |
|
|
1565 | height => 480, |
|
|
1566 | fullscreen => 0, |
2197 | fullscreen => 0, |
1567 | fast => 0, |
2198 | fast => 0, |
|
|
2199 | opengl11 => 0, |
1568 | map_scale => 1, |
2200 | map_scale => 1, |
1569 | fow_enable => 1, |
2201 | fow_enable => 1, |
1570 | fow_intensity => 0.45, |
2202 | fow_intensity => 0, |
1571 | fow_smooth => 0, |
2203 | map_smoothing => 1, |
1572 | gui_fontsize => 1, |
2204 | gui_fontsize => 1, |
1573 | log_fontsize => 1, |
2205 | log_fontsize => 0.7, |
1574 | gauge_fontsize=> 1, |
2206 | gauge_fontsize => 1, |
1575 | gauge_size => 0.35, |
2207 | gauge_size => 0.35, |
1576 | stat_fontsize => 1, |
2208 | stat_fontsize => 0.7, |
1577 | mapsize => 100, |
2209 | mapsize => 100, |
1578 | host => "crossfire.schmorp.de", |
|
|
1579 | say_command => 'say', |
|
|
1580 | audio_enable => 1, |
2210 | audio_enable => 1, |
|
|
2211 | effects_enable => 1, |
|
|
2212 | effects_volume => 1, |
1581 | bgm_enable => 1, |
2213 | bgm_enable => 1, |
1582 | bgm_volume => 0.25, |
2214 | bgm_volume => 0.5, |
1583 | face_prefetch => 0, |
|
|
1584 | output_sync => 1, |
2215 | output_sync => 1, |
1585 | output_count => 1, |
2216 | output_count => 1, |
|
|
2217 | output_rate => "", |
|
|
2218 | pickup => 0, |
|
|
2219 | inv_sort => "mtime", |
|
|
2220 | default => "profile", # default profile |
|
|
2221 | show_tips => 1, |
|
|
2222 | logview_max_par => 1000, |
|
|
2223 | ); |
1586 | ); |
2224 | |
1587 | |
|
|
1588 | while (my ($k, $v) = each %DEF_CFG) { |
2225 | while (my ($k, $v) = each %DEF_CFG) { |
1589 | $CFG->{$k} = $v unless exists $CFG->{$k}; |
2226 | $CFG->{$k} = $v unless exists $CFG->{$k}; |
1590 | } |
2227 | } |
1591 | |
2228 | |
|
|
2229 | $CFG->{profile}{default}{host} ||= "crossfire.schmorp.de"; |
|
|
2230 | $PROFILE = $CFG->{profile}{default}; |
|
|
2231 | |
|
|
2232 | # convert old bindings (only default profile matters) |
|
|
2233 | if (my $bindings = delete $PROFILE->{bindings}) { |
|
|
2234 | while (my ($mod, $syms) = each %$bindings) { |
|
|
2235 | while (my ($sym, $cmds) = each %$syms) { |
|
|
2236 | push @{ $PROFILE->{macro} }, { |
|
|
2237 | accelkey => [$mod*1, $sym*1], |
|
|
2238 | action => $cmds, |
|
|
2239 | }; |
|
|
2240 | } |
|
|
2241 | } |
|
|
2242 | } |
|
|
2243 | |
1592 | sdl_init; |
2244 | sdl_init; |
1593 | |
2245 | |
1594 | @SDL_MODES = reverse |
2246 | @SDL_MODES = CFPlus::SDL_ListModes 8, 8; |
1595 | grep $_->[0] >= 640 && $_->[1] >= 480, |
2247 | @SDL_MODES = CFPlus::SDL_ListModes 5, 0 unless @SDL_MODES; |
1596 | CFClient::SDL_ListModes; |
|
|
1597 | |
|
|
1598 | @SDL_MODES or CFClient::fatal "Unable to find a usable video mode\n(hardware accelerated opengl fullscreen)"; |
2248 | @SDL_MODES or CFPlus::fatal "Unable to find a usable video mode\n(hardware accelerated opengl fullscreen)"; |
|
|
2249 | |
|
|
2250 | @SDL_MODES = sort { $a->[0] * $a->[1] <=> $b->[0] * $b->[1] } @SDL_MODES; |
1599 | |
2251 | |
1600 | $CFG->{sdl_mode} = 0 if $CFG->{sdl_mode} > @SDL_MODES; |
2252 | $CFG->{sdl_mode} = 0 if $CFG->{sdl_mode} > @SDL_MODES; |
1601 | |
2253 | |
1602 | { |
2254 | { |
1603 | my @fonts = map CFClient::find_rcfile "fonts/$_", qw( |
2255 | my @fonts = map CFPlus::find_rcfile "fonts/$_", qw( |
1604 | DejaVuSans.ttf |
2256 | DejaVuSans.ttf |
1605 | DejaVuSansMono.ttf |
2257 | DejaVuSansMono.ttf |
1606 | DejaVuSans-Bold.ttf |
2258 | DejaVuSans-Bold.ttf |
1607 | DejaVuSansMono-Bold.ttf |
2259 | DejaVuSansMono-Bold.ttf |
1608 | DejaVuSans-Oblique.ttf |
2260 | DejaVuSans-Oblique.ttf |
1609 | DejaVuSansMono-Oblique.ttf |
2261 | DejaVuSansMono-Oblique.ttf |
1610 | DejaVuSans-BoldOblique.ttf |
2262 | DejaVuSans-BoldOblique.ttf |
1611 | DejaVuSansMono-BoldOblique.ttf |
2263 | DejaVuSansMono-BoldOblique.ttf |
1612 | ); |
2264 | ); |
1613 | |
2265 | |
1614 | CFClient::add_font $_ for @fonts; |
2266 | CFPlus::add_font $_ for @fonts; |
1615 | |
2267 | |
1616 | CFClient::pango_init; |
2268 | CFPlus::pango_init; |
1617 | |
2269 | |
1618 | $FONT_PROP = new_from_file CFClient::Font $fonts[0]; |
2270 | $FONT_PROP = new_from_file CFPlus::Font $fonts[0]; |
1619 | $FONT_FIXED = new_from_file CFClient::Font $fonts[1]; |
2271 | $FONT_FIXED = new_from_file CFPlus::Font $fonts[1]; |
1620 | |
2272 | |
1621 | $FONT_PROP->make_default; |
2273 | $FONT_PROP->make_default; |
1622 | } |
2274 | } |
1623 | |
2275 | |
1624 | # compare mono (ft) vs. rgba (cairo) |
2276 | # compare mono (ft) vs. rgba (cairo) |
1625 | # ft - 1.8s, cairo 3s, even in alpha-only mode |
2277 | # ft - 1.8s, cairo 3s, even in alpha-only mode |
1626 | # for my $rgba (0..1) { |
2278 | # for my $rgba (0..1) { |
1627 | # my $t1 = Time::HiRes::time; |
2279 | # my $t1 = Time::HiRes::time; |
1628 | # for (1..1000) { |
2280 | # for (1..1000) { |
1629 | # my $layout = CFClient::Layout->new ($rgba); |
2281 | # my $layout = CFPlus::Layout->new ($rgba); |
1630 | # $layout->set_text ("hallo" x 100); |
2282 | # $layout->set_text ("hallo" x 100); |
1631 | # $layout->render; |
2283 | # $layout->render; |
1632 | # } |
2284 | # } |
1633 | # my $t2 = Time::HiRes::time; |
2285 | # my $t2 = Time::HiRes::time; |
1634 | # warn $t2-$t1; |
2286 | # warn $t2-$t1; |
1635 | # } |
2287 | # } |
1636 | |
2288 | |
|
|
2289 | $startup_done->(); |
|
|
2290 | |
1637 | video_init; |
2291 | video_init; |
1638 | audio_init; |
2292 | audio_init; |
1639 | } |
2293 | } |
1640 | |
2294 | |
|
|
2295 | show_tip_of_the_day if $CFG->{show_tips}; |
|
|
2296 | |
1641 | Event::loop; |
2297 | Event::loop; |
1642 | |
2298 | |
1643 | END { CFClient::SDL_Quit } |
2299 | #video_shutdown; |
|
|
2300 | #audio_shutdown; |
|
|
2301 | CFPlus::SDL_Quit; |
|
|
2302 | CFPlus::DB::Server::stop; |
1644 | |
2303 | |
1645 | =head1 NAME |
2304 | =head1 NAME |
1646 | |
2305 | |
1647 | cfplus - A Crossfire+ and Crossfire game client |
2306 | cfplus - A Crossfire TRT and Crossfire game client |
1648 | |
2307 | |
1649 | =head1 SYNOPSIS |
2308 | =head1 SYNOPSIS |
1650 | |
2309 | |
1651 | Just run it - no commandline arguments are supported. |
2310 | Just run it - no commandline arguments are supported. |
1652 | |
2311 | |
1653 | =head1 USAGE |
2312 | =head1 USAGE |
1654 | |
2313 | |
1655 | cfplus utilises OpenGL for all UI elements and the game. It is supposed to be used |
2314 | cfplus utilises OpenGL for all UI elements and the game. It is supposed to |
1656 | fullscreen and interactively. |
2315 | be used in fullscreen mode and interactively. |
1657 | |
2316 | |
1658 | =head1 DEBUGGING |
2317 | =head1 DEBUGGING |
1659 | |
2318 | |
1660 | |
2319 | |
1661 | CFPLUS_DEBUG - environment variable |
2320 | CFPLUS_DEBUG - environment variable |