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.53 by root, Sun Apr 9 22:17:40 2006 UTC vs.
Revision 1.196 by elmex, Sun Apr 30 13:10:47 2006 UTC

1#!/opt/bin/perl 1#!/opt/bin/perl
2 2
3use strict; 3use strict;
4use utf8; 4use utf8;
5 5
6use Glib; 6BEGIN {
7use Gtk2 -init; 7 if (%PAR::LibCache) {
8 @INC = grep ref, @INC; # weed out all paths except pars loader refs
8 9
9use SDL; 10 while (my ($filename, $zip) = each %PAR::LibCache) {
10use SDL::App; 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
29use Time::HiRes 'time';
11use SDL::Event; 30use Event;
12use SDL::Surface;
13use SDL::OpenGL;
14use SDL::OpenGL::Constants;
15 31
16use Crossfire; 32use Crossfire;
17use Crossfire::Client;
18use Crossfire::Protocol; 33use Crossfire::Protocol;
19 34
20use Crossfire::Client::Widget; 35use Compress::LZF;
21 36
22our $FACECACHE; 37use CFClient;
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#
23 46
24our $VERSION = '0.1'; 47our $VERSION = '0.1';
25 48
26our %GL_EXT; 49my $MAX_FPS = 60;
50my $MIN_FPS = 5; # unused as of yet
51
52our $META_SERVER = "crossfire.real-time.com:13326";
53
54our $FACEMAP;
55our $TILECACHE;
56our $MAPCACHE;
57
58our $LAST_REFRESH;
59our $NOW;
27 60
28our $CFG; 61our $CFG;
29our $CONN; 62our $CONN;
63our $FAST; # fast, low-quality mode, possibly useful for software-rendering
30 64
65our @SDL_MODES;
31our $WIDTH; 66our $WIDTH;
32our $HEIGHT; 67our $HEIGHT;
33our $FULLSCREEN; 68our $FULLSCREEN;
34
35our $FONTSIZE; 69our $FONTSIZE;
36 70
71our $FONT_PROP;
72our $FONT_FIXED;
73
74our $MAP;
75our $MAPMAP;
76our $MAPWIDGET;
77our $BUTTONBAR;
78our $LOGVIEW;
79our $CONSOLE;
80our $METASERVER;
81
82our $FLOORBOX;
83our $GAUGES;
84our $STATWIDS;
85
37our $SDL_TIMER; 86our $SDL_ACTIVE;
38our $SDL_APP;
39our $SDL_EV = new SDL::Event;
40our %SDL_CB; 87our %SDL_CB;
41 88
42our @GL_INIT; # hooks called on every gl init 89our $SDL_MIXER;
90our @SOUNDS; # event => file mapping
91our %AUDIO_CHUNKS; # audio files
43 92
44our $ALT_ENTER_MESSAGE; 93our $ALT_ENTER_MESSAGE;
45our $STATUS_LINE; 94our $STATUS_LINE;
95our $DEBUG_STATUS;
46 96
47our $TOPLEVEL; 97our $INVWIN;
98our $INV;
48 99
49our $tw; # Test widget #d# 100sub status {
101 $STATUS_LINE->set_text ($_[0]);
102 $STATUS_LINE->move (0, $HEIGHT - $ALT_ENTER_MESSAGE->{h} - $STATUS_LINE->{h});
103}
50 104
51my $last_refresh; 105sub debug {
52my %ANIMATE; 106 $DEBUG_STATUS->set_text ($_[0]);
53my $refresh_handler; 107 $DEBUG_STATUS->move ($WIDTH - $DEBUG_STATUS->{w}, 0, $DEBUG_STATUS->{w}, $DEBUG_STATUS->{h});
108}
54 109
55sub init_screen { 110sub start_game {
56 $SDL_APP = new SDL::App 111 status "logging in...";
57 -flags => SDL_ANYFORMAT | SDL_HWSURFACE,
58 -title => "Crossfire+ Client",
59 -width => $WIDTH,
60 -height => $HEIGHT,
61 -opengl => 1,
62 -red_size => 5,
63 -green_size => 5,
64 -blue_size => 5,
65 -alpha_size => 0,
66 -double_buffer => 1,
67 -fullscreen => $FULLSCREEN,
68 -resizeable => 0;
69 112
70 $last_refresh = SDL::GetTicks; 113 my $mapsize = List::Util::min 32, List::Util::max 11, int $WIDTH * $CFG->{mapsize} * 0.01 / 32;
71 114
72 %GL_EXT = map +($_ => 1), split /\s+/, Crossfire::Client::gl_extensions; 115 $MAPCACHE = CFClient::db_table "mapcache_$CFG->{host}";
73 116
74 $GL_EXT{GL_ARB_texture_non_power_of_two} 117 $MAP = new CFClient::Map $mapsize, $mapsize;
75 or warn "WARNING: non-power-of-two opengl extension required";
76 118
77 $FONTSIZE = int $HEIGHT / 50; 119 my ($host, $port) = split /:/, $CFG->{host};
120
121 $CONN = eval {
122 new conn
123 host => $host,
124 port => $port || 13327,
125 user => $CFG->{user},
126 pass => $CFG->{password},
127 mapw => $mapsize,
128 maph => $mapsize,
129 ;
130 };
131
132 if ($CONN) {
133 status "login successful";
134
135 CFClient::lowdelay fileno $CONN->{fh};
136 } else {
137 status "unable to connect";
138 }
139}
140
141sub stop_game {
142 undef $CONN;
143}
144
145sub client_setup {
146 my $dialog = new CFClient::UI::FancyFrame
147 title => "Client Setup",
148 child => (my $vbox = new CFClient::UI::VBox);
149 $vbox->add (my $table = new CFClient::UI::Table expand => 1, col_expand => [0, 1]);
150
151 $table->add (0, 0, new CFClient::UI::Label valign => 0, align => 1, text => "Video Mode");
152 $table->add (1, 0, my $hbox = new CFClient::UI::HBox);
153
154 $hbox->add (my $mode_slider = new CFClient::UI::Slider expand => 1, req_w => 100, range => [$CFG->{sdl_mode}, 0, scalar @SDL_MODES, 1]);
155 $hbox->add (my $mode_label = new CFClient::UI::Label align => 0, valign => 0, height => 0.8, template => "9999x9999");
156
157 $mode_slider->connect (changed => sub {
158 my ($self, $value) = @_;
159
160 $CFG->{sdl_mode} = $self->{range}[0] = $value = int $value;
161 $mode_label->set_text (sprintf "%dx%d", @{$SDL_MODES[$value]});
162 });
163 $mode_slider->emit (changed => $mode_slider->{range}[0]);
164
165 my $row = 1;
166
167 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Fullscreen");
168 $table->add (1, $row++, new CFClient::UI::CheckBox
169 state => $CFG->{fullscreen},
170 tooltip => "Bring the client into fullscreen mode",
171 connect_changed => sub {
172 my ($self, $value) = @_;
173 $CFG->{fullscreen} = $value;
174 }
175 );
176
177 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Fast & Ugly");
178 $table->add (1, $row++, new CFClient::UI::CheckBox
179 state => $CFG->{fast},
180 tooltip => "Lower the visual quality considerably to speed up rendering.",
181 connect_changed => sub {
182 my ($self, $value) = @_;
183 $CFG->{fast} = $value;
184 }
185 );
186
187 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Map Scale");
188 $table->add (1, $row++, new CFClient::UI::Slider
189 range => [$CFG->{map_scale}, 0.25, 2, 0.05],
190 tooltip => "Enlarge or shrink the displayed map",
191 connect_changed => sub {
192 my ($self, $value) = @_;
193 $CFG->{map_scale} = 0.05 * int $value / 0.05;
194 }
195 );
196
197 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Fog of War");
198 $table->add (1, $row++, new CFClient::UI::CheckBox
199 state => $CFG->{fow_enable},
200 tooltip => "Fog-of-War marks areas that cannot be seen by the player",
201 connect_changed => sub {
202 my ($self, $value) = @_;
203 $CFG->{fow_enable} = $value;
204 }
205 );
206
207 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "FoW Intensity");
208 $table->add (1, $row++, new CFClient::UI::Slider
209 range => [$CFG->{fow_intensity}, 0, 1 + 0.001, 0.001],
210 tooltip => "The higher the intensity, the lighter the Fog-of-War color",
211 connect_changed => sub {
212 my ($self, $value) = @_;
213 $CFG->{fow_intensity} = $value;
214 }
215 );
216
217 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "FoW Smooth");
218 $table->add (1, $row++, new CFClient::UI::CheckBox
219 state => $CFG->{fow_smooth},
220 tooltip => "Smooth the Fog-of-War a bit to make it more realistic",
221 connect_changed => sub {
222 my ($self, $value) = @_;
223 $CFG->{fow_smooth} = $value;
224 status "Fog of War smoothing requires OpenGL 1.2 or higher" if $CFClient::GL_VERSION < 1.2;
225 }
226 );
227
228 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "GUI Fontsize");
229 $table->add (1, $row++, new CFClient::UI::Slider
230 range => [$CFG->{gui_fontsize}, 0.5, 2, 0.1],
231 tooltip => "The font size used by most GUI elements",
232 connect_changed => sub {
233 $CFG->{gui_fontsize} = 0.1 * int $_[1] * 10;
234# $FONTSIZE = int $HEIGHT / 40 * $CFG->{gui_fontsize};
235 }
236 );
237
238 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Server Log Fontsize");
239 $table->add (1, $row++, new CFClient::UI::Slider
240 range => [$CFG->{log_fontsize}, 0.5, 2, 0.1],
241 tooltip => "The font size used by the server log window only",
242 connect_changed => sub {
243 $LOGVIEW->set_fontsize ($CFG->{log_fontsize} = 0.1 * int $_[1] * 10);
244 }
245 );
246
247 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Stats Fontsize");
248
249 $table->add (1, $row++, new CFClient::UI::Slider
250 range => [$CFG->{stat_fontsize}, 0.5, 2, 0.1],
251 tooltip => "The font size used by the statistics window only",
252 connect_changed => sub {
253 $CFG->{stat_fontsize} = 0.1 * int $_[1] * 10;
254 &set_stats_window_fontsize;
255 }
256 );
257
258 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Gauge size");
259 $table->add (1, $row++, new CFClient::UI::Slider
260 range => [$CFG->{gauge_size}, 0.2, 0.8, 0.02],
261 tooltip => "Adjust the size of the stats gauges at the bottom right",
262 connect_changed => sub {
263 $CFG->{gauge_size} = $_[1];
264 my $h = int $HEIGHT * $CFG->{gauge_size};
265 $GAUGES->{win}->set_size ($WIDTH, $h);
266 $GAUGES->{win}->move (0, $HEIGHT - $h);
267 }
268 );
269
270 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Gauge fontsize");
271 $table->add (1, $row++, new CFClient::UI::Slider
272 range => [$CFG->{gauge_fontsize}, 0.5, 2.0, 0.1],
273 tooltip => "Adjusts the fontsize of the gauges at the bottom right",
274 connect_changed => sub {
275 $CFG->{gauge_fontsize} = 0.1 * int $_[1] * 10;
276 &set_gauge_window_fontsize;
277 }
278 );
279
280 $table->add (1, $row++, new CFClient::UI::Button
281 expand => 1, align => 0, text => "Apply",
282 tooltip => "Apply the video settings",
283 connect_activate => sub {
284 video_shutdown ();
285 video_init ();
286 }
287 );
288
289 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Audio Enable");
290 $table->add (1, $row++, new CFClient::UI::CheckBox
291 state => $CFG->{audio_enable},
292 tooltip => "If enabled, sound effects and music will be played. If disabled, no audio will be used and the soundcard will not be opened.",
293 connect_changed => sub {
294 $CFG->{audio_enable} = $_[1];
295 }
296 );
297# $table->add (0, 9, new CFClient::UI::Label valign => 0, align => 1, text => "Effects Volume");
298# $table->add (1, 8, new CFClient::UI::Slider range => [$CFG->{effects_volume}, 0, 128, 1], connect_changed => sub {
299# $CFG->{effects_volume} = $_[1];
300# });
301 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Background Music");
302 $table->add (1, $row++, my $hbox = new CFClient::UI::HBox);
303 $hbox->add (new CFClient::UI::CheckBox
304 expand => 1, state => $CFG->{bgm_enable},
305 tooltip => "Enable background music playing",
306 connect_changed => sub {
307 $CFG->{bgm_enable} = $_[1];
308 }
309 );
310 $hbox->add (new CFClient::UI::Slider
311 expand => 1, range => [$CFG->{bgm_volume}, 0, 1, 0.1],
312 tooltip => "The volume of the background music",
313 connect_changed => sub {
314 $CFG->{bgm_volume} = $_[1];
315 CFClient::MixMusic::volume $_[1] * 128;
316 }
317 );
318
319 $table->add (1, $row++, new CFClient::UI::Button
320 expand => 1, align => 0, text => "Apply",
321 tooltip => "Apply the audio settings",
322 connect_activate => sub {
323 audio_shutdown ();
324 audio_init ();
325 }
326 );
327
328 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Communication cmd");
329 $table->add (1, $row++, my $saycmd = new CFClient::UI::Entry
330 text => $CFG->{say_command},
331 tooltip => "This is the command that will be used if you write a line in the message window entry. "
332 ."Usually you want to enter something like 'say' or 'shout' or 'gsay' here. "
333 ."But you could also set it to 'tell &lt;playername&gt;' to only chat with that user.",
334 connect_changed => sub {
335 my ($self, $value) = @_;
336 $CFG->{say_command} = $value;
337 }
338 );
339
340 $dialog
341}
342
343sub set_stats_window_fontsize {
344 for (values %{$STATWIDS}) {
345 $_->set_fontsize ($::CFG->{stat_fontsize});
346 }
347}
348
349sub set_gauge_window_fontsize {
350 for (map { $GAUGES->{$_} } grep { $_ ne 'win' } keys %{$GAUGES}) {
351 $_->set_fontsize ($::CFG->{gauge_fontsize});
352 }
353
354# local $GAUGES->{win}{parent};#d#
355# use PApp::Util; open D, ">:utf8", "d"; print D PApp::Util::dumpval $GAUGES->{win}; close D;
356}
357
358sub make_gauge_window {
359 my $gh = int ($HEIGHT * $CFG->{gauge_size});
360# my $gw = int ($WIDTH * $CFG->{gauge_w_size});
361
362 my $win = new CFClient::UI::Frame (
363 y => $HEIGHT - $gh, x => 0, user_w => $WIDTH, user_h => $gh
364 );
365 $win->add (my $hbox = new CFClient::UI::HBox
366 children => [
367 (new CFClient::UI::HBox expand => 1),
368 ($FLOORBOX = new CFClient::UI::VBox),
369 (my $vbox = new CFClient::UI::VBox),
370 ],
371 );
372
373 $vbox->add (new CFClient::UI::HBox
374 expand => 1,
375 children => [
376 (new CFClient::UI::Empty expand => 1),
377 (my $hb = new CFClient::UI::HBox),
378 ],
379 );
380
381 $hb->add (my $hg = new CFClient::UI::Gauge type => 'hp',
382 tooltip => "Health points. Measures of how much damage you can take before dying. Hit points are determined from your level and are influenced by the value of your Con. Hp value may range between 1 to beyond 500 and higher values indicate a greater ability to withstand punishment.");
383 $hb->add (my $mg = new CFClient::UI::Gauge type => 'mana',
384 tooltip => "Spell points. Measures of how much \"fuel\" you have for casting spells and incantations. Mana is calculated from your level and your Pow. Mana values can range between 1 to beyond 500 (glowing crystals can increase the current spell points beyond your normal maximum). Higher values indicate greater amounts of mana.");
385 $hb->add (my $gg = new CFClient::UI::Gauge type => 'grace',
386 tooltip => "Grace points - how favored you are by your god. In game terms, how much divine magic you can cast. Your level, Wis and Pow effect what the value of grace is. Prayong on an altar of your god can increase this value beyond your normal maximum. Grace can take on large positive and negative values. Positive values indicate favor by the gods.");
387 $hb->add (my $fg = new CFClient::UI::Gauge type => 'food',
388 tooltip => "Food. Ranges between 0 (starving) and 999 (satiated). At a value of 0 the character begins to die. Some magic can speed up or slow down the character digestion. Healing wounds will speed up digestion too.");
389
390 $vbox->add (my $exp = new CFClient::UI::Label valign => 0, align => 1, can_hover => 1, can_events => 1,
391 tooltip => "Experience points and overall level - experience is increased as a reward for appropriate action (such as killing monsters) and may decrease as a result of a magical attack or dying. Level is directly derived from the experience value. As the level of the character increases, the character becomes able to succeed at more difficult tasks. A character's level starts at a value of 0 and may range up beyond 100.");
392 $vbox->add (my $rng = new CFClient::UI::Label valign => 0, align => 1, can_hover => 1, can_events => 1,
393 tooltip => "Ranged attack - how you attack when you press shift-cursor (spell, skill, weapon etc.)");
394
395 $GAUGES = {
396 exp => $exp, win => $win, range => $rng,
397 food => $fg, mana => $mg, hp => $hg, grace => $gg
398 };
399
400 &set_gauge_window_fontsize;
401
402 $win
403}
404
405sub make_stats_window {
406 my $tgw = new CFClient::UI::FancyFrame x => $WIDTH * 2/5, y => 0, title => "Stats";
407
408 $tgw->add (new CFClient::UI::Window child => my $vb = new CFClient::UI::VBox);
409 $vb->add ($STATWIDS->{title} = new CFClient::UI::Label valign => 0, align => -1, text => "Title:", expand => 1);
410 $vb->add ($STATWIDS->{map} = new CFClient::UI::Label valign => 0, align => -1, text => "Map:", expand => 1);
411
412 $vb->add (my $hb = new CFClient::UI::HBox expand => 1);
413
414 $hb->add (my $tbl = new CFClient::UI::Table expand => 1);
415
416 my $black = [0, 0, 0];
417
418 for (
419 [0, 0, st_str => "Str", 30, "Physical Strength, determines damage dealt with weapons, how much you can carry, and how often you can attack"],
420 [0, 1, st_dex => "Dex", 30, "Dexterity, your physical agility. Determines chance of being hit and affects armor class and speed"],
421 [0, 2, st_con => "Con", 30, "Constitution, physical health and toughness. Determines how many healthpoints you can have"],
422 [0, 3, st_int => "Int", 30, "Intelligence, your ability to learn and use skills and incantations (both prayers and magic) and determines how much spell points you can have"],
423 [0, 4, st_wis => "Wis", 30, "Wisdom, the ability to learn and use divine magic (prayers). Determines how many grace points you can have"],
424 [0, 5, st_pow => "Pow", 30, "Power, your magical potential. Influences the strength of spell effects, and also how much your spell and grace points increase when leveling up"],
425 [0, 6, st_cha => "Cha", 30, "Charisma, how well you are received by NPCs. Affects buying and selling prices in shops."],
426
427 [2, 0, st_wc => "Wc", -120, "Weapon Class, effectiveness of melee/missile attacks. Lower is more potent. Current weapon, level and Str are some things which effect the value of Wc. The value of Wc may range between 25 and -72."],
428 [2, 1, st_ac => "Ac", -120, "Armour Class, how protected you are from being hit by any attack. Lower values are better. Ac is based on your race and is modified by the Dex and current armour worn. For characters that cannot wear armour, Ac improves as their level increases."],
429 [2, 2, st_dam => "Dam", 120, "Damage, how much damage your melee/missile attack inflicts. Higher values indicate a greater amount of damage will be inflicted with each attack."],
430 [2, 3, st_arm => "Arm", 120, "Armour, how much damage (from physical attacks) will be subtracted from successful hits made upon you. This value ranges between 0 to 99%. Current armour worn primarily determines Arm value."],
431 [2, 4, st_spd => "Spd", 10.54, "Speed, how fast you can move. The value of speed may range between nearly 0 (\"very slow\") to higher than 5 (\"lightning fast\"). Base speed is determined from the Dex and modified downward proportionally by the amount of weight carried which exceeds the Max Carry limit. The armour worn also sets the upper limit on speed."],
432 [2, 5, st_wspd => "WSp", 10.54, "Weapon Speed, how many attacks you may make per unit of time (0.120s). Higher values indicate faster attack speed. Current weapon and Dex effect the value of weapon speed."],
433 ) {
434 my ($col, $row, $id, $label, $template, $tooltip) = @$_;
435
436 $tbl->add ($col , $row, $STATWIDS->{$id} = new CFClient::UI::Label
437 font => $FONT_FIXED, can_hover => 1, can_events => 1, valign => 0, align => +1, template => $template, tooltip => $tooltip);
438 $tbl->add ($col + 1, $row, $STATWIDS->{"$id\_lbl"} = new CFClient::UI::Label
439 font => $FONT_FIXED, can_hover => 1, can_events => 1, fg => $black, valign => 0, align => -1, text => $label, tooltip => $tooltip);
440 }
441
442 $hb->add (my $tbl2 = new CFClient::UI::Table expand => 1);
443
444 my $row = 0;
445 my $col = 0;
446
447 my %resist_names = (
448 slow => "Slow (slows you down when you are hit by the spell. Monsters will have an opportunity to come near you faster and hit you more often.)",
449 holyw => "Holy Word (resistance you against getting the fear when someone whose god doesn't like you spells the holy word on you.)",
450 conf => "Confusion (If you are hit by confusion you will move into random directions, and likely into monsters.)",
451 fire => "Fire (just your resistance to fire spells like burning hands, dragonbreath, meteor swarm fire, ...)",
452 depl => "Depletion (some monsters and other effects can cause stats depletion)",
453 magic => "Magic (resistance to magic spells like magic missile or similar)",
454 drain => "Draining (some monsters (e.g. vampires) and other effects can steal experience)",
455 acid => "Acid (resistance to acid, acid hurts pretty much and also corrodes your weapons)",
456 pois => "Poison (resistance to getting poisoned)",
457 para => "Paralysation (this resistance affects the chance you get paralysed)",
458 deat => "Death (resistance against death spells)",
459 phys => "Physical (this is the resistance against physical attacks, like when a monster hit you in melee combat)",
460 blind => "Blind (blind resistance affects the chance of a successful blinding attack)",
461 fear => "Fear (this attack will drive you away from monsters who cast this and hit you successfully, being resistant to this helps a lot when fighting those monsters)",
462 tund => "Turn undead",
463 elec => "Electricity (resistance againt electricity, spells like large lightning, small lightning, ...)",
464 cold => "Cold (this is your resistance against cold spells like icestorm, snowstorm, ...)",
465 ghit => "Ghost hit (special attack used by ghosts and ghost-like beings)",
466 );
467 for (qw/slow holyw conf fire depl magic
468 drain acid pois para deat phys
469 blind fear tund elec cold ghit/)
470 {
471 $tbl2->add ($col, $row,
472 $STATWIDS->{"res_$_"} =
473 new CFClient::UI::Label
474 font => $FONT_FIXED,
475 template => "-100%",
476 align => +1,
477 valign => 0,
478 can_events => 1,
479 can_hover => 1,
480 tooltip => $resist_names{$_},
481 );
482 $tbl2->add ($col + 1, $row, new CFClient::UI::Image
483 font => $FONT_FIXED,
484 can_hover => 1,
485 can_events => 1,
486 image => "ui/resist/resist_$_.png",
487 tooltip => $resist_names{$_},
488 );
489
490 $row++;
491 if ($row % 6 == 0) {
492 $col += 2;
493 $row = 0;
494 }
495 }
496
497 &set_stats_window_fontsize;
498 update_stats_window ({});
499
500 $tgw
501}
502
503sub formsep {
504 reverse join ",", grep length, split /(...)/, reverse $_[0] * 1
505}
506
507sub update_stats_window {
508 my ($stats) = @_;
509
510 # i love text protocols!!!
511 my $hp = $stats->{Crossfire::Protocol::CS_STAT_HP} * 1;
512 my $hp_m = $stats->{Crossfire::Protocol::CS_STAT_MAXHP} * 1;
513 my $sp = $stats->{Crossfire::Protocol::CS_STAT_SP} * 1;
514 my $sp_m = $stats->{Crossfire::Protocol::CS_STAT_MAXSP} * 1;
515 my $fo = $stats->{Crossfire::Protocol::CS_STAT_FOOD} * 1;
516 my $fo_m = 999;
517 my $gr = $stats->{Crossfire::Protocol::CS_STAT_GRACE} * 1;
518 my $gr_m = $stats->{Crossfire::Protocol::CS_STAT_MAXGRACE} * 1;
519
520 $GAUGES->{hp} ->set_value ($hp, $hp_m);
521 $GAUGES->{mana} ->set_value ($sp, $sp_m);
522 $GAUGES->{food} ->set_value ($fo, $fo_m);
523 $GAUGES->{grace} ->set_value ($gr, $gr_m);
524 $GAUGES->{exp} ->set_text ("Exp: " . (formsep $stats->{Crossfire::Protocol::CS_STAT_EXP64})
525 . " (lvl " . ($stats->{Crossfire::Protocol::CS_STAT_LEVEL} * 1) . ")");
526 my $rng = $stats->{Crossfire::Protocol::CS_STAT_RANGE};
527 $rng =~ s/^Range: //; # thank you so much dear server
528 $GAUGES->{range} ->set_text ("Rng: " . $rng);
529 my $title = $stats->{Crossfire::Protocol::CS_STAT_TITLE};
530 $title =~ s/^Player: //;
531 $STATWIDS->{title} ->set_text ("Title: " . $title);
532
533 $STATWIDS->{st_str} ->set_text (sprintf "%d", $stats->{5});
534 $STATWIDS->{st_dex} ->set_text (sprintf "%d", $stats->{8});
535 $STATWIDS->{st_con} ->set_text (sprintf "%d", $stats->{9});
536 $STATWIDS->{st_int} ->set_text (sprintf "%d", $stats->{6});
537 $STATWIDS->{st_wis} ->set_text (sprintf "%d", $stats->{7});
538 $STATWIDS->{st_pow} ->set_text (sprintf "%d", $stats->{22});
539 $STATWIDS->{st_cha} ->set_text (sprintf "%d", $stats->{10});
540 $STATWIDS->{st_wc} ->set_text (sprintf "%d", $stats->{13});
541 $STATWIDS->{st_ac} ->set_text (sprintf "%d", $stats->{14});
542 $STATWIDS->{st_dam} ->set_text (sprintf "%d", $stats->{15});
543 $STATWIDS->{st_arm} ->set_text (sprintf "%d", $stats->{16});
544 $STATWIDS->{st_spd} ->set_text (sprintf "%.1f", $stats->{Crossfire::Protocol::CS_STAT_SPEED});
545 $STATWIDS->{st_wspd}->set_text (sprintf "%.1f", $stats->{Crossfire::Protocol::CS_STAT_WEAP_SP});
546
547 my %tbl = (
548 phys => 100,
549 magic => 101,
550 fire => 102,
551 elec => 103,
552 cold => 104,
553 conf => 105,
554 acid => 106,
555 drain => 107,
556 ghit => 108,
557 pois => 109,
558 slow => 110,
559 para => 111,
560 tund => 112,
561 fear => 113,
562 depl => 113,
563 deat => 115,
564 holyw => 116,
565 blind => 117
566 );
567
568 for (keys %tbl) {
569 $STATWIDS->{"res_$_"}->set_text (sprintf "%d%", $stats->{$tbl{$_}});
570 }
571
572}
573
574sub metaserver_dialog {
575 my $dialog = new CFClient::UI::FancyFrame
576 title => "Metaserver",
577 child => (my $vbox = new CFClient::UI::VBox);
578
579 $vbox->add ($dialog->{table} = new CFClient::UI::Table);
580
581 $dialog
582}
583
584my $METASERVER_ATIME;
585
586sub update_metaserver {
587 my ($HOST) = @_;
588
589 return if $METASERVER_ATIME > time;
590 $METASERVER_ATIME = time + 60;
591
592 my $table = $METASERVER->{table};
593 $table->clear;
594 $table->add (0, 0, my $label = new CFClient::UI::Label max_w => $WIDTH * 0.8, text => "fetching server list...");
595
596 my $buf;
597
598 my $fh = new IO::Socket::INET PeerHost => $META_SERVER, Blocking => 0;
599
600 unless ($fh) {
601 $label->set_text ("unable to contact metaserver: $!");
602 return;
603 }
604
605 Event->io (fd => $fh, poll => 'r', cb => sub {
606 my $res = sysread $fh, $buf, 8192, length $buf;
607
608 if (!defined $res) {
609 $_[0]->w->cancel;
610 $label->set_text ("error while retrieving server list: $!");
611 } elsif ($res == 0) {
612 $_[0]->w->cancel;
613 status "server list retrieved";
614
615 utf8::decode $buf if utf8::valid $buf;
616
617 $table->clear;
618
619 my @col = qw(Use #Users Host Uptime Version Description);
620 $table->add ($_, 0, new CFClient::UI::Label align => 0, fg => [1, 1, 0], text => $col[$_])
621 for 0 .. $#col;
622
623 my @align = qw(1 0 1 1 -1);
624
625 my $y = 0;
626 for my $m (sort { $b->[3] <=> $a->[3] } map [split /\|/], split /\015?\012/, $buf) {
627 my ($ip, $last, $host, $users, $version, $desc, $ibytes, $obytes, $uptime) = @$m;
628
629 for ($desc) {
630 s/<br>/\n/gi;
631 s/<li>/\n· /gi;
632 s/<.*?>//sgi;
633 s/&/&amp;/g;
634 s/</&lt;/g;
635 s/>/&gt;/g;
636 }
637
638 $uptime = sprintf "%dd %02d:%02d:%02d",
639 (int $m->[8] / 86400),
640 (int $m->[8] / 3600) % 24,
641 (int $m->[8] / 60) % 60,
642 $m->[8] % 60;
643
644 $m = [$users, $host, $uptime, $version, $desc];
645
646 $y++;
647
648 $table->add (0, $y, new CFClient::UI::VBox children => [
649 (new CFClient::UI::Button text => "Use", connect_activate => sub {
650 $HOST->set_text ($CFG->{host} = $host);
651 }),
652 (new CFClient::UI::Empty expand => 1),
653 ]);
654
655 $table->add ($_ + 1, $y, new CFClient::UI::Label align => $align[$_], text => $m->[$_], fontsize => 0.8)
656 for 0 .. $#$m;
657 }
658 }
659 });
660}
661
662sub server_setup {
663 my $dialog = new CFClient::UI::FancyFrame
664 title => "Server Setup",
665 child => (my $vbox = new CFClient::UI::VBox);
666
667 $vbox->add (my $table = new CFClient::UI::Table expand => 1, col_expand => [0, 1]);
668 $table->add (0, 2, new CFClient::UI::Label valign => 0, align => 1, text => "Host:Port");
669
670 {
671 $table->add (1, 2, my $vbox = new CFClient::UI::VBox);
672
673 $vbox->add (
674 my $HOST = new CFClient::UI::Entry
675 expand => 1,
676 text => $CFG->{host},
677 tooltip => "The hostname or ip address of the Crossfire(+) server to connect to",
678 connect_changed => sub {
679 my ($self, $value) = @_;
680 $CFG->{host} = $value;
681 }
682 );
683
684 $METASERVER = metaserver_dialog;
685
686 $vbox->add (new CFClient::UI::Flopper
687 expand => 1,
688 text => "Metaserver",
689 other => $METASERVER,
690 tooltip => "Show a list of avaible crossfire servers",
691 connect_open => sub {
692 update_metaserver $HOST;
693 }
694 );
695 }
696
697 $table->add (0, 4, new CFClient::UI::Label valign => 0, align => 1, text => "Username");
698 $table->add (1, 4, new CFClient::UI::Entry
699 text => $CFG->{user},
700 tooltip => "The name of your character on the server",
701 connect_changed => sub {
702 my ($self, $value) = @_;
703 $CFG->{user} = $value;
704 }
705 );
706
707 $table->add (0, 5, new CFClient::UI::Label valign => 0, align => 1, text => "Password");
708 $table->add (1, 5, new CFClient::UI::Entry
709 text => $CFG->{password},
710 hidden => 1,
711 tooltip => "The password for your character",
712 connect_changed => sub {
713 my ($self, $value) = @_;
714 $CFG->{password} = $value;
715 }
716 );
717
718 $table->add (0, 7, new CFClient::UI::Label valign => 0, align => 1, text => "Map Size");
719 $table->add (1, 7, new CFClient::UI::Slider
720 req_w => 100,
721 range => [$CFG->{mapsize}, 10, 100 + 1, 1],
722 tooltip => "This is the size of the portion of the map update the server sends you. "
723 ."If you set this to a high value you will be able to see further for example.",
724 connect_changed => sub {
725 my ($self, $value) = @_;
726
727 $CFG->{mapsize} = $self->{range}[0] = $value = int $value;
728 },
729 );
730
731 $table->add (1, 8, new CFClient::UI::Button expand => 1, align => 0, text => "Login", connect_activate => sub {
732 start_game;
733 });
734
735 $dialog
736}
737
738sub message_window {
739 my $window = new CFClient::UI::FancyFrame
740 title => "Messages",
741 border_bg => [1, 1, 1, 1],
742 bg => [0, 0, 0, 0.5],
743 user_w => int $::WIDTH / 3,
744 user_h => int $::HEIGHT / 5,
745 child => (my $vbox = new CFClient::UI::VBox);
746
747 $vbox->add ($LOGVIEW = new CFClient::UI::TextView
748 expand => 1,
749 font => $FONT_FIXED,
750 fontsize => $::CFG->{log_fontsize},
751 );
752
753 $vbox->add (my $input = new CFClient::UI::Entry
754 connect_focus_in => sub {
755 my ($input, $prev_focus) = @_;
756
757 delete $input->{refocus_map};
758
759 if ($prev_focus == $MAPWIDGET && $input->{auto_activated}) {
760 $input->{refocus_map} = 1;
761 }
762 delete $input->{auto_activated};
763 },
764 connect_activate => sub {
765 my ($input, $text) = @_;
766 $input->set_text ('');
767
768 if ($text =~ /^\/(.*)/) {
769 $::CONN->user_send ($1);
770 } else {
771 my $say_cmd = $::CFG->{say_command} || 'say';
772 $::CONN->user_send ("$say_cmd $text");
773 }
774 if ($input->{refocus_map}) {
775 delete $input->{refocus_map};
776 $MAPWIDGET->focus_in
777 }
778 },
779 connect_escape => sub {
780 $MAPWIDGET->focus_in
781 },
782 );
783
784 $CONSOLE = {
785 window => $window,
786 input => $input
787 };
788
789 $window
790}
791
792sub make_inventory_window {
793 my $invwin = new CFClient::UI::FancyFrame user_w => 300, user_h => 300, title => "Inventory";
794 $invwin->add ($INV = new CFClient::UI::Inventory expand => 1);
795 $invwin
796}
797
798sub sdl_init {
799 CFClient::SDL_Init
800 and die "SDL::Init failed!\n";
801}
802
803sub video_init {
804 sdl_init;
805
806 ($WIDTH, $HEIGHT) = @{ $SDL_MODES[$CFG->{sdl_mode}] };
807 $FULLSCREEN = $CFG->{fullscreen};
808 $FAST = $CFG->{fast};
809
810 CFClient::SDL_SetVideoMode $WIDTH, $HEIGHT, $FULLSCREEN
811 or die "SDL_SetVideoMode failed!\n";
812
813 $SDL_ACTIVE = 1;
814
815 $LAST_REFRESH = time - 0.01;
816
817 CFClient::gl_init;
818
819 $FONTSIZE = int $HEIGHT / 40 * $CFG->{gui_fontsize};
78 820
79 ############################################################################# 821 #############################################################################
80 822
81 glClearColor 0, 0, 0, 0; 823 $DEBUG_STATUS = new CFClient::UI::Label padding => 0, z => 100;
82 824 $DEBUG_STATUS->show;
83 glEnable GL_TEXTURE_2D;
84 glEnable GL_COLOR_MATERIAL;
85 glShadeModel GL_FLAT;
86 glDisable GL_DEPTH_TEST;
87 glBlendFunc GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA;
88
89 $_->() for @GL_INIT;
90
91 #############################################################################
92 825
93 $STATUS_LINE = new Crossfire::Client::Widget::Label 826 $STATUS_LINE = new CFClient::UI::Label
94 0, $HEIGHT * 59 / 60 - $FONTSIZE, 1, $FONTSIZE, 827 padding => 0,
95 ""; 828 y => $HEIGHT - $FONTSIZE * 1.8;
96 $TOPLEVEL->add ($STATUS_LINE); 829 $STATUS_LINE->show;
97 830
98 $ALT_ENTER_MESSAGE = new Crossfire::Client::Widget::Label 831 $ALT_ENTER_MESSAGE = new CFClient::UI::Label
99 0, $HEIGHT * 59 / 60, 1, $HEIGHT / 60, 832 padding => 0,
833 fontsize => 0.8,
100 "Use <b>Alt-Enter</b> to toggle fullscreen mode"; 834 markup => "Use <b>Alt-Enter</b> to toggle fullscreen mode";
101 $TOPLEVEL->add ($ALT_ENTER_MESSAGE); 835 $ALT_ENTER_MESSAGE->show;
836 $ALT_ENTER_MESSAGE->move (0, $HEIGHT - $ALT_ENTER_MESSAGE->{h});
102 837
103 # Test code #d# 838 CFClient::UI::FancyFrame->new (
104 unless ($tw) { # haha... 839 border_bg => [1, 1, 1, 192/255],
105 $tw = new Crossfire::Client::Widget::Animator; 840 bg => [1, 1, 1, 0],
106 my $lbl1 = new Crossfire::Client::Widget::Label 841 child => ($MAPMAP = new CFClient::MapWidget::MapMap),
107 0, 0, 10, $FONTSIZE, "<i>This</i> is a\n<u>TEST</u>!\nOf a themed\nFrame!"; 842 )->show;
108 my $lbl2 = new Crossfire::Client::Widget::Label
109 0, 0, 10, $FONTSIZE, "LBL2";
110 843
111 my $vb = new Crossfire::Client::Widget::VBox; 844 $MAPWIDGET = new CFClient::MapWidget;
112 my $f = new Crossfire::Client::Widget::FancyFrame; 845 $MAPWIDGET->connect (activate_console => sub {
113 my $f2 = new Crossfire::Client::Widget::FancyFrame; 846 my ($mapwidget, $preset) = @_;
114 $f->add ($lbl1);
115 $f2->add ($lbl2);
116 $vb->add ($f);
117 $vb->add ($f2, 1);
118 847
119 $tw->add ($vb); 848 if ($CONSOLE) {
120 $tw->w (400); 849 $CONSOLE->{input}->{auto_activated} = 1;
121 $tw->h (300); 850 $CONSOLE->{input}->focus_in;
122 $tw->move ($WIDTH - 200, 0);
123 $tw->moveto (0, 0);
124 $TOPLEVEL->add ($tw);
125 851
126# $f->move ($WIDTH - 200, 0); 852 if ($preset && $CONSOLE->{input}->get_text eq '') {
127# $TOPLEVEL->add ($f); 853 $CONSOLE->{input}->set_text ($preset);
128 } 854 }
129}
130
131sub start_game {
132 $SDL_TIMER = add Glib::Timeout 1000/50, sub {
133 ($SDL_CB{$SDL_EV->type} || sub { warn "unhandled event ", $SDL_EV->type })->()
134 while $SDL_EV->poll;
135
136 1 855 }
137 }; 856 });
857 $MAPWIDGET->show;
858 $MAPWIDGET->focus_in;
138 859
139 $WIDTH = $CFG->{width}; 860 $BUTTONBAR = new CFClient::UI::HBox;
140 $HEIGHT = $CFG->{height};
141 $FULLSCREEN = 0;
142 861
143 init_screen; 862 $BUTTONBAR->add (new CFClient::UI::Flopper text => "Client Setup", other => client_setup);
863 $BUTTONBAR->add (new CFClient::UI::Flopper text => "Server Setup", other => server_setup);
864 $BUTTONBAR->add (new CFClient::UI::Flopper text => "Message Window", other => message_window);
144 865
145 my $mapsize = List::Util::min 64, List::Util::max 11, int $HEIGHT * $CFG->{mapsize} * 0.01 / 32; 866 make_gauge_window->show; # XXX: this has to be set before make_stats_window as make_stats_window calls update_stats_window which updated the gauges also X-D
146 867
147 $CONN = new conn 868 $BUTTONBAR->add (new CFClient::UI::Flopper text => "Stats Window", other => make_stats_window);
148 host => $CFG->{host}, 869 $BUTTONBAR->add (new CFClient::UI::Flopper text => "Inventory", other => make_inventory_window);
149 port => $CFG->{port}, 870
150 user => $CFG->{user}, 871 $BUTTONBAR->add (new CFClient::UI::Button text => "Save Config", connect_activate => sub {
151 pass => $CFG->{password}, 872 CFClient::write_cfg "$Crossfire::VARDIR/pclientrc";
152 mapw => $mapsize, 873 status "Configuration Saved";
153 maph => $mapsize,
154 ; 874 });
155 875
156 Crossfire::Client::lowdelay fileno $CONN->{fh}; 876 $BUTTONBAR->show;
157}
158 877
159sub stop_game { 878 $BUTTONBAR->{children}[1]->emit ("activate"); # pop up server setup
160 remove Glib::Source $SDL_TIMER; 879}
161 remove Glib::Source $refresh_handler if $refresh_handler;
162 undef $refresh_handler;
163 880
881sub video_shutdown {
882 $CFClient::UI::ROOT->{children} = [];
883 undef $CFClient::UI::GRAB;
884 undef $CFClient::UI::HOVER;
164 undef $SDL_APP; 885 undef $SDL_ACTIVE;
165 undef $CONN;
166 SDL::Quit;
167} 886}
168 887
888my @bgmusic = qw(game1.ogg game2.ogg game3.ogg game5.ogg game6.ogg ross1.ogg ross2.ogg ross3.ogg ross4.ogg ross5.ogg); #d#
889my $bgmusic;#TODO#hack#d#
890
891sub audio_music_finished {
892 return unless $CFG->{bgm_enable};
893
894 # TODO: hack, do play loop and mood music
895 $bgmusic = new_from_file CFClient::MixMusic CFClient::find_rcfile "music/$bgmusic[0]";
896 $bgmusic->play (0);
897
898 push @bgmusic, shift @bgmusic;
899}
900
901sub audio_init {
902 if ($CFG->{audio_enable}) {
903 if (open my $fh, "<", CFClient::find_rcfile "sounds/config") {
904 $SDL_MIXER = !CFClient::Mix_OpenAudio;
905 CFClient::Mix_AllocateChannels 8;
906 CFClient::MixMusic::volume $CFG->{bgm_volume} * 128;
907
908 audio_music_finished;
909
910 while (<$fh>) {
911 next if /^\s*#/;
912 next if /^\s*$/;
913
914 my ($file, $volume, $event) = split /\s+/, $_, 3;
915
916 push @SOUNDS, "$volume,$file";
917
918 $AUDIO_CHUNKS{"$volume,$file"} ||= do {
919 my $chunk = new_from_file CFClient::MixChunk CFClient::find_rcfile "sounds/$file";
920 $chunk->volume ($volume * 128 / 100);
921 $chunk
922 };
923 }
924 } else {
925 status "unable to open sound config: $!";
926 }
927 }
928}
929
930sub audio_shutdown {
931 CFClient::Mix_CloseAudio if $SDL_MIXER;
932 undef $SDL_MIXER;
933 @SOUNDS = ();
934 %AUDIO_CHUNKS = ();
935}
936
937my %animate_object;
938my $animate_timer;
939
940my $want_refresh;
941my $can_refresh;
942
943my $fps = 9;
169 944
170sub force_refresh { 945sub force_refresh {
171 glViewport 0, 0, $WIDTH, $HEIGHT; 946 $fps = $fps * 0.95 + 1 / ($NOW - $LAST_REFRESH) * 0.05;
947 debug sprintf "%3.2f", $fps;
172 948
173 glMatrixMode GL_PROJECTION; 949 $want_refresh = 0;
174 glLoadIdentity; 950 $can_refresh = 0;
175 glOrtho 0, $WIDTH, $HEIGHT, 0, -6000 , 6000;
176 glMatrixMode GL_MODELVIEW;
177 951
178 glClear GL_COLOR_BUFFER_BIT; 952 $CFClient::UI::ROOT->draw;
179 953
180 $TOPLEVEL->draw; 954 CFClient::SDL_GL_SwapBuffers;
181 955
182 SDL::GLSwapBuffers; 956 $LAST_REFRESH = $NOW;
183} 957}
958
959my $refresh_watcher = Event->timer (after => 0, hard => 1, interval => 1 / $MAX_FPS, cb => sub {
960 $NOW = time;
961
962 ($SDL_CB{$_->{type}} || sub { warn "unhandled event $_->{type}" })->($_)
963 for CFClient::SDL_PollEvent;
964
965 if (%animate_object) {
966 $_->animate ($LAST_REFRESH - $NOW) for values %animate_object;
967 $want_refresh++;
968 }
969
970 if ($want_refresh) {
971 force_refresh;
972 } else {
973 $can_refresh = 1;
974 }
975});
184 976
185sub refresh { 977sub refresh {
186 $refresh_handler ||= add Glib::Idle sub { 978 $want_refresh++;
187 return unless $SDL_APP;
188
189 my $next_refresh = SDL::GetTicks;
190 my $interval = ($next_refresh - $last_refresh) * 0.001;
191 $last_refresh = $next_refresh;
192
193 force_refresh;
194 $_->animate ($interval) for grep $_, values %ANIMATE;
195
196 if (%ANIMATE) {
197 1
198 } else {
199 undef $refresh_handler;
200 0
201 }
202 };
203} 979}
204 980
205sub animation_start { 981sub animation_start {
206 my ($widget) = @_; 982 my ($widget) = @_;
207 $ANIMATE{$widget} = $widget; 983 $animate_object{$widget} = $widget;
208 Scalar::Util::weaken $ANIMATE{$widget};
209
210 refresh;
211} 984}
212 985
213sub animation_stop { 986sub animation_stop {
214 my ($widget) = @_; 987 my ($widget) = @_;
215 delete $ANIMATE{$widget}; 988 delete $animate_object{$widget};
216} 989}
217
218%SDL_CB = (
219 SDL_QUIT() => sub {
220 main_quit Gtk2;
221 },
222 SDL_VIDEORESIZE() => sub {
223 },
224 SDL_VIDEOEXPOSE() => sub {
225 refresh;
226 },
227 SDL_KEYDOWN() => sub {
228 if ($SDL_EV->key_mod & KMOD_ALT && $SDL_EV->key_sym == SDLK_RETURN) {
229 # alt-enter
230 $FULLSCREEN = !$FULLSCREEN;
231 init_screen;
232 } else {
233 Crossfire::Client::Widget::feed_sdl_key_down_event ($SDL_EV);
234 }
235 },
236 SDL_KEYUP() => sub {
237 Crossfire::Client::Widget::feed_sdl_key_up_event ($SDL_EV);
238 },
239 SDL_MOUSEMOTION() => sub {
240 my ($x, $y) = ($SDL_EV->motion_x, $SDL_EV->motion_y);
241 my $widget = $TOPLEVEL->find_widget ($x, $y);
242
243 warn "mouse $x, $y = $widget\n";
244 },
245 SDL_MOUSEBUTTONDOWN() => sub {
246 Crossfire::Client::Widget::feed_sdl_button_down_event ($SDL_EV);
247 },
248 SDL_MOUSEBUTTONUP() => sub {
249 Crossfire::Client::Widget::feed_sdl_button_up_event ($SDL_EV);
250 },
251 SDL_ACTIVEEVENT() => sub {
252 warn "active\n";#d#
253 },
254);
255 990
256@conn::ISA = Crossfire::Protocol::; 991@conn::ISA = Crossfire::Protocol::;
257 992
258sub conn::map_update { 993sub conn::stats_update {
259 my ($self, $dirty) = @_; 994 my ($self, $stats) = @_;
260 995
261 refresh; 996 update_stats_window ($stats);
997}
998
999sub conn::user_send {
1000 my ($self, $command) = @_;
1001
1002 $self->send_command ($command);
1003 status $command;
262} 1004}
263 1005
264sub conn::map_scroll { 1006sub conn::map_scroll {
265 my ($self, $dx, $dy) = @_; 1007 my ($self, $dx, $dy) = @_;
266 1008
267# refresh; 1009 $MAP->scroll ($dx, $dy);
1010}
1011
1012sub conn::feed_map1a {
1013 my ($self, $data) = @_;
1014
1015# $self->Crossfire::Protocol::feed_map1a ($data);
1016
1017 $MAP->map1a_update ($data);
1018 $MAPWIDGET->update;
1019}
1020
1021sub conn::flush_map {
1022 my ($self) = @_;
1023
1024 my $map_info = delete $self->{map_info}
1025 or return;
1026
1027 my ($hash, $x, $y, $w, $h) = @$map_info;
1028
1029 my $data = $MAP->get_rect ($x, $y, $w, $h);
1030 $MAPCACHE->put ($hash => Compress::LZF::compress $data);
1031 #warn sprintf "SAVEmap[%s] length %d\n", $hash, length $data;#d#
268} 1032}
269 1033
270sub conn::map_clear { 1034sub conn::map_clear {
271 my ($self) = @_; 1035 my ($self) = @_;
272 1036
273# refresh; 1037 $self->flush_map;
1038 delete $self->{neigh_map};
1039
1040 $MAP->clear;
1041}
1042
1043
1044sub conn::load_map($$$) {
1045 my ($self, $hash, $x, $y) = @_;
1046
1047 if (defined (my $data = $MAPCACHE->get ($hash))) {
1048 $data = Compress::LZF::decompress $data;
1049 #warn sprintf "LOADmap[%s,%d,%d] length %d\n", $hash, $x, $y, length $data;#d#
1050 for my $id ($MAP->set_rect ($x, $y, $data)) {
1051 my $data = $TILECACHE->get ($id)
1052 or next;
1053
1054 $self->set_texture ($id => $data);
1055 }
1056 }
1057}
1058
1059# this method does a "flood fill" into every tile direction
1060# it assumes that tiles are arranged in a rectangular grid,
1061# i.e. a map is the same as the left of the right map etc.
1062# failure to comply are harmless and result in display errors
1063# at worst.
1064sub conn::flood_fill {
1065 my ($self, $gx, $gy, $path, $hash, $flags) = @_;
1066
1067 # the server does not allow map paths > 6
1068 return if 7 <= length $path;
1069
1070 my ($x0, $y0, $x1, $y1) = @{$self->{neigh_rect}};
1071
1072 for (
1073 [1, 0, -1],
1074 [2, 1, 0],
1075 [3, 0, 1],
1076 [4, -1, 0],
1077 ) {
1078 my ($tile, $dx, $dy) = @$_;
1079
1080 my $gx = $gx + $dx;
1081 my $gy = $gy + $dy;
1082
1083 next unless $flags & (1 << ($tile - 1));
1084 next if $self->{neigh_grid}{$gx, $gy}++;
1085
1086 my $neigh = $self->{neigh_map}{$hash} ||= [];
1087 if (my $info = $neigh->[$tile]) {
1088 my ($flags, $x, $y, $w, $h, $hash) = @$info;
1089
1090 $self->flood_fill ($gx, $gy, "$path$tile", $hash, $flags)
1091 if $x >= $x0 && $x + $w < $x1 && $y >= $y0 && $y + $h < $y1;
1092
1093 } else {
1094 $self->send_mapinfo ("spatial $path$tile", sub {
1095 my ($mode, $flags, $x, $y, $w, $h, $hash) = @_;
1096
1097 return if $mode ne "spatial";
1098
1099 $x += $MAP->ox;
1100 $y += $MAP->oy;
1101
1102 $self->load_map ($hash, $x, $y)
1103 unless $self->{neigh_map}{$hash}[5]++;#d#
1104
1105 $neigh->[$tile] = [$flags, $x, $y, $w, $h, $hash];
1106
1107 $self->flood_fill ($gx, $gy, "$path$tile", $hash, $flags)
1108 if $x >= $x0 && $x + $w < $x1 && $y >= $y0 && $y + $h < $y1;
1109 });
1110 }
1111 }
1112}
1113
1114sub conn::map_change {
1115 my ($self, $mode, $flags, $x, $y, $w, $h, $hash) = @_;
1116
1117 $self->flush_map;
1118
1119 my ($ox, $oy) = ($::MAP->ox, $::MAP->oy);
1120
1121 my $mapmapw = $MAPMAP->{w};
1122 my $mapmaph = $MAPMAP->{h};
1123
1124 $self->{neigh_rect} = [
1125 $ox - $mapmapw * 0.5, $oy - $mapmapw * 0.5,
1126 $ox + $mapmapw * 0.5 + $w, $oy + $mapmapw * 0.5 + $h,
1127 ];
1128
1129 delete $self->{neigh_grid};
1130 $self->flood_fill (0, 0, "", $hash, $flags);
1131
1132 $x += $ox;
1133 $y += $oy;
1134
1135 $self->{map_info} = [$hash, $x, $y, $w, $h];
1136
1137 my $map = $self->{map_info}[0];
1138 $map =~ s/^.*?\/([^\/]+)$/\1/;
1139 $STATWIDS->{map}->set_text ("Map: " . $map);
1140
1141 $self->load_map ($hash, $x, $y);
274} 1142}
275 1143
276sub conn::face_find { 1144sub conn::face_find {
277 my ($self, $face) = @_; 1145 my ($self, $facenum, $face) = @_;
278 1146
279 $FACECACHE->{"$face->{chksum},$face->{name}"} 1147 my $hash = "$face->{chksum},$face->{name}";
1148
1149 my $id = $FACEMAP->get ($hash);
1150
1151 unless ($id) {
1152 # create new id for face
1153 # i love transactions
1154 for (1..100) {
1155 my $txn = $CFClient::DB_ENV->txn_begin;
1156 my $status = $FACEMAP->db_get (id => $id, BerkeleyDB::DB_RMW);
1157 if ($status == 0 || $status == BerkeleyDB::DB_NOTFOUND) {
1158 $id++;
1159 if ($FACEMAP->put (id => $id) == 0
1160 && $FACEMAP->put ($hash => $id) == 0) {
1161 $txn->txn_commit;
1162
1163 goto gotid;
1164 }
1165 }
1166 $txn->abort;
1167 }
1168
1169 CFClient::fatal "maximum number of transaction retries reached - database problems?";
1170 }
1171
1172gotid:
1173 $face->{id} = $id;
1174 $MAP->set_face ($facenum => $id);
1175 $self->{faceid}[$facenum] = $id;#d#
1176 $TILECACHE->get ($id)
280} 1177}
281 1178
282sub conn::face_update { 1179sub conn::face_update {
283 my ($self, $face) = @_; 1180 my ($self, $facenum, $face) = @_;
284 1181
285 $FACECACHE->{"$face->{chksum},$face->{name}"} = $face->{image}; 1182 $TILECACHE->put ($face->{id} => $face->{image}); #TODO: try to avoid duplicate writes
286 1183
287 $face->{texture} = new_from_image Crossfire::Client::Texture delete $face->{image}; 1184 $self->set_texture ($face->{id} => delete $face->{image});
288} 1185}
1186
1187sub conn::set_texture {
1188 my ($self, $id, $data) = @_;
1189
1190 $self->{texture}[$id] ||= do {
1191 my $tex =
1192 new_from_image CFClient::Texture
1193 $data, minify => 1, mipmap => 1;
1194
1195 $MAP->set_texture ($id, @$tex{qw(name w h s t)}, @{$tex->{minified}});
1196 $MAPWIDGET->update;
1197
1198 $tex
1199 };
1200}
1201
1202sub conn::sound_play {
1203 my ($self, $x, $y, $soundnum, $type) = @_;
1204
1205 $SDL_MIXER
1206 or return;
1207
1208 my $chunk = $AUDIO_CHUNKS{$SOUNDS[$soundnum]}
1209 or return;
1210
1211 $chunk->play;
1212# warn "sound $x,$y,$soundnum,$type\n";#d#
1213}
1214
1215my $LAST_QUERY; # server is stupid, stupid, stupid
289 1216
290sub conn::query { 1217sub conn::query {
291 my ($self, $flags, $prompt) = @_; 1218 my ($self, $flags, $prompt) = @_;
292 1219
293 warn "<<<<QUERY:$flags:$prompt>>>\n";#d# 1220 $prompt = $LAST_QUERY unless length $prompt;
294} 1221 $LAST_QUERY = $prompt;
295 1222
296sub gtk_add_cfg_field { 1223 my $dialog = new CFClient::UI::FancyFrame
297 my ($tbl, $cfg, $klbl, $key, $value) = @_; 1224 title => "Query",
298 my $i = $cfg->{_i}++; 1225 child => my $vbox = new CFClient::UI::VBox;
299 $tbl->attach_defaults (my $lbl = Gtk2::Label->new ($klbl), 0, 1, $i, $i + 1); 1226
300 $tbl->attach_defaults (my $ent = Gtk2::Entry->new, 1, 2, $i, $i + 1); 1227 $vbox->add (new CFClient::UI::Label
301 if ($key eq 'password') { 1228 max_w => $::WIDTH * 0.4,
302 $ent->set_invisible_char ("*"); 1229 text => $prompt);
303 $ent->set (visibility => 0) 1230
304 } 1231 if ($flags & Crossfire::Protocol::CS_QUERY_YESNO) {
305 $ent->set_text ($value); 1232 $vbox->add (my $hbox = new CFClient::HBox);
306 $ent->signal_connect (changed => sub { 1233 $hbox->add (new CFClient::Button
307 my ($ent) = @_; 1234 text => "No",
308 $cfg->{$key} = $ent->get_text; 1235 connect_activate => sub {
309 # TODO: Mapsize should be a slider in the game gui 1236 $self->send ("reply n");
310 if ($key eq 'mapsize' and $cfg->{$key} > 100) { 1237 $dialog->destroy;
311 $cfg->{$key} = 100; 1238 $MAPWIDGET->focus_in;
312 } elsif ($key eq 'mapsize' and $cfg->{$key} < 50) {
313 $cfg->{$key} = 50;
314 } 1239 }
315 }); 1240 );
316} 1241 $hbox->add (new CFClient::Button
317 1242 text => "Yes",
318sub run_config_dialog { 1243 connect_activate => sub {
319 my (%events) = @_; 1244 $self->send ("reply y");
320 1245 $dialog->destroy;
321 my $w = Gtk2::Window->new; 1246 $MAPWIDGET->focus_in;
322
323 my @cfg = (
324 [qw/Host host/],
325 [qw/Port port/],
326 [qw/Mapsize% mapsize/],
327 [qw/Username user/],
328 [qw/Password password/],
329 );
330
331 my $cfg = {};
332
333 my $a = SDL::ListModes (0, SDL_FULLSCREEN|SDL_HWSURFACE);
334 my @modes = map { [SDL::RectW ($_), SDL::RectH ($_)] } @$a;
335
336 $w->add (my $vb = Gtk2::VBox->new);
337 $vb->pack_start (my $t = Gtk2::Table->new (2, scalar @cfg), 0, 0, 0);
338 my $selmode = $::CFG->{width} . 'x' . $::CFG->{height};
339 $t->attach_defaults (Gtk2::Label->new ("Modes"), 0, 1, 0, 1);
340 $t->attach_defaults (my $cb = Gtk2::ComboBox->new_text, 1, 2, 0, 1);
341 my $i = 0;
342 my $act = 0;
343 for (map { "$_->[0]x$_->[1]" } reverse @modes) {
344 if ($_ eq $selmode) { $act = $i }
345 $cb->append_text ($_);
346 $i++;
347 }
348 $cb->set_active ($act);
349 $cb->signal_connect (changed => sub {
350 my ($cb) = @_;
351 my $txt = $cb->get_active_text;
352 if ($txt =~ m/(\d+)x(\d+)/) {
353 $::CFG->{width} = $1;
354 $::CFG->{height} = $2;
355 }
356 });
357
358 $cfg->{_i} = 1;
359 for (@cfg) {
360 gtk_add_cfg_field ($t, $cfg, $_->[0], $_->[1], $::CFG->{$_->[1]});
361 } 1247 },
1248 );
362 1249
363 $vb->pack_start (my $hb = Gtk2::HBox->new, 0, 0, 0); 1250 $dialog->focus_in;
364 $hb->pack_start (my $cb = Gtk2::Button->new ("save"), 1, 1, 5); 1251
365 $cb->signal_connect (clicked => sub { 1252 } elsif ($flags & Crossfire::Protocol::CS_QUERY_SINGLECHAR) {
366 for (keys %$cfg) { 1253 $dialog->{tooltip} = "Press a key (click on the entry to make sure it has keyboard focus)";
367 $::CFG->{$_} = $cfg->{$_} 1254 $vbox->add (my $entry = new CFClient::UI::Entry
368 if $_ ne '_i'; 1255 connect_changed => sub {
369 } 1256 $self->send ("reply $_[1]");
370 Crossfire::Client::write_cfg "$Crossfire::VARDIR/pclientrc"; 1257 $dialog->destroy;
1258 $MAPWIDGET->focus_in;
371 }); 1259 },
372 $hb->pack_start (my $cb = Gtk2::Button->new ("login"), 1, 1, 5); 1260 );
373 $cb->signal_connect (clicked => sub { 1261
374 for (keys %$cfg) { 1262 $entry->focus_in;
375 $::CFG->{$_} = $cfg->{$_} 1263
376 if $_ ne '_i'; 1264 } else {
377 } 1265 $dialog->{tooltip} = "Enter the reply and press return (click on the entry to make sure it has keyboard focus)";
378 my $cb = $events{login} || sub {}; 1266
379 $cb->($::CFG->{user}, $::CFG->{password}); 1267 $vbox->add (my $entry = new CFClient::UI::Entry
1268 $flags & Crossfire::Protocol::CS_QUERY_HIDEINPUT ? (hiddenchar => "*") : (),
1269 connect_activate => sub {
1270 $self->send ("reply $_[1]");
1271 $dialog->destroy;
1272 $MAPWIDGET->focus_in;
380 }); 1273 },
381 $hb->pack_start (my $cb = Gtk2::Button->new ("logout"), 1, 1, 5); 1274 );
382 $cb->signal_connect (clicked => sub {
383 my $cb = $events{logout} || sub {};
384 $cb->();
385 });
386 $hb->pack_start (my $cb = Gtk2::Button->new ("quit"), 1, 1, 5);
387 $cb->signal_connect (clicked => sub { $w->destroy });
388 1275
389 $w->show_all; 1276 $entry->focus_in;
1277 }
390 1278
391 $w->signal_connect (destroy => sub { Gtk2->main_quit }); 1279 $dialog->show;
392} 1280}
393 1281
1282sub conn::drawinfo {
1283 my ($self, $color, $text) = @_;
1284
1285 my @color = (
1286 [1.00, 1.00, 1.00], #[0.00, 0.00, 0.00],
1287 [1.00, 1.00, 1.00],
1288 [0.50, 0.50, 1.00], #[0.00, 0.00, 0.55]
1289 [1.00, 0.00, 0.00],
1290 [1.00, 0.54, 0.00],
1291 [0.11, 0.56, 1.00],
1292 [0.93, 0.46, 0.00],
1293 [0.18, 0.54, 0.34],
1294 [0.56, 0.73, 0.56],
1295 [0.80, 0.80, 0.80],
1296 [0.55, 0.41, 0.13],
1297 [0.99, 0.77, 0.26],
1298 [0.74, 0.65, 0.41],
1299 );
1300
1301 $LOGVIEW->add_paragraph ($color[$color], $text);
1302}
1303
1304sub conn::spell_add {
1305 my ($self, $spell) = @_;
1306
1307 # TODO
1308 # create a widget dynamically, using spell face (CF::Protocol downloads them)
1309 $MAPWIDGET->add_command ("invoke $spell->{name}", $spell->{message});
1310 $MAPWIDGET->add_command ("cast $spell->{name}", $spell->{message});
1311}
1312
1313sub conn::spell_delete {
1314 my ($self, $spell) = @_;
1315}
1316
1317sub conn::addme_success {
1318 my ($self) = @_;
1319
1320 for my $skill (values %{$self->{skill_info}}) {
1321 $MAPWIDGET->add_command ("ready_skill $skill", "Ready the skill '$skill'");
1322 $MAPWIDGET->add_command ("use_skill $skill", "Immediately use the skill '$skill'");
1323 }
1324}
1325
1326sub update_floorbox {
1327 $CFClient::UI::ROOT->on_refresh ($FLOORBOX => sub {
1328 $FLOORBOX->clear;
1329 $FLOORBOX->add (new CFClient::UI::Empty expand => 1);
1330
1331 # we basically have to use the same sorting as everybody else
1332 for my $item (@{ $CONN->{container}{0} }) {
1333 $FLOORBOX->add (new CFClient::UI::InventoryItem item => $item);
1334 }
1335 });
1336 refresh;
1337}
1338
1339sub conn::container_add {
1340 my ($self, $id, $items) = @_;
1341
1342 update_floorbox if $id == 0;
1343 if ($self->{player}{tag} == $id) {
1344 $INV->set_items ($self->{container}{$self->{player}{tag}});
1345 }
1346 # $self-<{player}{tag} => player inv
1347 #use PApp::Util; warn PApp::Util::dumpval $self->{container}{$self->{player}{tag}};
1348}
1349
1350sub conn::container_clear {
1351 my ($self, $id) = @_;
1352
1353 update_floorbox if $id == 0;
1354 if ($self->{player}{tag} == $id) {
1355 $INV->set_items ($self->{container}{$id});
1356 }
1357
1358# use PApp::Util; warn PApp::Util::dumpval $self->{container}{0};
1359}
1360
1361sub conn::item_delete {
1362 my ($self, @items) = @_;
1363
1364 for (@items) {
1365 update_floorbox if $_->{container} == 0;
1366 if ($self->{player}{tag} == $_->{container}) {
1367 $INV->set_items ($self->{container}{$_->{container}});
1368 }
1369 }
1370}
1371
1372sub conn::item_update {
1373 my ($self, $item) = @_;
1374
1375 update_floorbox if $item->{container} == 0;
1376 if ($self->{player}{tag} == $item->{container}) {
1377 $INV->set_items ($self->{container}{$item->{container}});
1378 }
1379}
1380
1381%SDL_CB = (
1382 CFClient::SDL_QUIT => sub {
1383 Event::unloop -1;
1384 },
1385 CFClient::SDL_VIDEORESIZE => sub {
1386 },
1387 CFClient::SDL_VIDEOEXPOSE => \&refresh,
1388 CFClient::SDL_ACTIVEEVENT => sub {
1389# printf "active %x %x\n", $SDL_EV->active_gain, $SDL_EV->active_state;#d#
1390 },
1391 CFClient::SDL_KEYDOWN => sub {
1392 if ($_[0]{mod} & CFClient::KMOD_ALT && $_[0]{sym} == 13) {
1393 # alt-enter
1394 video_shutdown;
1395 $CFG->{fullscreen} = !$CFG->{fullscreen};
1396 video_init;
1397 } else {
1398 CFClient::UI::feed_sdl_key_down_event ($_[0]);
1399 }
1400 },
1401 CFClient::SDL_KEYUP => \&CFClient::UI::feed_sdl_key_up_event,
1402 CFClient::SDL_MOUSEMOTION => \&CFClient::UI::feed_sdl_motion_event,
1403 CFClient::SDL_MOUSEBUTTONDOWN => \&CFClient::UI::feed_sdl_button_down_event,
1404 CFClient::SDL_MOUSEBUTTONUP => \&CFClient::UI::feed_sdl_button_up_event,
1405 CFClient::SDL_USEREVENT => \&audio_music_finished,
1406);
394 1407
395############################################################################# 1408#############################################################################
396 1409
397SDL::Init SDL_INIT_EVERYTHING; 1410$SIG{INT} = $SIG{TERM} = sub { exit };
398 1411
399$TOPLEVEL = Crossfire::Client::Widget::Toplevel->new;
400
401my $mapwidget = Crossfire::Client::Widget::MapWidget->new;
402
403$TOPLEVEL->add ($mapwidget);
404$mapwidget->focus_in;
405
406Crossfire::Client::read_cfg "$Crossfire::VARDIR/pclientrc"; 1412CFClient::read_cfg "$Crossfire::VARDIR/pclientrc";
407 1413
408$CFG ||= { 1414$TILECACHE = CFClient::db_table "tilecache";
1415$FACEMAP = CFClient::db_table "facemap";
1416
1417my %DEF_CFG = (
1418 sdl_mode => 0,
409 width => 640, 1419 width => 640,
410 height => 480, 1420 height => 480,
1421 fullscreen => 0,
1422 fast => 0,
1423 map_scale => 0.5,
1424 fow_enable => 1,
1425 fow_intensity => 0.45,
1426 fow_smooth => 0,
1427 gui_fontsize => 1,
1428 log_fontsize => 1,
1429 gauge_fontsize => 1,
1430 gauge_size => 0.35,
1431 stat_fontsize => 1,
411 mapsize => 100, 1432 mapsize => 100,
412 fullscreen => 0,
413 host => "crossfire.schmorp.de", 1433 host => "crossfire.schmorp.de",
414 port => 13327, 1434 say_command => 'say',
415}; 1435 audio_enable => 1,
1436 bgm_enable => 1,
1437 bgm_volume => 0.25,
1438);
416 1439
417Crossfire::Client::set_font Crossfire::Client::find_rcfile "uifont.ttf"; 1440while (my ($k, $v) = each %DEF_CFG) {
1441 $CFG->{$k} = $v unless exists $CFG->{$k};
1442}
418 1443
419$FACECACHE = eval { Crossfire::load_ref "$Crossfire::VARDIR/pclient.faces" } || {}; 1444sdl_init;
420 1445
421run_config_dialog 1446@SDL_MODES = reverse
422 login => sub { start_game }, 1447 grep $_->[0] >= 640 && $_->[1] >= 480,
423 logout => sub { stop_game }; 1448 CFClient::SDL_ListModes;
424 1449
425main Gtk2; 1450@SDL_MODES or CFClient::fatal "Unable to find a usable video mode\n(hardware accelerated opengl fullscreen)";
426 1451
427Crossfire::save_ref $FACECACHE, "$Crossfire::VARDIR/pclient.faces"; 1452$CFG->{sdl_mode} = 0 if $CFG->{sdl_mode} > @SDL_MODES;
1453
1454{
1455 my @fonts = map CFClient::find_rcfile "fonts/$_", qw(
1456 DejaVuSans.ttf
1457 DejaVuSansMono.ttf
1458 DejaVuSans-Bold.ttf
1459 DejaVuSansMono-Bold.ttf
1460 DejaVuSans-Oblique.ttf
1461 DejaVuSansMono-Oblique.ttf
1462 DejaVuSans-BoldOblique.ttf
1463 DejaVuSansMono-BoldOblique.ttf
1464 );
1465
1466 CFClient::add_font $_ for @fonts;
1467
1468 $FONT_PROP = new_from_file CFClient::Font $fonts[0];
1469 $FONT_FIXED = new_from_file CFClient::Font $fonts[1];
1470
1471 $FONT_PROP->make_default;
1472}
1473
1474video_init;
1475audio_init;
1476
1477Event::loop;
1478
1479END { CFClient::SDL_Quit }
1480
1481=head1 pclient - Crossfire+ and Crossfire game client
1482
1483Pclient is a Crossfire+ and Crossfire game client.
1484
1485=head2 Features
1486
1487=over 4
1488
1489=item Fullscreen Map
1490
1491PClient can uses a fullscreen map, which greatly enhances how much of the
1492game world you can see.
1493
1494=item Persistent Map Cache (Crossfire+ only)
1495
1496PClient can persistently cache all map data it received from the
1497server. This not only allows it to display an overview map, but also
1498ensures that once-explored areas will be available the next time you want
1499to explore more.
1500
1501=item Hardware acceleration
1502
1503Unlike most Crossfire clients, PClient take advantage of OpenGL hardware
1504acceleration. Most modern graphics cards have difficulties with 2D
1505acceleration, while 3D graphics is accelerated well.
1506
1507=item No arbitrary limits
1508
1509Unlike other Crossfire clients, pclient does not suffer from arbitrary
1510limits (like a fixed amount of face numbers). There are still limits, but
1511they are not arbitrarily low :)
1512
1513=back
1514
1515=head1 USAGE
1516
1517=head2 The Map
1518
1519The map is always displayed in the background, behind all other windows and UI elements.
1520
1521#TODO# middle-click scrolls
1522#
1523# keys:
1524#
1525# a apply
1526# keypad moves, kp_5 applies ranged attack to self
1527
1528Starting to type enters the I<completion mode>. In that mode, you can type
1529abbreviations or commands and have them executed as soon as they match a
1530valid command. This is best explained by a few examples:
1531
1532Typing B<climb> will display a list of commands with I<climb> in their
1533name, such as I<ready_skill climbing> and I<use_skill climbing>.
1534
1535You can abbreviate commands by typing only the first character of every
1536word. For example, typing I<iwor> will likely select I<invoke word of
1537recall>, while I<ccfo> will select I<cast create food>. Likewise, I<rscli>
1538will likely select I<ready_skill climbing> and I<usl> will give you
1539I<use_skill levitation>.
1540
1541=head2 The map overview
1542
1543#TODO#
1544
1545=head2 The Status area in the lower right corner
1546
1547#TODO#
1548
1549=head2 The I<Statistics>/I>Stats> window
1550
1551#TODO#
1552
1553=head1 FAQ
1554
1555=over 4
1556
1557=item The client is very sluggish and slow, what can I do about this?
1558
1559Most likely, you don't have accelerated OpenGL support. Try to find a
1560newer driver, or a driver from your hardware vendor, that features OpenGL
1561support.
1562
1563If this is not an option, the following Setup options reduce the load and
1564will likely make the client playable with sofwtare rendering (it will
1565still be slow, though):
1566
1567=over 4
1568
1569=item B<Video Mode> should be set as low as possible (e.g. 640x480)
1570
1571=item Enable B<Fast & Ugly> mode
1572
1573=item Disable B<Fog of War>
1574
1575=item Increase B<Map Scale>
1576
1577=back
1578
1579=back
1580
1581=head1 AUTHOR
1582
1583Marc Lehmann <crossfire@schmorp.de>, Robin Redeker <elmex@ta-sa.org>
1584
1585
1586

Diff Legend

Removed lines
+ Added lines
< Changed lines
> Changed lines