ViewVC Help
View File | Revision Log | Show Annotations | Download File
/cvs/deliantra/Deliantra-Client/bin/pclient
(Generate patch)

Comparing deliantra/Deliantra-Client/bin/pclient (file contents):
Revision 1.130 by root, Tue Apr 18 01:36:13 2006 UTC vs.
Revision 1.178 by root, Tue Apr 25 08:39:18 2006 UTC

1#!/opt/bin/perl 1#!/opt/bin/perl
2 2
3use strict; 3use strict;
4use utf8; 4use utf8;
5 5
6BEGIN {
7 if (%PAR::LibCache) {
8 @INC = grep ref, @INC; # weed out all paths except pars loader refs
9
10 while (my ($filename, $zip) = each %PAR::LibCache) {
11 for ($zip->memberNames) {
12 next unless /^\/root\/(.*)/;
13 $zip->extractMember ($_, "$ENV{PAR_TEMP}/$1")
14 unless -e "$ENV{PAR_TEMP}/$1";
15 }
16 }
17
18 unshift @INC, $ENV{PAR_TEMP};
19
20 if ($^O eq "MSWin32") {
21 $ENV{GTK_RC_FILES} = "$ENV{PAR_TEMP}/share/themes/MS-Windows/gtk-2.0/gtkrc";
22 }
23 }
24}
25
26# need to do it again because that pile of garbage called PAR nukes it before main
27unshift @INC, $ENV{PAR_TEMP};
28
6use Time::HiRes 'time'; 29use Time::HiRes 'time';
7use Event; 30use Event;
8 31
9use SDL;
10use SDL::App;
11use SDL::Event;
12use SDL::Surface;
13use SDL::OpenGL;
14
15use Crossfire; 32use Crossfire;
16use Crossfire::Protocol; 33use Crossfire::Protocol;
17 34
18use Compress::LZF; 35use Compress::LZF;
19 36
20use CFClient; 37use CFClient;
21use CFClient::UI; 38use CFClient::UI;
39use CFClient::MapWidget;
40
41$Event::DIED = sub {
42 CFClient::error $_[1];
43};
44
45#$SIG{__WARN__} = sub { Carp::cluck $_[0] };#d#
22 46
23our $VERSION = '0.1'; 47our $VERSION = '0.1';
24 48
25my $MAX_FPS = 60; 49my $MAX_FPS = 60;
26my $MIN_FPS = 5; # unused as of yet 50my $MIN_FPS = 5; # unused as of yet
41our @SDL_MODES; 65our @SDL_MODES;
42our $WIDTH; 66our $WIDTH;
43our $HEIGHT; 67our $HEIGHT;
44our $FULLSCREEN; 68our $FULLSCREEN;
45our $FONTSIZE; 69our $FONTSIZE;
70
71our $FONT_PROP;
72our $FONT_FIXED;
46 73
47our $MAP; 74our $MAP;
48our $MAPWIDGET; 75our $MAPWIDGET;
49our $BUTTONBAR; 76our $BUTTONBAR;
50our $LOGVIEW; 77our $LOGVIEW;
51our $CONSOLE; 78our $CONSOLE;
52our $METASERVER; 79our $METASERVER;
53 80
81our $FLOORBOX;
54our $GAUGES; 82our $GAUGES;
83our $STATWIDS;
55 84
56our $SDL_ACTIVE; 85our $SDL_ACTIVE;
57our $SDL_EV;
58our %SDL_CB; 86our %SDL_CB;
87
88our $SDL_MIXER;
89our @SOUNDS; # event => file mapping
90our %AUDIO_CHUNKS; # audio files
59 91
60our $ALT_ENTER_MESSAGE; 92our $ALT_ENTER_MESSAGE;
61our $STATUS_LINE; 93our $STATUS_LINE;
62our $DEBUG_STATUS; 94our $DEBUG_STATUS;
63 95
100 undef $CONN; 132 undef $CONN;
101} 133}
102 134
103sub client_setup { 135sub client_setup {
104 my $dialog = new CFClient::UI::FancyFrame 136 my $dialog = new CFClient::UI::FancyFrame
137 title => "Client Setup",
105 child => (my $vbox = new CFClient::UI::VBox); 138 child => (my $vbox = new CFClient::UI::VBox);
106 $vbox->add (new CFClient::UI::Label align => 0, text => "Client Setup");
107 $vbox->add (my $table = new CFClient::UI::Table expand => 1, col_expand => [0, 1]); 139 $vbox->add (my $table = new CFClient::UI::Table expand => 1, col_expand => [0, 1]);
108 140
109 $table->add (0, 0, new CFClient::UI::Label align => 1, text => "Video Mode"); 141 $table->add (0, 0, new CFClient::UI::Label valign => 0, align => 1, text => "Video Mode");
110 $table->add (1, 0, my $hbox = new CFClient::UI::HBox); 142 $table->add (1, 0, my $hbox = new CFClient::UI::HBox);
111 143
112 $hbox->add (my $mode_slider = new CFClient::UI::Slider expand => 1, req_w => 100, range => [$CFG->{sdl_mode}, 0, scalar @SDL_MODES, 1]); 144 $hbox->add (my $mode_slider = new CFClient::UI::Slider expand => 1, req_w => 100, range => [$CFG->{sdl_mode}, 0, scalar @SDL_MODES, 1]);
113 $hbox->add (my $mode_label = new CFClient::UI::Label height => $FONTSIZE * 0.8); 145 $hbox->add (my $mode_label = new CFClient::UI::Label align => 0, valign => 0, height => 0.8, template => "9999x9999");
114 146
115 $mode_slider->connect (changed => sub { 147 $mode_slider->connect (changed => sub {
116 my ($self, $value) = @_; 148 my ($self, $value) = @_;
117 149
118 $CFG->{sdl_mode} = $self->{range}[0] = $value = int $value; 150 $CFG->{sdl_mode} = $self->{range}[0] = $value = int $value;
119 $mode_label->set_text (sprintf "%dx%d", @{$SDL_MODES[$value]}); 151 $mode_label->set_text (sprintf "%dx%d", @{$SDL_MODES[$value]});
120 }); 152 });
121 $mode_slider->emit (changed => $mode_slider->{range}[0]); 153 $mode_slider->emit (changed => $mode_slider->{range}[0]);
122 154
155 my $row = 1;
156
123 $table->add (0, 1, new CFClient::UI::Label align => 1, text => "Fullscreen"); 157 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Fullscreen");
124 $table->add (1, 1, new CFClient::UI::CheckBox state => $CFG->{fullscreen}, connect_changed => sub { 158 $table->add (1, $row++, new CFClient::UI::CheckBox
159 state => $CFG->{fullscreen},
160 tooltip => "Bring the client into fullscreen mode",
161 connect_changed => sub {
125 my ($self, $value) = @_; 162 my ($self, $value) = @_;
126 $CFG->{fullscreen} = $value; 163 $CFG->{fullscreen} = $value;
164 }
127 }); 165 );
128 166
129 $table->add (0, 2, new CFClient::UI::Label align => 1, text => "Fast & Ugly"); 167 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Fast & Ugly");
130 $table->add (1, 2, new CFClient::UI::CheckBox state => $CFG->{fast}, connect_changed => sub { 168 $table->add (1, $row++, new CFClient::UI::CheckBox
169 state => $CFG->{fast},
170 tooltip => "Lower the visual quality considerably to speed up rendering.",
171 connect_changed => sub {
131 my ($self, $value) = @_; 172 my ($self, $value) = @_;
132 $CFG->{fast} = $value; 173 $CFG->{fast} = $value;
174 }
133 }); 175 );
134 176
135 $table->add (0, 3, new CFClient::UI::Label align => 1, text => "Fog of War"); 177 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Map Scale");
136 $table->add (1, 3, new CFClient::UI::CheckBox state => $CFG->{fow_enable}, connect_changed => sub { 178 $table->add (1, $row++, new CFClient::UI::Slider
179 range => [$CFG->{map_scale}, 0.25, 2, 0.05],
180 tooltip => "Enlarge or shrink the displayed map",
181 connect_changed => sub {
137 my ($self, $value) = @_; 182 my ($self, $value) = @_;
183 $CFG->{map_scale} = 0.05 * int $value / 0.05;
184 }
185 );
186
187 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Fog of War");
188 $table->add (1, $row++, new CFClient::UI::CheckBox
189 state => $CFG->{fow_enable},
190 tooltip => "Fog-of-War marks areas that cannot be seen by the player",
191 connect_changed => sub {
192 my ($self, $value) = @_;
138 $CFG->{fow_enable} = $value; 193 $CFG->{fow_enable} = $value;
194 }
139 }); 195 );
140 196
141 $table->add (0, 4, new CFClient::UI::Label align => 1, text => "FoW Intensity"); 197 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "FoW Intensity");
142 $table->add (1, 4, new CFClient::UI::Slider range => [$CFG->{fow_intensity}, 0, 1 + 0.001, 0.001], connect_changed => sub { 198 $table->add (1, $row++, new CFClient::UI::Slider
199 range => [$CFG->{fow_intensity}, 0, 1 + 0.001, 0.001],
200 tooltip => "The higher the intensity, the lighter the Fog-of-War color",
201 connect_changed => sub {
143 my ($self, $value) = @_; 202 my ($self, $value) = @_;
144 $CFG->{fow_intensity} = $value; 203 $CFG->{fow_intensity} = $value;
204 }
145 }); 205 );
146 206
147 $table->add (0, 5, new CFClient::UI::Label align => 1, text => "FoW Smooth"); 207 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "FoW Smooth");
148 $table->add (1, 5, new CFClient::UI::CheckBox state => $CFG->{fow_smooth}, connect_changed => sub { 208 $table->add (1, $row++, new CFClient::UI::CheckBox
209 state => $CFG->{fow_smooth},
210 tooltip => "Smooth the Fog-of-War a bit to make it more realistic",
211 connect_changed => sub {
149 my ($self, $value) = @_; 212 my ($self, $value) = @_;
150 $CFG->{fow_smooth} = $value; 213 $CFG->{fow_smooth} = $value;
214 status "Fog of War smoothing requires OpenGL 1.2 or higher" if $CFClient::GL_VERSION < 1.2;
215 }
216 );
217
218 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "GUI Fontsize");
219 $table->add (1, $row++, new CFClient::UI::Slider
220 range => [$CFG->{gui_fontsize}, 0.5, 2, 0.1],
221 tooltip => "The font size used by most GUI elements",
222 connect_changed => sub {
223 $CFG->{gui_fontsize} = 0.1 * int $_[1] * 10;
224# $FONTSIZE = int $HEIGHT / 40 * $CFG->{gui_fontsize};
225 }
226 );
227
228 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Server Log Fontsize");
229 $table->add (1, $row++, new CFClient::UI::Slider
230 range => [$CFG->{log_fontsize}, 0.5, 2, 0.1],
231 tooltip => "The font size used by the server log window only",
232 connect_changed => sub {
233 $LOGVIEW->set_fontsize ($CFG->{log_fontsize} = 0.1 * int $_[1] * 10);
234 }
235 );
236
237 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Stats Fontsize");
238
239 $table->add (1, $row++, new CFClient::UI::Slider
240 range => [$CFG->{stat_fontsize}, 0.5, 2, 0.1],
241 tooltip => "The font size used by the statistics window only",
242 connect_changed => sub {
243 $CFG->{stat_fontsize} = 0.1 * int $_[1] * 10;
244 &set_stats_window_fontsize;
245 }
246 );
247
248 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Gauge size");
249 $table->add (1, $row++, new CFClient::UI::Slider
250 range => [$CFG->{gauge_size}, 0.2, 0.8, 0.02],
251 tooltip => "Adjust the size of the stats gauges at the bottom right",
252 connect_changed => sub {
253 $CFG->{gauge_size} = $_[1];
254 my $h = int $HEIGHT * $CFG->{gauge_size};
255 $GAUGES->{win}->set_size ($WIDTH, $h);
256 $GAUGES->{win}->move (0, $HEIGHT - $h);
257 }
258 );
259
260 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Gauge fontsize");
261 $table->add (1, $row++, new CFClient::UI::Slider
262 range => [$CFG->{gauge_fontsize}, 0.5, 2.0, 0.1],
263 tooltip => "Adjusts the fontsize of the gauges at the bottom right",
264 connect_changed => sub {
265 $CFG->{gauge_fontsize} = 0.1 * int $_[1] * 10;
266 &set_gauge_window_fontsize;
267 }
268 );
269
270 $table->add (1, $row++, new CFClient::UI::Button
271 expand => 1, align => 0, text => "Apply",
272 tooltip => "Apply the video settings",
273 connect_activate => sub {
274 video_shutdown ();
275 video_init ();
276 }
277 );
278
279 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Audio Enable");
280 $table->add (1, $row++, new CFClient::UI::CheckBox
281 state => $CFG->{audio_enable},
282 tooltip => "If enabled, sound effects and music will be played. If disabled, no audio will be used and the soundcard will not be opened.",
283 connect_changed => sub {
284 $CFG->{audio_enable} = $_[1];
285 }
286 );
287# $table->add (0, 9, new CFClient::UI::Label valign => 0, align => 1, text => "Effects Volume");
288# $table->add (1, 8, new CFClient::UI::Slider range => [$CFG->{effects_volume}, 0, 128, 1], connect_changed => sub {
289# $CFG->{effects_volume} = $_[1];
151 }); 290# });
152
153 $table->add (0, 6, new CFClient::UI::Label align => 1, text => "Log Fontsize"); 291 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Background Music");
154 $table->add (1, 6, new CFClient::UI::Slider range => [$CFG->{log_fontsize}, 8, 30, 1], connect_changed => sub { 292 $table->add (1, $row++, my $hbox = new CFClient::UI::HBox);
155 my ($self, $value) = @_; 293 $hbox->add (new CFClient::UI::CheckBox
156 $LOGVIEW->set_fontsize ($CFG->{log_fontsize} = int $value); 294 expand => 1, state => $CFG->{bgm_enable},
295 tooltip => "Enable background music playing",
296 connect_changed => sub {
297 $CFG->{bgm_enable} = $_[1];
298 }
157 }); 299 );
158 300 $hbox->add (new CFClient::UI::Slider
159 $table->add (1, 7, new CFClient::UI::Button expand => 1, align => 0, text => "Apply", connect_activate => sub { 301 expand => 1, range => [$CFG->{bgm_volume}, 0, 1, 0.1],
160 destroy_screen (); 302 tooltip => "The volume of the background music",
161 init_screen (); 303 connect_changed => sub {
304 $CFG->{bgm_volume} = $_[1];
305 CFClient::MixMusic::volume $_[1] * 128;
306 }
162 }); 307 );
308
309 $table->add (1, $row++, new CFClient::UI::Button
310 expand => 1, align => 0, text => "Apply",
311 tooltip => "Apply the audio settings",
312 connect_activate => sub {
313 audio_shutdown ();
314 audio_init ();
315 }
316 );
163 317
164 $dialog 318 $dialog
319}
320
321sub set_stats_window_fontsize {
322 for (values %{$STATWIDS}) {
323 $_->set_fontsize ($::CFG->{stat_fontsize});
324 }
325}
326
327sub set_gauge_window_fontsize {
328 for (map { $GAUGES->{$_} } grep { $_ ne 'win' } keys %{$GAUGES}) {
329 $_->set_fontsize ($::CFG->{gauge_fontsize});
330 }
331
332# local $GAUGES->{win}{parent};#d#
333# use PApp::Util; open D, ">:utf8", "d"; print D PApp::Util::dumpval $GAUGES->{win}; close D;
334}
335
336sub make_gauge_window {
337 my $gh = int ($HEIGHT * $CFG->{gauge_size});
338# my $gw = int ($WIDTH * $CFG->{gauge_w_size});
339
340 my $win = new CFClient::UI::Frame (
341 y => $HEIGHT - $gh, x => 0, user_w => $WIDTH, user_h => $gh
342 );
343 $win->add (my $hbox = new CFClient::UI::HBox
344 children => [
345 (new CFClient::UI::HBox expand => 1),
346 ($FLOORBOX = new CFClient::UI::VBox),
347 (my $vbox = new CFClient::UI::VBox),
348 ],
349 );
350
351 $vbox->add (new CFClient::UI::HBox
352 expand => 1,
353 children => [
354 (new CFClient::UI::Empty expand => 1),
355 (my $hb = new CFClient::UI::HBox),
356 ],
357 );
358
359 $hb->add (my $hg = new CFClient::UI::Gauge type => 'hp',
360 tooltip => "Health points - depletes when you get wounded, refills when you heal or idle");
361 $hb->add (my $mg = new CFClient::UI::Gauge type => 'mana',
362 tooltip => "Spell points - deplete when you cast wizard spells, refills when you idle");
363 $hb->add (my $gg = new CFClient::UI::Gauge type => 'grace',
364 tooltip => "Grace points - deplete when you cast priest spells, refills when you pray");
365 $hb->add (my $fg = new CFClient::UI::Gauge type => 'food',
366 tooltip => "Food - depletes with time, faster when you heal or build mana, refills when you eat healthy food");
367
368 $vbox->add (my $exp = new CFClient::UI::Label valign => 0, align => 1, can_hover => 1, can_events => 1,
369 tooltip => "Experience points and level - increases when you kill monsters or successfully use skills");
370 $vbox->add (my $rng = new CFClient::UI::Label valign => 0, align => 1, can_hover => 1, can_events => 1,
371 tooltip => "Ranged attack - how you attack when you press shift-cursor (spell, skill, weapon etc.)");
372
373 $GAUGES = {
374 exp => $exp, win => $win, range => $rng,
375 food => $fg, mana => $mg, hp => $hg, grace => $gg
376 };
377
378 &set_gauge_window_fontsize;
379
380 $win
381}
382
383sub make_stats_window {
384 my $tgw = new CFClient::UI::FancyFrame (x => $WIDTH * 2/5, y => 0, title => "Stats");
385
386 $tgw->add (my $vb = new CFClient::UI::VBox);
387 $vb->add ($STATWIDS->{title} = new CFClient::UI::Label valign => 0, align => -1, text => "Title:", expand => 1);
388 $vb->add ($STATWIDS->{map} = new CFClient::UI::Label valign => 0, align => -1, text => "Map:", expand => 1);
389
390 $vb->add (my $hb = new CFClient::UI::HBox expand => 1);
391
392 $hb->add (my $tbl = new CFClient::UI::Table expand => 1);
393
394 my $black = [0, 0, 0];
395
396 $tbl->add (0, 0, $STATWIDS->{st_str} = new CFClient::UI::Label valign => 0, align => +1, template => "30");
397 $tbl->add (0, 1, $STATWIDS->{st_dex} = new CFClient::UI::Label valign => 0, align => +1, template => "30");
398 $tbl->add (0, 2, $STATWIDS->{st_con} = new CFClient::UI::Label valign => 0, align => +1, template => "30");
399 $tbl->add (0, 3, $STATWIDS->{st_int} = new CFClient::UI::Label valign => 0, align => +1, template => "30");
400 $tbl->add (0, 4, $STATWIDS->{st_wis} = new CFClient::UI::Label valign => 0, align => +1, template => "30");
401 $tbl->add (0, 5, $STATWIDS->{st_pow} = new CFClient::UI::Label valign => 0, align => +1, template => "30");
402 $tbl->add (0, 6, $STATWIDS->{st_cha} = new CFClient::UI::Label valign => 0, align => +1, template => "30");
403
404 $tbl->add (1, 0, $STATWIDS->{st_str_lbl} = new CFClient::UI::Label fg => $black, valign => 0, align => -1, text => "Str");
405 $tbl->add (1, 1, $STATWIDS->{st_dex_lbl} = new CFClient::UI::Label fg => $black, valign => 0, align => -1, text => "Dex");
406 $tbl->add (1, 2, $STATWIDS->{st_con_lbl} = new CFClient::UI::Label fg => $black, valign => 0, align => -1, text => "Con");
407 $tbl->add (1, 3, $STATWIDS->{st_int_lbl} = new CFClient::UI::Label fg => $black, valign => 0, align => -1, text => "Int");
408 $tbl->add (1, 4, $STATWIDS->{st_wis_lbl} = new CFClient::UI::Label fg => $black, valign => 0, align => -1, text => "Wis");
409 $tbl->add (1, 5, $STATWIDS->{st_pow_lbl} = new CFClient::UI::Label fg => $black, valign => 0, align => -1, text => "Pow");
410 $tbl->add (1, 6, $STATWIDS->{st_cha_lbl} = new CFClient::UI::Label fg => $black, valign => 0, align => -1, text => "Cha");
411
412 $tbl->add (2, 0, $STATWIDS->{st_wc} = new CFClient::UI::Label valign => 0, align => +1, template => "-120");
413 $tbl->add (2, 1, $STATWIDS->{st_ac} = new CFClient::UI::Label valign => 0, align => +1, template => "-120");
414 $tbl->add (2, 2, $STATWIDS->{st_dam} = new CFClient::UI::Label valign => 0, align => +1, template => "120");
415 $tbl->add (2, 3, $STATWIDS->{st_arm} = new CFClient::UI::Label valign => 0, align => +1, template => "120");
416 $tbl->add (2, 4, $STATWIDS->{st_spd} = new CFClient::UI::Label valign => 0, align => +1, template => "10.54");
417 $tbl->add (2, 5, $STATWIDS->{st_wspd} = new CFClient::UI::Label valign => 0, align => +1, template => "9");
418
419 $tbl->add (3, 0, $STATWIDS->{st_wc_lbl} = new CFClient::UI::Label fg => $black, valign => 0, align => -1, text => "Wc");
420 $tbl->add (3, 1, $STATWIDS->{st_ac_lbl} = new CFClient::UI::Label fg => $black, valign => 0, align => -1, text => "Ac");
421 $tbl->add (3, 2, $STATWIDS->{st_dam_lbl} = new CFClient::UI::Label fg => $black, valign => 0, align => -1, text => "Dam");
422 $tbl->add (3, 3, $STATWIDS->{st_arm_lbl} = new CFClient::UI::Label fg => $black, valign => 0, align => -1, text => "Arm");
423 $tbl->add (3, 4, $STATWIDS->{st_spd_lbl} = new CFClient::UI::Label fg => $black, valign => 0, align => -1, text => "Sp");
424 $tbl->add (3, 5, $STATWIDS->{st_wspd_lbl} = new CFClient::UI::Label fg => $black, valign => 0, align => -1, text => "WSp");
425
426 $hb->add (my $tbl2 = new CFClient::UI::Table expand => 1);
427
428 my $row = 0;
429 my $col = 0;
430
431 my %resist_names = (
432 slow => "Slow",
433 holyw => "Holy Word",
434 conf => "Confusion",
435 fire => "Fire",
436 depl => "Depletion",
437 magic => "Magic",
438 drain => "Draining",
439 acid => "Acid",
440 pois => "Poison",
441 para => "Paralysation",
442 deat => "Death",
443 phys => "Physical",
444 blind => "Blind",
445 fear => "Fear",
446 tund => "Turn undead",
447 elec => "Electricity",
448 cold => "Cold",
449 ghit => "Ghost hit",
450 );
451 for (qw/slow holyw conf fire depl magic
452 drain acid pois para deat phys
453 blind fear tund elec cold ghit/)
454 {
455 $tbl2->add ($col, $row,
456 $STATWIDS->{"res_$_"} =
457 new CFClient::UI::Label
458 template => "-100%",
459 align => +1,
460 valign => 0,
461 tooltip => $resist_names{$_}
462 );
463 $tbl2->add ($col + 1, $row, new CFClient::UI::Image
464 can_hover => 1,
465 can_events => 1,
466 image => "ui/resist/resist_$_.png",
467 tooltip => $resist_names{$_}
468 );
469
470 $row++;
471 if ($row % 6 == 0) {
472 $col += 2;
473 $row = 0;
474 }
475 }
476
477 &set_stats_window_fontsize;
478 update_stats_window ({});
479
480 $tgw
481}
482
483sub formsep {
484 reverse join ",", grep length, split /(...)/, reverse $_[0] * 1
485}
486
487sub update_stats_window {
488 my ($stats) = @_;
489
490 # i love text protocols!!!
491 my $hp = $stats->{Crossfire::Protocol::CS_STAT_HP} * 1;
492 my $hp_m = $stats->{Crossfire::Protocol::CS_STAT_MAXHP} * 1;
493 my $sp = $stats->{Crossfire::Protocol::CS_STAT_SP} * 1;
494 my $sp_m = $stats->{Crossfire::Protocol::CS_STAT_MAXSP} * 1;
495 my $fo = $stats->{Crossfire::Protocol::CS_STAT_FOOD} * 1;
496 my $fo_m = 999;
497 my $gr = $stats->{Crossfire::Protocol::CS_STAT_GRACE} * 1;
498 my $gr_m = $stats->{Crossfire::Protocol::CS_STAT_MAXGRACE} * 1;
499
500 $GAUGES->{hp} ->set_value ($hp, $hp_m);
501 $GAUGES->{mana} ->set_value ($sp, $sp_m);
502 $GAUGES->{food} ->set_value ($fo, $fo_m);
503 $GAUGES->{grace} ->set_value ($gr, $gr_m);
504 $GAUGES->{exp} ->set_text ("Exp: " . (formsep $stats->{Crossfire::Protocol::CS_STAT_EXP64})
505 . " (lvl " . ($stats->{Crossfire::Protocol::CS_STAT_LEVEL} * 1) . ")");
506 my $rng = $stats->{Crossfire::Protocol::CS_STAT_RANGE};
507 $rng =~ s/^Range: //; # thank you so much dear server
508 $GAUGES->{range} ->set_text ("Rng: " . $rng);
509 my $title = $stats->{Crossfire::Protocol::CS_STAT_TITLE};
510 $title =~ s/^Player: //;
511 $STATWIDS->{title} ->set_text ("Title: " . $title);
512
513 $STATWIDS->{st_str} ->set_text (sprintf "%d", $stats->{5});
514 $STATWIDS->{st_dex} ->set_text (sprintf "%d", $stats->{8});
515 $STATWIDS->{st_con} ->set_text (sprintf "%d", $stats->{9});
516 $STATWIDS->{st_int} ->set_text (sprintf "%d", $stats->{6});
517 $STATWIDS->{st_wis} ->set_text (sprintf "%d", $stats->{7});
518 $STATWIDS->{st_pow} ->set_text (sprintf "%d", $stats->{22});
519 $STATWIDS->{st_cha} ->set_text (sprintf "%d", $stats->{10});
520 $STATWIDS->{st_wc} ->set_text (sprintf "%d", $stats->{13});
521 $STATWIDS->{st_ac} ->set_text (sprintf "%d", $stats->{14});
522 $STATWIDS->{st_dam} ->set_text (sprintf "%d", $stats->{15});
523 $STATWIDS->{st_arm} ->set_text (sprintf "%d", $stats->{16});
524 $STATWIDS->{st_spd} ->set_text (sprintf "%.1f", $stats->{Crossfire::Protocol::CS_STAT_SPEED});
525 $STATWIDS->{st_wspd}->set_text (sprintf "%.1f", $stats->{Crossfire::Protocol::CS_STAT_WEAP_SP});
526
527 my %tbl = (
528 phys => 100,
529 magic => 101,
530 fire => 102,
531 elec => 103,
532 cold => 104,
533 conf => 105,
534 acid => 106,
535 drain => 107,
536 ghit => 108,
537 pois => 109,
538 slow => 110,
539 para => 111,
540 tund => 112,
541 fear => 113,
542 depl => 113,
543 deat => 115,
544 holyw => 116,
545 blind => 117
546 );
547
548 for (keys %tbl) {
549 $STATWIDS->{"res_$_"}->set_text (sprintf "%d%", $stats->{$tbl{$_}});
550 }
551
165} 552}
166 553
167sub metaserver_dialog { 554sub metaserver_dialog {
168 my $dialog = new CFClient::UI::FancyFrame 555 my $dialog = new CFClient::UI::FancyFrame
556 title => "Metaserver",
169 child => (my $vbox = new CFClient::UI::VBox); 557 child => (my $vbox = new CFClient::UI::VBox);
170 558
171 $vbox->add ($dialog->{table} = new CFClient::UI::Table); 559 $vbox->add ($dialog->{table} = new CFClient::UI::Table);
172 560
173 $dialog 561 $dialog
174} 562}
175 563
176sub update_metaserver { 564sub update_metaserver {
177 my ($HOST) = @_; 565 my ($HOST) = @_;
178 566
179 status "fetching metaserver list..."; 567 my $table = $METASERVER->{table};
568 $table->clear;
569 $table->add (0, 0, my $label = new CFClient::UI::Label max_w => $WIDTH * 0.8, text => "fetching metaserver list...");
180 570
181 my $buf; 571 my $buf;
182 572
183 my $fh = new IO::Socket::INET PeerHost => $META_SERVER, Blocking => 0; 573 my $fh = new IO::Socket::INET PeerHost => $META_SERVER, Blocking => 0;
574
575 unless ($fh) {
576 $label->set_text ("unable to contact metaserver: $!");
577 return;
578 }
184 579
185 Event->io (fd => $fh, poll => 'r', cb => sub { 580 Event->io (fd => $fh, poll => 'r', cb => sub {
186 my $res = sysread $fh, $buf, 8192, length $buf; 581 my $res = sysread $fh, $buf, 8192, length $buf;
187 582
188 if (!defined $res) { 583 if (!defined $res) {
189 $_[0]->w->cancel; 584 $_[0]->w->cancel;
190 status "metaserver: $!"; 585 $label->set_text ("error while retrieving server list: $!");
191 } elsif ($res == 0) { 586 } elsif ($res == 0) {
192 $_[0]->w->cancel; 587 $_[0]->w->cancel;
193 status "server list retrieved"; 588 status "server list retrieved";
194 589
195 my $table = $METASERVER->{table}; 590 utf8::decode $buf if utf8::valid $buf;
196 591
197 $table->clear; 592 $table->clear;
198 593
199 my @col = qw(Use #Users Host Uptime Version Description); 594 my @col = qw(Use #Users Host Uptime Version Description);
200 $table->add ($_, 0, new CFClient::UI::Label align => 0, fg => [1, 1, 0], text => $col[$_]) 595 $table->add ($_, 0, new CFClient::UI::Label align => 0, fg => [1, 1, 0], text => $col[$_])
224 $m = [$users, $host, $uptime, $version, $desc]; 619 $m = [$users, $host, $uptime, $version, $desc];
225 620
226 $y++; 621 $y++;
227 622
228 $table->add (0, $y, new CFClient::UI::VBox children => [ 623 $table->add (0, $y, new CFClient::UI::VBox children => [
229 (new CFClient::UI::Button text => " ", connect_activate => sub { 624 (new CFClient::UI::Button text => "Use", connect_activate => sub {
230 $HOST->set_text ($CFG->{host} = $host); 625 $HOST->set_text ($CFG->{host} = $host);
231 }), 626 }),
232 (new CFClient::UI::Empty expand => 1), 627 (new CFClient::UI::Empty expand => 1),
233 ]); 628 ]);
234 629
235 $table->add ($_ + 1, $y, new CFClient::UI::Label align => $align[$_], text => $m->[$_], fontsize => $FONTSIZE * 0.8) 630 $table->add ($_ + 1, $y, new CFClient::UI::Label align => $align[$_], text => $m->[$_], fontsize => 0.8)
236 for 0 .. $#$m; 631 for 0 .. $#$m;
237 } 632 }
238 } 633 }
239 }); 634 });
240} 635}
241 636
242sub server_setup { 637sub server_setup {
243 my $dialog = new CFClient::UI::FancyFrame 638 my $dialog = new CFClient::UI::FancyFrame
639 title => "Server Setup",
244 child => (my $vbox = new CFClient::UI::VBox); 640 child => (my $vbox = new CFClient::UI::VBox);
245 641
246 $vbox->add (new CFClient::UI::Label align => 0, text => "Server Setup");
247 $vbox->add (my $table = new CFClient::UI::Table expand => 1, col_expand => [0, 1]); 642 $vbox->add (my $table = new CFClient::UI::Table expand => 1, col_expand => [0, 1]);
248 $table->add (0, 2, new CFClient::UI::Label align => 1, text => "Host:Port"); 643 $table->add (0, 2, new CFClient::UI::Label valign => 0, align => 1, text => "Host:Port");
249 644
250 { 645 {
251 $table->add (1, 2, my $vbox = new CFClient::UI::VBox); 646 $table->add (1, 2, my $vbox = new CFClient::UI::VBox);
252 647
253 $vbox->add (my $HOST = new CFClient::UI::Entry text => $CFG->{host}, connect_changed => sub { 648 $vbox->add (
649 my $HOST = new CFClient::UI::Entry
650 expand => 1,
651 text => $CFG->{host},
652 tooltip => "The hostname or ip address of the Crossfire(+) server to connect to",
653 connect_changed => sub {
654 my ($self, $value) = @_;
655 $CFG->{host} = $value;
656 }
657 );
658
659 $METASERVER = metaserver_dialog;
660
661 $vbox->add (new CFClient::UI::Flopper
662 expand => 1,
663 text => "Metaserver",
664 other => $METASERVER,
665 tooltip => "Show a list of avaible crossfire servers",
666 connect_open => sub {
667 update_metaserver $HOST;
668 }
669 );
670 }
671
672 $table->add (0, 4, new CFClient::UI::Label valign => 0, align => 1, text => "Username");
673 $table->add (1, 4, new CFClient::UI::Entry
674 text => $CFG->{user},
675 tooltip => "The name of your character on the server",
676 connect_changed => sub {
254 my ($self, $value) = @_; 677 my ($self, $value) = @_;
255 $CFG->{host} = $value; 678 $CFG->{user} = $value;
256 }); 679 }
680 );
257 681
258 $METASERVER = metaserver_dialog;
259
260 $vbox->add (new CFClient::UI::Flopper text => "Metaserver", other => $METASERVER, connect_open => sub {
261 update_metaserver $HOST;
262 });
263 }
264
265 $table->add (0, 4, new CFClient::UI::Label align => 1, text => "Username"); 682 $table->add (0, 5, new CFClient::UI::Label valign => 0, align => 1, text => "Password");
266 $table->add (1, 4, new CFClient::UI::Entry text => $CFG->{user}, connect_changed => sub { 683 $table->add (1, 5, new CFClient::UI::Entry
684 text => $CFG->{password},
685 hidden => 1,
686 tooltip => "The password for your character",
687 connect_changed => sub {
267 my ($self, $value) = @_; 688 my ($self, $value) = @_;
268 $CFG->{user} = $value;
269 });
270
271 $table->add (0, 5, new CFClient::UI::Label align => 1, text => "Password");
272 $table->add (1, 5, new CFClient::UI::Entry text => $CFG->{password}, hidden => 1, connect_changed => sub {
273 my ($self, $value) = @_;
274 $CFG->{password} = $value; 689 $CFG->{password} = $value;
690 }
275 }); 691 );
276 692
277 $table->add (0, 6, new CFClient::UI::Label align => 1, text => "Def. say cmd"); 693 $table->add (0, 6, new CFClient::UI::Label valign => 0, align => 1, text => "Def. say cmd");
278 $table->add (1, 6, my $saycmd = new CFClient::UI::Entry text => $CFG->{say_command}, connect_changed => sub { 694 $table->add (1, 6, my $saycmd = new CFClient::UI::Entry
695 text => $CFG->{say_command},
696 tooltip => "This is the command that will be used if you write a line in the message window entry. "
697 ."Usually you want to enter something like 'say' or 'shout' or 'gsay' here. "
698 ."But you could also set it to 'tell &lt;playername&gt;' to only chat with that user.",
699 connect_changed => sub {
279 my ($self, $value) = @_; 700 my ($self, $value) = @_;
280 $CFG->{say_command} = $value; 701 $CFG->{say_command} = $value;
702 }
281 }); 703 );
282 704
283 $table->add (0, 7, new CFClient::UI::Label align => 1, text => "Map Size"); 705 $table->add (0, 7, new CFClient::UI::Label valign => 0, align => 1, text => "Map Size");
284 $table->add (1, 7, new CFClient::UI::Slider 706 $table->add (1, 7, new CFClient::UI::Slider
285 req_w => 100, 707 req_w => 100,
286 range => [$CFG->{mapsize}, 10, 100 + 1, 1], 708 range => [$CFG->{mapsize}, 10, 100 + 1, 1],
709 tooltip => "This is the size of the portion of the map update the server sends you. "
710 ."If you set this to a high value you will be able to see further for example.",
287 connect_changed => sub { 711 connect_changed => sub {
288 my ($self, $value) = @_; 712 my ($self, $value) = @_;
289 713
290 $CFG->{mapsize} = $self->{range}[0] = $value = int $value; 714 $CFG->{mapsize} = $self->{range}[0] = $value = int $value;
291 }, 715 },
298 $dialog 722 $dialog
299} 723}
300 724
301sub message_window { 725sub message_window {
302 my $window = new CFClient::UI::FancyFrame 726 my $window = new CFClient::UI::FancyFrame
727 title => "Messages",
303 border_bg => [1, 1, 1, 0.5], 728 border_bg => [1, 1, 1, 0.5],
304 bg => [0.3, 0.3, 0.3, 0.8], 729 bg => [0.3, 0.3, 0.3, 0.8],
305 user_w => int $::WIDTH / 3, 730 user_w => int $::WIDTH / 3,
306 user_h => int $::HEIGHT / 5, 731 user_h => int $::HEIGHT / 5,
307 child => (my $vbox = new CFClient::UI::VBox); 732 child => (my $vbox = new CFClient::UI::VBox);
308 733
309 $vbox->add ($LOGVIEW = new CFClient::UI::TextView 734 $vbox->add ($LOGVIEW = new CFClient::UI::TextView
310 expand => 1, 735 expand => 1,
736 font => $FONT_FIXED,
311 fontsize => $::CFG->{log_fontsize}, 737 fontsize => $::CFG->{log_fontsize},
312 ); 738 );
313 739
314 $vbox->add (my $input = new CFClient::UI::Entry 740 $vbox->add (my $input = new CFClient::UI::Entry
315 connect_focus_in => sub { 741 connect_focus_in => sub {
349 775
350 $window 776 $window
351} 777}
352 778
353sub sdl_init { 779sub sdl_init {
354 SDL::Init SDL_INIT_AUDIO | SDL_INIT_VIDEO | SDL_INIT_NOPARACHUTE 780 CFClient::SDL_Init
355 #SDL::Init SDL_INIT_AUDIO | SDL_INIT_VIDEO
356 and die "SDL::Init failed!\n"; 781 and die "SDL::Init failed!\n";
357} 782}
358 783
359sub init_screen { 784sub video_init {
360 sdl_init; 785 sdl_init;
361 786
362 ($WIDTH, $HEIGHT) = @{ $SDL_MODES[$CFG->{sdl_mode}] }; 787 ($WIDTH, $HEIGHT) = @{ $SDL_MODES[$CFG->{sdl_mode}] };
363 $FULLSCREEN = $CFG->{fullscreen}; 788 $FULLSCREEN = $CFG->{fullscreen};
364 $FAST = $CFG->{fast}; 789 $FAST = $CFG->{fast};
365 790
366 SDL::GLSetAttribute SDL_GL_RED_SIZE, 5; 791 CFClient::SDL_SetVideoMode $WIDTH, $HEIGHT, $FULLSCREEN
367 SDL::GLSetAttribute SDL_GL_GREEN_SIZE, 5;
368 SDL::GLSetAttribute SDL_GL_BLUE_SIZE, 5;
369 SDL::GLSetAttribute SDL_GL_ALPHA_SIZE, 1;
370
371 SDL::GLSetAttribute SDL_GL_ACCUM_RED_SIZE, 0;
372 SDL::GLSetAttribute SDL_GL_ACCUM_GREEN_SIZE, 0;
373 SDL::GLSetAttribute SDL_GL_ACCUM_BLUE_SIZE, 0;
374 SDL::GLSetAttribute SDL_GL_ACCUM_ALPHA_SIZE, 0;
375
376 SDL::GLSetAttribute SDL_GL_DOUBLEBUFFER, 1;
377 SDL::GLSetAttribute SDL_GL_BUFFER_SIZE, 15;
378 SDL::GLSetAttribute SDL_GL_DEPTH_SIZE, 0;
379
380 SDL::SetVideoMode $WIDTH, $HEIGHT, 0,
381 SDL_HWSURFACE | SDL_ANYFORMAT | SDL_OPENGL | SDL_DOUBLEBUF
382 | ($FULLSCREEN ? SDL_FULLSCREEN : 0)
383 or die "SDL::SetVideoMode failed!\n"; 792 or die "SDL_SetVideoMode failed!\n";
384
385 SDL::WMSetCaption "Crossfire+ Client", "Crossfire+";
386
387 $SDL_EV = new SDL::Event;
388 $SDL_EV->set_unicode (1);
389 793
390 $SDL_ACTIVE = 1; 794 $SDL_ACTIVE = 1;
391 795
392 $LAST_REFRESH = time - 0.01; 796 $LAST_REFRESH = time - 0.01;
393 797
394 CFClient::gl_init; 798 CFClient::gl_init;
395 799
396 $FONTSIZE = int $HEIGHT / 40; 800 $FONTSIZE = int $HEIGHT / 40 * $CFG->{gui_fontsize};
397 801
398 ############################################################################# 802 #############################################################################
399 803
400 $DEBUG_STATUS = new CFClient::UI::Label padding => 0, z => 100; 804 $DEBUG_STATUS = new CFClient::UI::Label padding => 0, z => 100;
401 $CFClient::UI::ROOT->add ($DEBUG_STATUS); 805 $DEBUG_STATUS->show;
402 806
403 $STATUS_LINE = new CFClient::UI::Label 807 $STATUS_LINE = new CFClient::UI::Label
404 padding => 0, 808 padding => 0,
405 y => $HEIGHT * 44 / 45 - $FONTSIZE; 809 y => $HEIGHT - $FONTSIZE * 1.8;
406 $CFClient::UI::ROOT->add ($STATUS_LINE); 810 $STATUS_LINE->show;
407 811
408 $ALT_ENTER_MESSAGE = new CFClient::UI::Label 812 $ALT_ENTER_MESSAGE = new CFClient::UI::Label
409 padding => 0, 813 padding => 0,
410 y => $HEIGHT * 44 / 45,
411 fontsize => $HEIGHT / 45, 814 fontsize => 0.8,
412 markup => "Use <b>Alt-Enter</b> to toggle fullscreen mode"; 815 markup => "Use <b>Alt-Enter</b> to toggle fullscreen mode";
413 $CFClient::UI::ROOT->add ($ALT_ENTER_MESSAGE); 816 $ALT_ENTER_MESSAGE->show;
817 $ALT_ENTER_MESSAGE->move (0, $HEIGHT - $ALT_ENTER_MESSAGE->{h});
414 818
415 $CFClient::UI::ROOT->add ($MAPWIDGET = new CFClient::UI::MapWidget); 819 $CFClient::UI::ROOT->add ($MAPWIDGET = new CFClient::MapWidget);
416 $MAPWIDGET->focus_in; 820 $MAPWIDGET->focus_in;
417 $MAPWIDGET->connect (activate_console => sub { 821 $MAPWIDGET->connect (activate_console => sub {
418 my ($mapwidget, $preset) = @_; 822 my ($mapwidget, $preset) = @_;
419 823
420 if ($CONSOLE) { 824 if ($CONSOLE) {
430 $CFClient::UI::ROOT->add ($BUTTONBAR = new CFClient::UI::HBox); 834 $CFClient::UI::ROOT->add ($BUTTONBAR = new CFClient::UI::HBox);
431 835
432 $BUTTONBAR->add (new CFClient::UI::Flopper text => "Client Setup", other => client_setup); 836 $BUTTONBAR->add (new CFClient::UI::Flopper text => "Client Setup", other => client_setup);
433 $BUTTONBAR->add (new CFClient::UI::Flopper text => "Server Setup", other => server_setup); 837 $BUTTONBAR->add (new CFClient::UI::Flopper text => "Server Setup", other => server_setup);
434 $BUTTONBAR->add (new CFClient::UI::Flopper text => "Message Window", other => message_window); 838 $BUTTONBAR->add (new CFClient::UI::Flopper text => "Message Window", other => message_window);
839
840 $CFClient::UI::ROOT->add (make_gauge_window); # 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
841 $BUTTONBAR->add (new CFClient::UI::Flopper text => "Stats Window", other => make_stats_window);
435 842
436 $BUTTONBAR->add (new CFClient::UI::Button text => "Save Config", connect_activate => sub { 843 $BUTTONBAR->add (new CFClient::UI::Button text => "Save Config", connect_activate => sub {
437 CFClient::write_cfg "$Crossfire::VARDIR/pclientrc"; 844 CFClient::write_cfg "$Crossfire::VARDIR/pclientrc";
438 status "Configuration Saved"; 845 status "Configuration Saved";
439 }); 846 });
440 847
441 $BUTTONBAR->{children}[1]->emit ("activate"); # pop up server setup 848 $BUTTONBAR->{children}[1]->emit ("activate"); # pop up server setup
442
443 my $tgw = new CFClient::UI::FancyFrame (x => $WIDTH - 300, y => 0);
444 $tgw->add (my $hbox = new CFClient::UI::HBox ());
445
446 $hbox->add (my $hg = new CFClient::UI::VGauge (gauge => 'hp'));
447 $hbox->add (my $mg = new CFClient::UI::VGauge (gauge => 'mana'));
448 $hbox->add (my $gg = new CFClient::UI::VGauge (gauge => 'grace'));
449 $hbox->add (my $fg = new CFClient::UI::VGauge (gauge => 'food'));
450
451 $GAUGES = { food => $fg, mana => $mg, hp => $hg, grace => $gg };
452 $CFClient::UI::ROOT->add ($tgw);
453} 849}
454 850
455sub destroy_screen { 851sub video_shutdown {
456 $CFClient::UI::ROOT->{children} = []; 852 $CFClient::UI::ROOT->{children} = [];
853 undef $CFClient::UI::GRAB;
854 undef $CFClient::UI::HOVER;
457 undef $SDL_ACTIVE; 855 undef $SDL_ACTIVE;
856}
857
858my @bgmusic = qw(game1.ogg game2.ogg game3.ogg game5.ogg game6.ogg ross1.ogg ross2.ogg ross3.ogg ross4.ogg ross5.ogg); #d#
859my $bgmusic;#TODO#hack#d#
860
861sub audio_music_finished {
862 return unless $CFG->{bgm_enable};
863
864 # TODO: hack, do play loop and mood music
865 $bgmusic = new_from_file CFClient::MixMusic CFClient::find_rcfile "music/$bgmusic[0]";
866 $bgmusic->play (0);
867
868 push @bgmusic, shift @bgmusic;
869}
870
871sub audio_init {
872 if ($CFG->{audio_enable}) {
873 if (open my $fh, "<:utf8", CFClient::find_rcfile "sounds/config") {
874 $SDL_MIXER = !CFClient::Mix_OpenAudio;
875 CFClient::Mix_AllocateChannels 8;
876 CFClient::MixMusic::volume $CFG->{bgm_volume} * 128;
877
878 audio_music_finished;
879
880 while (<$fh>) {
881 next if /^\s*#/;
882 next if /^\s*$/;
883
884 my ($file, $volume, $event) = split /\s+/, $_, 3;
885
886 push @SOUNDS, "$volume,$file";
887
888 $AUDIO_CHUNKS{"$volume,$file"} ||= do {
889 my $chunk = new_from_file CFClient::MixChunk CFClient::find_rcfile "sounds/$file";
890 $chunk->volume ($volume * 128 / 100);
891 $chunk
892 };
893 }
894 } else {
895 status "unable to open sound config: $!";
896 }
897 }
898}
899
900sub audio_shutdown {
901 CFClient::Mix_CloseAudio if $SDL_MIXER;
458 undef $SDL_EV; 902 undef $SDL_MIXER;
459 SDL::Quit; 903 @SOUNDS = ();
904 %AUDIO_CHUNKS = ();
460} 905}
461 906
462my %animate_object; 907my %animate_object;
463my $animate_timer; 908my $animate_timer;
464 909
474 $want_refresh = 0; 919 $want_refresh = 0;
475 $can_refresh = 0; 920 $can_refresh = 0;
476 921
477 $CFClient::UI::ROOT->draw; 922 $CFClient::UI::ROOT->draw;
478 923
479 SDL::GLSwapBuffers; 924 CFClient::SDL_GL_SwapBuffers;
480 925
481 $LAST_REFRESH = $NOW; 926 $LAST_REFRESH = $NOW;
482} 927}
483 928
484my $refresh_watcher = Event->timer (after => 0, hard => 1, interval => 1 / $MAX_FPS, cb => sub { 929my $refresh_watcher = Event->timer (after => 0, hard => 1, interval => 1 / $MAX_FPS, cb => sub {
485 $NOW = time; 930 $NOW = time;
486 931
487 ($SDL_CB{$SDL_EV->type} || sub { warn "unhandled event ", $SDL_EV->type })->() 932 ($SDL_CB{$_->{type}} || sub { warn "unhandled event $_->{type}" })->($_)
488 while $SDL_EV->poll; 933 for CFClient::SDL_PollEvent;
489 934
490 if (%animate_object) { 935 if (%animate_object) {
491 $_->animate ($LAST_REFRESH - $NOW) for values %animate_object; 936 $_->animate ($LAST_REFRESH - $NOW) for values %animate_object;
492 $want_refresh++; 937 $want_refresh++;
493 } 938 }
516@conn::ISA = Crossfire::Protocol::; 961@conn::ISA = Crossfire::Protocol::;
517 962
518sub conn::stats_update { 963sub conn::stats_update {
519 my ($self, $stats) = @_; 964 my ($self, $stats) = @_;
520 965
521 # i love text protocols!!! 966 update_stats_window ($stats);
522 # FIXME: the stats are somehow weird
523 my $hp = $stats->{1};
524 my $hp_m = $stats->{2};
525 my $sp = $stats->{3};
526 my $sp_m = $stats->{4};
527 my $fo = $stats->{18};
528 my $fo_m = 1000;
529 my $gr = $stats->{23};
530 my $gr_m = $stats->{24};
531
532 #d# warn "DATA $hp $hp_m $sp $sp_m $fo $fo_m $gr $gr_m\n";
533 $GAUGES->{hp}->set_value ($hp, $hp_m);
534 $GAUGES->{mana}->set_value ($sp, $sp_m);
535 $GAUGES->{food}->set_value ($fo, $fo_m);
536 $GAUGES->{grace}->set_value ($gr, $gr_m);
537} 967}
538 968
539sub conn::user_send { 969sub conn::user_send {
540 my ($self, $command) = @_; 970 my ($self, $command) = @_;
541 971
566 996
567 my ($hash, $x, $y, $w, $h) = @$map_info; 997 my ($hash, $x, $y, $w, $h) = @$map_info;
568 998
569 my $data = $MAP->get_rect ($x, $y, $w, $h); 999 my $data = $MAP->get_rect ($x, $y, $w, $h);
570 $MAPCACHE->put ($hash => Compress::LZF::compress $data); 1000 $MAPCACHE->put ($hash => Compress::LZF::compress $data);
571
572 warn sprintf "SAVEmap[%s] length %d\n", $hash, length $data;#d# 1001 #warn sprintf "SAVEmap[%s] length %d\n", $hash, length $data;#d#
573
574} 1002}
575 1003
576sub conn::map_clear { 1004sub conn::map_clear {
577 my ($self) = @_; 1005 my ($self) = @_;
578 1006
579 $self->flush_map; 1007 $self->flush_map;
580 delete $self->{neigh}; 1008 delete $self->{neigh_map};
581 1009
582 $MAP->clear; 1010 $MAP->clear;
583} 1011}
584 1012
585 1013
586sub conn::load_map($$$) { 1014sub conn::load_map($$$) {
587 my ($self, $hash, $x, $y) = @_; 1015 my ($self, $hash, $x, $y) = @_;
588 1016
589 if (defined (my $data = $MAPCACHE->get ($hash))) { 1017 if (defined (my $data = $MAPCACHE->get ($hash))) {
590 $data = Compress::LZF::decompress $data; 1018 $data = Compress::LZF::decompress $data;
591 warn sprintf "LOADmap[%s,%d,%d] length %d\n", $hash, $x, $y, length $data;#d# 1019 #warn sprintf "LOADmap[%s,%d,%d] length %d\n", $hash, $x, $y, length $data;#d#
592 for my $id ($MAP->set_rect ($x, $y, $data)) { 1020 for my $id ($MAP->set_rect ($x, $y, $data)) {
593 my $data = $TILECACHE->get ($id) 1021 my $data = $TILECACHE->get ($id)
594 or next; 1022 or next;
595 1023
596 $self->set_texture ($id => $data); 1024 $self->set_texture ($id => $data);
597 } 1025 }
598 } 1026 }
599} 1027}
600 1028
1029# this method does a "flood fill" into every tile direction
1030# it assumes that tiles are arranged in a rectangular grid,
1031# i.e. a map is the same as the left of the right map etc.
1032# failure to comply are harmless and result in display errors
1033# at worst.
601sub conn::flood_fill { 1034sub conn::flood_fill {
602 my ($self, $path, $hash, $flags, $x0, $y0, $x1, $y1) = @_; 1035 my ($self, $gx, $gy, $path, $hash, $flags) = @_;
603 1036
604 # the server does not allow map paths > 6 1037 # the server does not allow map paths > 6
605 return if 6 <= length $path; 1038 return if 6 <= length $path;
606 1039
607 for my $tile (1..4) { 1040 my ($x0, $y0, $x1, $y1) = @{$self->{neigh_rect}};
608 next if $self->{neigh}{$hash}[$tile]; 1041
1042 for (
1043 [1, 0, -1],
1044 [2, 1, 0],
1045 [3, 0, 1],
1046 [4, -1, 0],
1047 ) {
1048 my ($tile, $dx, $dy) = @$_;
1049
1050 my $gx = $gx + $dx;
1051 my $gy = $gy + $dy;
1052
609 next unless $flags & (1 << ($tile - 1)); 1053 next unless $flags & (1 << ($tile - 1));
1054 next if $self->{neigh_grid}{$gx, $gy}++;
610 1055
611 my $neigh = $self->{neigh}{$hash} ||= []; 1056 my $neigh = $self->{neigh_map}{$hash} ||= [];
612 1057 if (my $info = $neigh->[$tile]) {
613 $self->send_mapinfo ("spatial $path$tile", sub {
614 my ($mode, $flags, $x, $y, $w, $h, $hash) = @_; 1058 my ($flags, $x, $y, $w, $h, $hash) = @$info;
615 1059
616 #warn "map<$path>_$tile=<$mode,$x,$y,$w,$h,$hash>\n";#d#
617 return if $mode ne "spatial";
618
619 $x += $MAP->ox;
620 $y += $MAP->oy;
621
622 $self->load_map ($hash, $x, $y)
623 unless $self->{neigh}{$hash}[5]++;#d#
624
625 $neigh->[$tile] = [$x, $y, $w, $h];
626
627 $self->flood_fill ("$path$tile", $hash, $flags, $x0, $y0, $x1, $y1) 1060 $self->flood_fill ($gx, $gy, "$path$tile", $hash, $flags)
628 if $x >= $x0 && $x + $w < $x1 && $y >= $y0 && $y + $h < $y1; 1061 if $x >= $x0 && $x + $w < $x1 && $y >= $y0 && $y + $h < $y1;
1062
1063 } else {
1064 $self->send_mapinfo ("spatial $path$tile", sub {
1065 my ($mode, $flags, $x, $y, $w, $h, $hash) = @_;
1066
1067 return if $mode ne "spatial";
1068
1069 $x += $MAP->ox;
1070 $y += $MAP->oy;
1071
1072 $self->load_map ($hash, $x, $y)
1073 unless $self->{neigh_map}{$hash}[5]++;#d#
1074
1075 $neigh->[$tile] = [$flags, $x, $y, $w, $h, $hash];
1076
1077 $self->flood_fill ($gx, $gy, "$path$tile", $hash, $flags)
1078 if $x >= $x0 && $x + $w < $x1 && $y >= $y0 && $y + $h < $y1;
1079 });
629 }); 1080 }
630 } 1081 }
631} 1082}
632 1083
633sub conn::map_change { 1084sub conn::map_change {
634 my ($self, $mode, $flags, $x, $y, $w, $h, $hash) = @_; 1085 my ($self, $mode, $flags, $x, $y, $w, $h, $hash) = @_;
635 1086
636 $self->flush_map; 1087 $self->flush_map;
637 1088
638 my ($ox, $oy) = ($::MAP->ox, $::MAP->oy); 1089 my ($ox, $oy) = ($::MAP->ox, $::MAP->oy);
639 1090
640 warn "$ox $oy map_info<$flags, $x, $y, $w, $h, $hash>\n";#d#
641
642 my $mapmapw = 250; 1091 my $mapmapw = 250;
643 my $mapmaph = 250; 1092 my $mapmaph = 250;
1093
1094 $self->{neigh_rect} = [
1095 $ox - $mapmapw * 0.5, $oy - $mapmapw * 0.5,
1096 $ox + $mapmapw * 0.5 + $w, $oy + $mapmapw * 0.5 + $h,
1097 ];
644 1098
1099 delete $self->{neigh_grid};
645 $self->flood_fill ("", $hash, $flags, 1100 $self->flood_fill (0, 0, "", $hash, $flags);
646 $ox - $mapmapw * 0.5, $oy - $mapmapw * 0.5,
647 $ox + $mapmapw * 0.5, $oy + $mapmapw * 0.5);
648 1101
649 $x += $ox; 1102 $x += $ox;
650 $y += $oy; 1103 $y += $oy;
651 1104
652 $self->{map_info} = [$hash, $x, $y, $w, $h]; 1105 $self->{map_info} = [$hash, $x, $y, $w, $h];
1106
1107 my $map = $self->{map_info}[0];
1108 $map =~ s/^.*?\/([^\/]+)$/\1/;
1109 $STATWIDS->{map}->set_text ("Map: " . $map);
653 1110
654 $self->load_map ($hash, $x, $y); 1111 $self->load_map ($hash, $x, $y);
655} 1112}
656 1113
657sub conn::face_find { 1114sub conn::face_find {
683 } 1140 }
684 1141
685gotid: 1142gotid:
686 $face->{id} = $id; 1143 $face->{id} = $id;
687 $MAP->set_face ($facenum => $id); 1144 $MAP->set_face ($facenum => $id);
1145 $self->{faceid}[$facenum] = $id;#d#
688 $TILECACHE->get ($id) 1146 $TILECACHE->get ($id)
689} 1147}
690 1148
691sub conn::face_update { 1149sub conn::face_update {
692 my ($self, $facenum, $face) = @_; 1150 my ($self, $facenum, $face) = @_;
700 my ($self, $id, $data) = @_; 1158 my ($self, $id, $data) = @_;
701 1159
702 $self->{texture}[$id] ||= do { 1160 $self->{texture}[$id] ||= do {
703 my $tex = 1161 my $tex =
704 new_from_image CFClient::Texture 1162 new_from_image CFClient::Texture
705 $data, minify => 1; 1163 $data, minify => 1, mipmap => 1;
706 1164
707 $MAP->set_texture ($id, @$tex{qw(name w h s t)}, @{$tex->{minified}}); 1165 $MAP->set_texture ($id, @$tex{qw(name w h s t)}, @{$tex->{minified}});
708 $MAPWIDGET->update; 1166 $MAPWIDGET->update;
709 1167
710 $tex 1168 $tex
711 }; 1169 };
712} 1170}
713 1171
1172sub conn::sound_play {
1173 my ($self, $x, $y, $soundnum, $type) = @_;
1174
1175 $SDL_MIXER
1176 or return;
1177
1178 my $chunk = $AUDIO_CHUNKS{$SOUNDS[$soundnum]}
1179 or return;
1180
1181 $chunk->play;
1182# warn "sound $x,$y,$soundnum,$type\n";#d#
1183}
1184
1185my $LAST_QUERY; # server is stupid, stupid, stupid
1186
714sub conn::query { 1187sub conn::query {
715 my ($self, $flags, $prompt) = @_; 1188 my ($self, $flags, $prompt) = @_;
716 1189
717 #TODO 1190 $prompt = $LAST_QUERY unless length $prompt;
718 warn "<<<<QUERY:$flags:$prompt>>>\n";#d# 1191 $LAST_QUERY = $prompt;
1192
1193 my $dialog = new CFClient::UI::FancyFrame
1194 title => "Query",
1195 child => my $vbox = new CFClient::UI::VBox;
1196
1197 $vbox->add (new CFClient::UI::Label
1198 max_w => $::WIDTH * 0.4,
1199 text => $prompt);
1200
1201 if ($flags & Crossfire::Protocol::CS_QUERY_YESNO) {
1202 $vbox->add (my $hbox = new CFClient::HBox);
1203 $hbox->add (new CFClient::Button
1204 text => "No",
1205 connect_activate => sub {
1206 $self->send ("reply n");
1207 $dialog->destroy;
1208 $MAPWIDGET->focus_in;
1209 }
1210 );
1211 $hbox->add (new CFClient::Button
1212 text => "Yes",
1213 connect_activate => sub {
1214 $self->send ("reply y");
1215 $dialog->destroy;
1216 $MAPWIDGET->focus_in;
1217 },
1218 );
1219
1220 $dialog->focus_in;
1221
1222 } elsif ($flags & Crossfire::Protocol::CS_QUERY_SINGLECHAR) {
1223 $dialog->{tooltip} = "Press a key (click on the entry to make sure it has keyboard focus)";
1224 $vbox->add (my $entry = new CFClient::UI::Entry
1225 connect_changed => sub {
1226 $self->send ("reply $_[1]");
1227 $dialog->destroy;
1228 $MAPWIDGET->focus_in;
1229 },
1230 );
1231
1232 $entry->focus_in;
1233
1234 } else {
1235 $dialog->{tooltip} = "Enter the reply and press return (click on the entry to make sure it has keyboard focus)";
1236
1237 $vbox->add (my $entry = new CFClient::UI::Entry
1238 $flags & Crossfire::Protocol::CS_QUERY_HIDEINPUT ? (hiddenchar => "*") : (),
1239 connect_activate => sub {
1240 $self->send ("reply $_[1]");
1241 $dialog->destroy;
1242 $MAPWIDGET->focus_in;
1243 },
1244 );
1245
1246 $entry->focus_in;
1247 }
1248
1249 $dialog->show;
719} 1250}
720 1251
721sub conn::drawinfo { 1252sub conn::drawinfo {
722 my ($self, $color, $text) = @_; 1253 my ($self, $color, $text) = @_;
723 1254
738 ); 1269 );
739 1270
740 $LOGVIEW->add_paragraph ($color[$color], $text); 1271 $LOGVIEW->add_paragraph ($color[$color], $text);
741} 1272}
742 1273
1274sub conn::spell_add {
1275 my ($self, $spell) = @_;
1276
1277 # TODO
1278 # create a widget dynamically, using spell face (CF::Protocol downloads them)
1279 $MAPWIDGET->add_command ("invoke $spell->{name}", $spell->{message});
1280 $MAPWIDGET->add_command ("cast $spell->{name}", $spell->{message});
1281}
1282
1283sub conn::spell_delete {
1284 my ($self, $spell) = @_;
1285}
1286
1287sub conn::addme_success {
1288 my ($self) = @_;
1289
1290 for my $skill (values %{$self->{skill_info}}) {
1291 $MAPWIDGET->add_command ("ready_skill $skill", "Ready the skill '$skill'");
1292 $MAPWIDGET->add_command ("use_skill $skill", "Immediately use the skill '$skill'");
1293 }
1294}
1295
1296sub update_floorbox {
1297 $CFClient::UI::ROOT->on_refresh ($FLOORBOX => sub {
1298 $FLOORBOX->clear;
1299 $FLOORBOX->add (new CFClient::UI::Empty expand => 1);
1300
1301 my @items = values %{ $CONN->{container}{0} };
1302
1303 # we basically have to use the same sorting as everybody else
1304 @items = sort { $a->{type} <=> $b->{type} } @items;
1305
1306 for my $item (reverse @items) {
1307 my $desc = $item->{nrof} < 2
1308 ? $item->{name}
1309 : "$item->{nrof} $item->{name_pl}";
1310 # todo: animation widget, face widget, weight(?) etc.
1311 $FLOORBOX->add (my $hbox = new CFClient::UI::HBox
1312 tooltip => (CFClient::UI::Label->escape ($desc)
1313 . "\n<small>leftclick - pick up\nmiddle click - apply\nrightclick - menu</small>"),
1314 can_hover => 1,
1315 can_events => 1,
1316 connect_button_down => sub {
1317 my ($self, $ev, $x, $y) = @_;
1318
1319 # todo: maybe put examine on 1? but should just be a tooltip :(
1320 if ($ev->{button} == 1) {
1321 $CONN->send ("move $CONN->{player}{tag} $item->{tag} 0");
1322 } elsif ($ev->{button} == 2) {
1323 $CONN->send ("apply $item->{tag}");
1324 } elsif ($ev->{button} == 3) {
1325 # examine, lock, mark, maybe other things
1326 warn "MENU not implemented yet\n";
1327 }
1328
1329 1
1330 },
1331 );
1332
1333 $hbox->add (new CFClient::UI::Face
1334 can_events => 0,
1335 face => $item->{face},
1336 anim => $item->{anim},
1337 animspeed => $item->{animspeed},
1338 );
1339
1340 $hbox->add (new CFClient::UI::Label
1341 can_events => 0,
1342 text => $desc,
1343 );
1344 }
1345 });
1346 refresh;
1347}
1348
1349sub conn::container_add {
1350 my ($self, $id, $items) = @_;
1351
1352 update_floorbox if $id == 0;
1353 # $self-<{player}{tag} => player inv
1354 #use PApp::Util; warn PApp::Util::dumpval $self->{container}{$self->{player}{tag}};
1355}
1356
1357sub conn::container_clear {
1358 my ($self, $id) = @_;
1359
1360 update_floorbox if $id == 0;
1361# use PApp::Util; warn PApp::Util::dumpval $self->{container}{0};
1362}
1363
1364sub conn::item_delete {
1365 my ($self, @items) = @_;
1366
1367 for (@items) {
1368 update_floorbox if $_->{container} == 0;
1369 }
1370}
1371
1372sub conn::item_update {
1373 my ($self, $item) = @_;
1374
1375 update_floorbox if $item->{container} == 0;
1376}
1377
743%SDL_CB = ( 1378%SDL_CB = (
744 SDL_QUIT() => sub { 1379 CFClient::SDL_QUIT => sub {
745 Event::unloop -1; 1380 Event::unloop -1;
746 }, 1381 },
747 SDL_VIDEORESIZE() => sub { 1382 CFClient::SDL_VIDEORESIZE => sub {
748 }, 1383 },
749 SDL_VIDEOEXPOSE() => sub { 1384 CFClient::SDL_VIDEOEXPOSE => \&refresh,
750 refresh;
751 },
752 SDL_KEYDOWN() => sub {
753 if ($SDL_EV->key_mod & KMOD_ALT && $SDL_EV->key_sym == SDLK_RETURN) {
754 # alt-enter
755 destroy_screen;
756 $CFG->{fullscreen} = !$CFG->{fullscreen};
757 init_screen;
758 } else {
759 CFClient::UI::feed_sdl_key_down_event ($SDL_EV);
760 }
761 },
762 SDL_KEYUP() => sub {
763 CFClient::UI::feed_sdl_key_up_event ($SDL_EV);
764 },
765 SDL_MOUSEMOTION() => sub {
766 CFClient::UI::feed_sdl_motion_event ($SDL_EV);
767 },
768 SDL_MOUSEBUTTONDOWN() => sub {
769 CFClient::UI::feed_sdl_button_down_event ($SDL_EV);
770 },
771 SDL_MOUSEBUTTONUP() => sub {
772 CFClient::UI::feed_sdl_button_up_event ($SDL_EV);
773 },
774 SDL_ACTIVEEVENT() => sub { 1385 CFClient::SDL_ACTIVEEVENT => sub {
775# printf "active %x %x\n", $SDL_EV->active_gain, $SDL_EV->active_state;#d# 1386# printf "active %x %x\n", $SDL_EV->active_gain, $SDL_EV->active_state;#d#
776 }, 1387 },
1388 CFClient::SDL_KEYDOWN => sub {
1389 if ($_[0]{mod} & CFClient::KMOD_ALT && $_[0]{sym} == 13) {
1390 # alt-enter
1391 video_shutdown;
1392 $CFG->{fullscreen} = !$CFG->{fullscreen};
1393 video_init;
1394 } else {
1395 CFClient::UI::feed_sdl_key_down_event ($_[0]);
1396 }
1397 },
1398 CFClient::SDL_KEYUP => \&CFClient::UI::feed_sdl_key_up_event,
1399 CFClient::SDL_MOUSEMOTION => \&CFClient::UI::feed_sdl_motion_event,
1400 CFClient::SDL_MOUSEBUTTONDOWN => \&CFClient::UI::feed_sdl_button_down_event,
1401 CFClient::SDL_MOUSEBUTTONUP => \&CFClient::UI::feed_sdl_button_up_event,
1402 CFClient::SDL_USEREVENT => \&audio_music_finished,
777); 1403);
778 1404
779############################################################################# 1405#############################################################################
1406
1407$SIG{INT} = $SIG{TERM} = sub { exit };
780 1408
781$TILECACHE = CFClient::db_table "tilecache"; 1409$TILECACHE = CFClient::db_table "tilecache";
782$FACEMAP = CFClient::db_table "facemap"; 1410$FACEMAP = CFClient::db_table "facemap";
783 1411
784CFClient::read_cfg "$Crossfire::VARDIR/pclientrc"; 1412CFClient::read_cfg "$Crossfire::VARDIR/pclientrc";
787 sdl_mode => 0, 1415 sdl_mode => 0,
788 width => 640, 1416 width => 640,
789 height => 480, 1417 height => 480,
790 fullscreen => 0, 1418 fullscreen => 0,
791 fast => 0, 1419 fast => 0,
1420 map_scale => 0.5,
792 fow_enable => 1, 1421 fow_enable => 1,
793 fow_intensity => 0.45, 1422 fow_intensity => 0.45,
794 fow_smooth => 0, 1423 fow_smooth => 0,
1424 gui_fontsize => 1,
795 log_fontsize => 14, 1425 log_fontsize => 1,
1426 gauge_fontsize => 1,
1427 gauge_size => 0.35,
1428 stat_fontsize => 1,
796 mapsize => 100, 1429 mapsize => 100,
797 host => "crossfire.schmorp.de", 1430 host => "crossfire.schmorp.de",
798 say_command => 'say', 1431 say_command => 'say',
1432 audio_enable => 1,
1433 bgm_enable => 1,
1434 bgm_volume => 0.25,
799); 1435);
800 1436
801while (my ($k, $v) = each %DEF_CFG) { 1437while (my ($k, $v) = each %DEF_CFG) {
802 $CFG->{$k} = $v unless exists $CFG->{$k}; 1438 $CFG->{$k} = $v unless exists $CFG->{$k};
803} 1439}
804 1440
805sdl_init; 1441sdl_init;
806 1442
807@SDL_MODES = reverse 1443@SDL_MODES = reverse
808 grep $_->[0] >= 640 && $_->[1] >= 480, 1444 grep $_->[0] >= 640 && $_->[1] >= 480,
809 map [SDL::RectW ($_), SDL::RectH ($_)], 1445 CFClient::SDL_ListModes;
810 @{ SDL::ListModes 0, SDL_FULLSCREEN | SDL_HWSURFACE | SDL_OPENGL };
811 1446
812@SDL_MODES or CFClient::fatal "Unable to find a usable video mode\n(hardware accelerated opengl fullscreen)"; 1447@SDL_MODES or CFClient::fatal "Unable to find a usable video mode\n(hardware accelerated opengl fullscreen)";
813 1448
814$CFG->{sdl_mode} = 0 if $CFG->{sdl_mode} > @SDL_MODES; 1449$CFG->{sdl_mode} = 0 if $CFG->{sdl_mode} > @SDL_MODES;
815 1450
816{ 1451{
817 my @fonts = map CFClient::find_rcfile $_, qw(uifont.ttf uifontb.ttf uifonti.ttf uifontbi.ttf); 1452 my @fonts = map CFClient::find_rcfile "fonts/$_", qw(
1453 DejaVuSans.ttf
1454 DejaVuSansMono.ttf
1455 DejaVuSans-Bold.ttf
1456 DejaVuSansMono-Bold.ttf
1457 DejaVuSans-Oblique.ttf
1458 DejaVuSansMono-Oblique.ttf
1459 DejaVuSans-BoldOblique.ttf
1460 DejaVuSansMono-BoldOblique.ttf
1461 );
818 1462
819 CFClient::add_font $_ for @fonts; 1463 CFClient::add_font $_ for @fonts;
820 CFClient::set_font $fonts[0]; 1464
821} 1465 $FONT_PROP = new_from_file CFClient::Font $fonts[0];
1466 $FONT_FIXED = new_from_file CFClient::Font $fonts[1];
822 1467
823init_screen; 1468 $FONT_PROP->make_default;
1469}
1470
1471video_init;
1472audio_init;
824 1473
825Event::loop; 1474Event::loop;
826 1475
1476END { CFClient::SDL_Quit }
827 1477
1478=head1 pclient - Crossfire+ and Crossfire game client
1479
1480Pclient is a Crossfire+ and Crossfire game client.
1481
1482=head2 Features
1483
1484=over 4
1485
1486=item Fullscreen Map
1487
1488PClient can uses a fullscreen map, which greatly enhances how much of the
1489game world you can see.
1490
1491=item Persistent Map Cache (Crossfire+ only)
1492
1493PClient can persistently cache all map data it received from the
1494server. This not only allows it to display an overview map, but also
1495ensures that once-explored areas will be available the next time you want
1496to explore more.
1497
1498=item Hardware acceleration
1499
1500Unlike most Crossfire clients, PClient take advantage of OpenGL hardware
1501acceleration. Most modern graphics cards have difficulties with 2D
1502acceleration, while 3D graphics is accelerated well.
1503
1504=item No arbitrary limits
1505
1506Unlike other Crossfire clients, pclient does not suffer from arbitrary
1507limits (like a fixed amount of face numbers). There are still limits, but
1508they are not arbitrarily low :)
1509
1510=back
1511
1512=head1 FAQ
1513
1514=over 4
1515
1516=item The client is very sluggish and slow, what can I do about this?
1517
1518Most likely, you don't have accelerated OpenGL support. Try to find a
1519newer driver, or a driver from your hardware vendor, that features OpenGL
1520support.
1521
1522If this is not an option, the following Setup options reduce the load and
1523will likely make the client playable with sofwtare rendering (it will
1524still be slow, though):
1525
1526=over 4
1527
1528=item B<Video Mode> should be set as low as possible (e.g. 640x480)
1529
1530=item Enable B<Fast & Ugly> mode
1531
1532=item Disable B<Fog of War>
1533
1534=item Increase B<Map Scale>
1535
1536=back
1537
1538=back
1539
1540=head1 AUTHOR
1541
1542Marc Lehmann <crossfire@schmorp.de>, Robin Redeker <elmex@ta-sa.org>
1543
1544
1545

Diff Legend

Removed lines
+ Added lines
< Changed lines
> Changed lines