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.117 by root, Sun Apr 16 07:39:51 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;
82our $GAUGES;
83our $STATWIDS;
84
54our $SDL_ACTIVE; 85our $SDL_ACTIVE;
55our $SDL_EV;
56our %SDL_CB; 86our %SDL_CB;
87
88our $SDL_MIXER;
89our @SOUNDS; # event => file mapping
90our %AUDIO_CHUNKS; # audio files
57 91
58our $ALT_ENTER_MESSAGE; 92our $ALT_ENTER_MESSAGE;
59our $STATUS_LINE; 93our $STATUS_LINE;
60our $DEBUG_STATUS; 94our $DEBUG_STATUS;
61 95
62sub status { 96sub status {
63 $STATUS_LINE->set_text ($_[0]); 97 $STATUS_LINE->set_text ($_[0]);
64 my ($w, $h) = $STATUS_LINE->size_request;
65 $STATUS_LINE->size_allocate (0, $HEIGHT - $ALT_ENTER_MESSAGE->{h} - $h, $w, $h); 98 $STATUS_LINE->move (0, $HEIGHT - $ALT_ENTER_MESSAGE->{h} - $STATUS_LINE->{h});
66} 99}
67 100
68sub debug { 101sub debug {
69 $DEBUG_STATUS->set_text ($_[0]); 102 $DEBUG_STATUS->set_text ($_[0]);
70 my ($w, $h) = $DEBUG_STATUS->size_request; 103 $DEBUG_STATUS->move ($WIDTH - $DEBUG_STATUS->{w}, 0, $DEBUG_STATUS->{w}, $DEBUG_STATUS->{h});
71 $DEBUG_STATUS->size_allocate ($WIDTH - $w, 0, $w, $h);
72} 104}
73 105
74sub start_game { 106sub start_game {
75 status "logging in..."; 107 status "logging in...";
76 108
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 => $::WIDTH/4, 730 user_w => int $::WIDTH / 3,
306 user_h => $::HEIGHT, 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::LineEntry 740 $vbox->add (my $input = new CFClient::UI::Entry
741 connect_focus_in => sub {
742 my ($input, $prev_focus) = @_;
743
744 delete $input->{refocus_map};
745
746 if ($prev_focus == $MAPWIDGET && $input->{auto_activated}) {
747 $input->{refocus_map} = 1;
748 }
749 delete $input->{auto_activated};
750 },
315 connect_activate => sub { 751 connect_activate => sub {
316 my ($input, $text) = @_; 752 my ($input, $text) = @_;
317 $input->set_text (''); 753 $input->set_text ('');
318 754
319 if ($text =~ /^\/(.*)/) { 755 if ($text =~ /^\/(.*)/) {
320 $::CONN->user_send ("command $1"); 756 $::CONN->user_send ($1);
321 } else { 757 } else {
322 my $say_cmd = $::CFG->{say_command} || 'say'; 758 my $say_cmd = $::CFG->{say_command} || 'say';
323 $::CONN->user_send ("command $say_cmd $text"); 759 $::CONN->user_send ("$say_cmd $text");
760 }
761 if ($input->{refocus_map}) {
762 delete $input->{refocus_map};
763 $MAPWIDGET->focus_in
324 } 764 }
325 }, 765 },
326 connect_escape => sub { 766 connect_escape => sub {
327 $MAPWIDGET->focus_in 767 $MAPWIDGET->focus_in
328 }, 768 },
335 775
336 $window 776 $window
337} 777}
338 778
339sub sdl_init { 779sub sdl_init {
340 #SDL::Init SDL_INIT_AUDIO | SDL_INIT_VIDEO | SDL_INIT_NOPARACHUTE 780 CFClient::SDL_Init
341 SDL::Init SDL_INIT_AUDIO | SDL_INIT_VIDEO
342 and die "SDL::Init failed!\n"; 781 and die "SDL::Init failed!\n";
343} 782}
344 783
345sub init_screen { 784sub video_init {
346 sdl_init; 785 sdl_init;
347 786
348 ($WIDTH, $HEIGHT) = @{ $SDL_MODES[$CFG->{sdl_mode}] }; 787 ($WIDTH, $HEIGHT) = @{ $SDL_MODES[$CFG->{sdl_mode}] };
349 $FULLSCREEN = $CFG->{fullscreen}; 788 $FULLSCREEN = $CFG->{fullscreen};
350 $FAST = $CFG->{fast}; 789 $FAST = $CFG->{fast};
351 790
352 SDL::GLSetAttribute SDL_GL_RED_SIZE, 5; 791 CFClient::SDL_SetVideoMode $WIDTH, $HEIGHT, $FULLSCREEN
353 SDL::GLSetAttribute SDL_GL_GREEN_SIZE, 5;
354 SDL::GLSetAttribute SDL_GL_BLUE_SIZE, 5;
355 SDL::GLSetAttribute SDL_GL_ALPHA_SIZE, 1;
356
357 SDL::GLSetAttribute SDL_GL_ACCUM_RED_SIZE, 0;
358 SDL::GLSetAttribute SDL_GL_ACCUM_GREEN_SIZE, 0;
359 SDL::GLSetAttribute SDL_GL_ACCUM_BLUE_SIZE, 0;
360 SDL::GLSetAttribute SDL_GL_ACCUM_ALPHA_SIZE, 0;
361
362 SDL::GLSetAttribute SDL_GL_DOUBLEBUFFER, 1;
363 SDL::GLSetAttribute SDL_GL_BUFFER_SIZE, 15;
364 SDL::GLSetAttribute SDL_GL_DEPTH_SIZE, 0;
365
366 SDL::SetVideoMode $WIDTH, $HEIGHT, 0,
367 SDL_HWSURFACE | SDL_ANYFORMAT | SDL_OPENGL | SDL_DOUBLEBUF
368 | ($FULLSCREEN ? SDL_FULLSCREEN : 0)
369 or die "SDL::SetVideoMode failed!\n"; 792 or die "SDL_SetVideoMode failed!\n";
370
371 SDL::WMSetCaption "Crossfire+ Client", "Crossfire+";
372
373 $SDL_EV = new SDL::Event;
374 $SDL_EV->set_unicode (1);
375 793
376 $SDL_ACTIVE = 1; 794 $SDL_ACTIVE = 1;
377 795
378 $LAST_REFRESH = time - 0.01; 796 $LAST_REFRESH = time - 0.01;
379 797
380 CFClient::gl_init; 798 CFClient::gl_init;
381 799
382 $FONTSIZE = int $HEIGHT / 40; 800 $FONTSIZE = int $HEIGHT / 40 * $CFG->{gui_fontsize};
383 801
384 ############################################################################# 802 #############################################################################
385 803
386 $DEBUG_STATUS = new CFClient::UI::Label padding => 0, z => 100; 804 $DEBUG_STATUS = new CFClient::UI::Label padding => 0, z => 100;
387 $CFClient::UI::ROOT->add ($DEBUG_STATUS); 805 $DEBUG_STATUS->show;
388 806
389 $STATUS_LINE = new CFClient::UI::Label 807 $STATUS_LINE = new CFClient::UI::Label
390 padding => 0, 808 padding => 0,
391 y => $HEIGHT * 44 / 45 - $FONTSIZE; 809 y => $HEIGHT - $FONTSIZE * 1.8;
392 $CFClient::UI::ROOT->add ($STATUS_LINE); 810 $STATUS_LINE->show;
393 811
394 $ALT_ENTER_MESSAGE = new CFClient::UI::Label 812 $ALT_ENTER_MESSAGE = new CFClient::UI::Label
395 padding => 0, 813 padding => 0,
396 y => $HEIGHT * 44 / 45, 814 fontsize => 0.8,
397 height => $HEIGHT / 45,
398 text => "Use <b>Alt-Enter</b> to toggle fullscreen mode"; 815 markup => "Use <b>Alt-Enter</b> to toggle fullscreen mode";
399 $CFClient::UI::ROOT->add ($ALT_ENTER_MESSAGE); 816 $ALT_ENTER_MESSAGE->show;
817 $ALT_ENTER_MESSAGE->move (0, $HEIGHT - $ALT_ENTER_MESSAGE->{h});
400 818
401 $CFClient::UI::ROOT->add ($MAPWIDGET = new CFClient::UI::MapWidget); 819 $CFClient::UI::ROOT->add ($MAPWIDGET = new CFClient::MapWidget);
402 $MAPWIDGET->focus_in; 820 $MAPWIDGET->focus_in;
403 $MAPWIDGET->connect (activate_console => sub { 821 $MAPWIDGET->connect (activate_console => sub {
404 my ($mapwidget, $preset) = @_; 822 my ($mapwidget, $preset) = @_;
405 823
406 if ($CONSOLE) { 824 if ($CONSOLE) {
825 $CONSOLE->{input}->{auto_activated} = 1;
407 $CONSOLE->{input}->focus_in; 826 $CONSOLE->{input}->focus_in;
408 827
409 if ($preset && $CONSOLE->{input}->get_text eq '') { 828 if ($preset && $CONSOLE->{input}->get_text eq '') {
410 $CONSOLE->{input}->set_text ($preset); 829 $CONSOLE->{input}->set_text ($preset);
411 } 830 }
415 $CFClient::UI::ROOT->add ($BUTTONBAR = new CFClient::UI::HBox); 834 $CFClient::UI::ROOT->add ($BUTTONBAR = new CFClient::UI::HBox);
416 835
417 $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);
418 $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);
419 $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);
420 842
421 $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 {
422 CFClient::write_cfg "$Crossfire::VARDIR/pclientrc"; 844 CFClient::write_cfg "$Crossfire::VARDIR/pclientrc";
423 status "Configuration Saved"; 845 status "Configuration Saved";
424 }); 846 });
425 847
426 $BUTTONBAR->{children}[0]->emit ("activate"); 848 $BUTTONBAR->{children}[1]->emit ("activate"); # pop up server setup
427} 849}
428 850
429sub destroy_screen { 851sub video_shutdown {
430 $CFClient::UI::ROOT->{children} = []; 852 $CFClient::UI::ROOT->{children} = [];
853 undef $CFClient::UI::GRAB;
854 undef $CFClient::UI::HOVER;
431 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;
432 undef $SDL_EV; 902 undef $SDL_MIXER;
433 SDL::Quit; 903 @SOUNDS = ();
904 %AUDIO_CHUNKS = ();
434} 905}
435 906
436my %animate_object; 907my %animate_object;
437my $animate_timer; 908my $animate_timer;
438 909
448 $want_refresh = 0; 919 $want_refresh = 0;
449 $can_refresh = 0; 920 $can_refresh = 0;
450 921
451 $CFClient::UI::ROOT->draw; 922 $CFClient::UI::ROOT->draw;
452 923
453 SDL::GLSwapBuffers; 924 CFClient::SDL_GL_SwapBuffers;
454 925
455 $LAST_REFRESH = $NOW; 926 $LAST_REFRESH = $NOW;
456} 927}
457 928
458my $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 {
459 $NOW = time; 930 $NOW = time;
460 931
461 ($SDL_CB{$SDL_EV->type} || sub { warn "unhandled event ", $SDL_EV->type })->() 932 ($SDL_CB{$_->{type}} || sub { warn "unhandled event $_->{type}" })->($_)
462 while $SDL_EV->poll; 933 for CFClient::SDL_PollEvent;
463 934
464 if (%animate_object) { 935 if (%animate_object) {
465 $_->animate ($LAST_REFRESH - $NOW) for values %animate_object; 936 $_->animate ($LAST_REFRESH - $NOW) for values %animate_object;
466 $want_refresh++; 937 $want_refresh++;
467 } 938 }
487 delete $animate_object{$widget}; 958 delete $animate_object{$widget};
488} 959}
489 960
490@conn::ISA = Crossfire::Protocol::; 961@conn::ISA = Crossfire::Protocol::;
491 962
963sub conn::stats_update {
964 my ($self, $stats) = @_;
965
966 update_stats_window ($stats);
967}
968
492sub conn::user_send { 969sub conn::user_send {
493 my ($self, $command) = @_; 970 my ($self, $command) = @_;
494 971
495 $self->send ($command); 972 $self->send_command ($command);
496 status $command; 973 status $command;
974}
975
976sub conn::map_scroll {
977 my ($self, $dx, $dy) = @_;
978
979 $MAP->scroll ($dx, $dy);
497} 980}
498 981
499sub conn::feed_map1a { 982sub conn::feed_map1a {
500 my ($self, $data) = @_; 983 my ($self, $data) = @_;
501 984
502# $self->Crossfire::Protocol::feed_map1a ($data); 985# $self->Crossfire::Protocol::feed_map1a ($data);
503 986
504 $MAP->scroll (delete $self->{delayed_scroll_x}, delete $self->{delayed_scroll_y});
505 $MAP->map1a_update ($data); 987 $MAP->map1a_update ($data);
506 $MAPWIDGET->update; 988 $MAPWIDGET->update;
507} 989}
508 990
509sub conn::flush_map { 991sub conn::flush_map {
514 996
515 my ($hash, $x, $y, $w, $h) = @$map_info; 997 my ($hash, $x, $y, $w, $h) = @$map_info;
516 998
517 my $data = $MAP->get_rect ($x, $y, $w, $h); 999 my $data = $MAP->get_rect ($x, $y, $w, $h);
518 $MAPCACHE->put ($hash => Compress::LZF::compress $data); 1000 $MAPCACHE->put ($hash => Compress::LZF::compress $data);
519
520 warn sprintf "SAVEmap[%s] length %d\n", $hash, length $data;#d# 1001 #warn sprintf "SAVEmap[%s] length %d\n", $hash, length $data;#d#
521
522} 1002}
523 1003
524sub conn::map_clear { 1004sub conn::map_clear {
525 my ($self) = @_; 1005 my ($self) = @_;
526 1006
527 $self->flush_map; 1007 $self->flush_map;
1008 delete $self->{neigh_map};
528 1009
529 $MAP->clear; 1010 $MAP->clear;
530} 1011}
531 1012
532sub conn::map_info {
533 my ($self, $mode, $flags, $x, $y, $w, $h, $hash) = @_;
534 1013
535 $self->flush_map; 1014sub conn::load_map($$$) {
536 1015 my ($self, $hash, $x, $y) = @_;
537 $x = $::MAP->ox - $x;
538 $y = $::MAP->oy - $y;
539
540 $self->{map_info} = [$hash, $x, $y, $w, $h];
541 1016
542 if (defined (my $data = $MAPCACHE->get ($hash))) { 1017 if (defined (my $data = $MAPCACHE->get ($hash))) {
543 $data = Compress::LZF::decompress $data; 1018 $data = Compress::LZF::decompress $data;
544 warn sprintf "LOADmap[%s] length %d\n", $hash, length $data;#d# 1019 #warn sprintf "LOADmap[%s,%d,%d] length %d\n", $hash, $x, $y, length $data;#d#
545 for my $id ($MAP->set_rect ($x, $y, $data)) { 1020 for my $id ($MAP->set_rect ($x, $y, $data)) {
546 my $data = $TILECACHE->get ($id) 1021 my $data = $TILECACHE->get ($id)
547 or next; 1022 or next;
548 1023
549 $self->set_texture ($id => $data); 1024 $self->set_texture ($id => $data);
550 } 1025 }
551 } 1026 }
1027}
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.
1034sub conn::flood_fill {
1035 my ($self, $gx, $gy, $path, $hash, $flags) = @_;
1036
1037 # the server does not allow map paths > 6
1038 return if 6 <= length $path;
1039
1040 my ($x0, $y0, $x1, $y1) = @{$self->{neigh_rect}};
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
1053 next unless $flags & (1 << ($tile - 1));
1054 next if $self->{neigh_grid}{$gx, $gy}++;
1055
1056 my $neigh = $self->{neigh_map}{$hash} ||= [];
1057 if (my $info = $neigh->[$tile]) {
1058 my ($flags, $x, $y, $w, $h, $hash) = @$info;
1059
1060 $self->flood_fill ($gx, $gy, "$path$tile", $hash, $flags)
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 });
1080 }
1081 }
1082}
1083
1084sub conn::map_change {
1085 my ($self, $mode, $flags, $x, $y, $w, $h, $hash) = @_;
1086
1087 $self->flush_map;
1088
1089 my ($ox, $oy) = ($::MAP->ox, $::MAP->oy);
1090
1091 my $mapmapw = 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 ];
1098
1099 delete $self->{neigh_grid};
1100 $self->flood_fill (0, 0, "", $hash, $flags);
1101
1102 $x += $ox;
1103 $y += $oy;
1104
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);
1110
1111 $self->load_map ($hash, $x, $y);
552} 1112}
553 1113
554sub conn::face_find { 1114sub conn::face_find {
555 my ($self, $facenum, $face) = @_; 1115 my ($self, $facenum, $face) = @_;
556 1116
580 } 1140 }
581 1141
582gotid: 1142gotid:
583 $face->{id} = $id; 1143 $face->{id} = $id;
584 $MAP->set_face ($facenum => $id); 1144 $MAP->set_face ($facenum => $id);
1145 $self->{faceid}[$facenum] = $id;#d#
585 $TILECACHE->get ($id) 1146 $TILECACHE->get ($id)
586} 1147}
587 1148
588sub conn::face_update { 1149sub conn::face_update {
589 my ($self, $facenum, $face) = @_; 1150 my ($self, $facenum, $face) = @_;
597 my ($self, $id, $data) = @_; 1158 my ($self, $id, $data) = @_;
598 1159
599 $self->{texture}[$id] ||= do { 1160 $self->{texture}[$id] ||= do {
600 my $tex = 1161 my $tex =
601 new_from_image CFClient::Texture 1162 new_from_image CFClient::Texture
602 $data, minify => 1; 1163 $data, minify => 1, mipmap => 1;
603 1164
604 $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}});
605 $MAPWIDGET->update; 1166 $MAPWIDGET->update;
606 1167
607 $tex 1168 $tex
608 }; 1169 };
609} 1170}
610 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
611sub conn::query { 1187sub conn::query {
612 my ($self, $flags, $prompt) = @_; 1188 my ($self, $flags, $prompt) = @_;
613 1189
614 #TODO 1190 $prompt = $LAST_QUERY unless length $prompt;
615 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;
616} 1250}
617 1251
618sub conn::drawinfo { 1252sub conn::drawinfo {
619 my ($self, $color, $text) = @_; 1253 my ($self, $color, $text) = @_;
620 1254
635 ); 1269 );
636 1270
637 $LOGVIEW->add_paragraph ($color[$color], $text); 1271 $LOGVIEW->add_paragraph ($color[$color], $text);
638} 1272}
639 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
640%SDL_CB = ( 1378%SDL_CB = (
641 SDL_QUIT() => sub { 1379 CFClient::SDL_QUIT => sub {
642 Event::unloop -1; 1380 Event::unloop -1;
643 }, 1381 },
644 SDL_VIDEORESIZE() => sub { 1382 CFClient::SDL_VIDEORESIZE => sub {
645 }, 1383 },
646 SDL_VIDEOEXPOSE() => sub { 1384 CFClient::SDL_VIDEOEXPOSE => \&refresh,
647 refresh;
648 },
649 SDL_KEYDOWN() => sub {
650 if ($SDL_EV->key_mod & KMOD_ALT && $SDL_EV->key_sym == SDLK_RETURN) {
651 # alt-enter
652 destroy_screen;
653 $CFG->{fullscreen} = !$CFG->{fullscreen};
654 init_screen;
655 } else {
656 CFClient::UI::feed_sdl_key_down_event ($SDL_EV);
657 }
658 },
659 SDL_KEYUP() => sub {
660 CFClient::UI::feed_sdl_key_up_event ($SDL_EV);
661 },
662 SDL_MOUSEMOTION() => sub {
663 CFClient::UI::feed_sdl_motion_event ($SDL_EV);
664 },
665 SDL_MOUSEBUTTONDOWN() => sub {
666 CFClient::UI::feed_sdl_button_down_event ($SDL_EV);
667 },
668 SDL_MOUSEBUTTONUP() => sub {
669 CFClient::UI::feed_sdl_button_up_event ($SDL_EV);
670 },
671 SDL_ACTIVEEVENT() => sub { 1385 CFClient::SDL_ACTIVEEVENT => sub {
672# 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#
673 }, 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,
674); 1403);
675 1404
676############################################################################# 1405#############################################################################
1406
1407$SIG{INT} = $SIG{TERM} = sub { exit };
677 1408
678$TILECACHE = CFClient::db_table "tilecache"; 1409$TILECACHE = CFClient::db_table "tilecache";
679$FACEMAP = CFClient::db_table "facemap"; 1410$FACEMAP = CFClient::db_table "facemap";
680 1411
681CFClient::read_cfg "$Crossfire::VARDIR/pclientrc"; 1412CFClient::read_cfg "$Crossfire::VARDIR/pclientrc";
684 sdl_mode => 0, 1415 sdl_mode => 0,
685 width => 640, 1416 width => 640,
686 height => 480, 1417 height => 480,
687 fullscreen => 0, 1418 fullscreen => 0,
688 fast => 0, 1419 fast => 0,
1420 map_scale => 0.5,
689 fow_enable => 1, 1421 fow_enable => 1,
690 fow_intensity => 0.45, 1422 fow_intensity => 0.45,
691 fow_smooth => 0, 1423 fow_smooth => 0,
1424 gui_fontsize => 1,
692 log_fontsize => 14, 1425 log_fontsize => 1,
1426 gauge_fontsize => 1,
1427 gauge_size => 0.35,
1428 stat_fontsize => 1,
693 mapsize => 100, 1429 mapsize => 100,
694 host => "crossfire.schmorp.de", 1430 host => "crossfire.schmorp.de",
695 say_command => 'say', 1431 say_command => 'say',
1432 audio_enable => 1,
1433 bgm_enable => 1,
1434 bgm_volume => 0.25,
696); 1435);
697 1436
698while (my ($k, $v) = each %DEF_CFG) { 1437while (my ($k, $v) = each %DEF_CFG) {
699 $CFG->{$k} = $v unless exists $CFG->{$k}; 1438 $CFG->{$k} = $v unless exists $CFG->{$k};
700} 1439}
701 1440
702sdl_init; 1441sdl_init;
703 1442
704@SDL_MODES = reverse 1443@SDL_MODES = reverse
705 grep $_->[0] >= 640 && $_->[1] >= 480, 1444 grep $_->[0] >= 640 && $_->[1] >= 480,
706 map [SDL::RectW ($_), SDL::RectH ($_)], 1445 CFClient::SDL_ListModes;
707 @{ SDL::ListModes 0, SDL_FULLSCREEN | SDL_HWSURFACE | SDL_OPENGL };
708 1446
709@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)";
710 1448
711$CFG->{sdl_mode} = 0 if $CFG->{sdl_mode} > @SDL_MODES; 1449$CFG->{sdl_mode} = 0 if $CFG->{sdl_mode} > @SDL_MODES;
712 1450
713init_screen;
714
715{ 1451{
716 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 );
717 1462
718 CFClient::add_font $_ for @fonts; 1463 CFClient::add_font $_ for @fonts;
719 CFClient::set_font $fonts[0]; 1464
1465 $FONT_PROP = new_from_file CFClient::Font $fonts[0];
1466 $FONT_FIXED = new_from_file CFClient::Font $fonts[1];
1467
1468 $FONT_PROP->make_default;
720} 1469}
1470
1471video_init;
1472audio_init;
721 1473
722Event::loop; 1474Event::loop;
723 1475
1476END { CFClient::SDL_Quit }
724 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