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.171 by root, Mon Apr 24 03:33:51 2006 UTC vs.
Revision 1.215 by root, Mon May 15 00:15:08 2006 UTC

1#!/opt/bin/perl 1#!/opt/bin/perl
2 2
3use strict; 3use strict;
4use utf8; 4use utf8;
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};
5 28
6use Time::HiRes 'time'; 29use Time::HiRes 'time';
7use Event; 30use Event;
8 31
9use Crossfire; 32use Crossfire;
13 36
14use CFClient; 37use CFClient;
15use CFClient::UI; 38use CFClient::UI;
16use CFClient::MapWidget; 39use CFClient::MapWidget;
17 40
41$Event::DIED = sub {
42 # TODO: display dialog box or so
43 CFClient::error $_[1];
44};
45
46#$SIG{__WARN__} = sub { Carp::cluck $_[0] };#d#
47
18our $VERSION = '0.1'; 48our $VERSION = '0.1';
19 49
20my $MAX_FPS = 60; 50my $MAX_FPS = 60;
21my $MIN_FPS = 5; # unused as of yet 51my $MIN_FPS = 5; # unused as of yet
22 52
30our $NOW; 60our $NOW;
31 61
32our $CFG; 62our $CFG;
33our $CONN; 63our $CONN;
34our $FAST; # fast, low-quality mode, possibly useful for software-rendering 64our $FAST; # fast, low-quality mode, possibly useful for software-rendering
65
66our $WANT_REFRESH;
67our $CAN_REFRESH;
35 68
36our @SDL_MODES; 69our @SDL_MODES;
37our $WIDTH; 70our $WIDTH;
38our $HEIGHT; 71our $HEIGHT;
39our $FULLSCREEN; 72our $FULLSCREEN;
41 74
42our $FONT_PROP; 75our $FONT_PROP;
43our $FONT_FIXED; 76our $FONT_FIXED;
44 77
45our $MAP; 78our $MAP;
79our $MAPMAP;
46our $MAPWIDGET; 80our $MAPWIDGET;
47our $BUTTONBAR; 81our $BUTTONBAR;
48our $LOGVIEW; 82our $LOGVIEW;
49our $CONSOLE; 83our $CONSOLE;
50our $METASERVER; 84our $METASERVER;
85our $LOGIN_BUTTON;
51 86
87our $FLOORBOX;
52our $GAUGES; 88our $GAUGES;
53our $STATWIDS; 89our $STATWIDS;
54 90
55our $SDL_ACTIVE; 91our $SDL_ACTIVE;
56our %SDL_CB; 92our %SDL_CB;
58our $SDL_MIXER; 94our $SDL_MIXER;
59our @SOUNDS; # event => file mapping 95our @SOUNDS; # event => file mapping
60our %AUDIO_CHUNKS; # audio files 96our %AUDIO_CHUNKS; # audio files
61 97
62our $ALT_ENTER_MESSAGE; 98our $ALT_ENTER_MESSAGE;
63our $STATUS_LINE; 99our $STATUSBOX;
64our $DEBUG_STATUS; 100our $DEBUG_STATUS;
65 101
102our $INVWIN;
103our $INV;
104
66sub status { 105sub status {
67 $STATUS_LINE->set_text ($_[0]); 106 $STATUSBOX->add ($_[0], pri => -10, group => "status", timeout => 20, fg => [1, 1, 0, 1]);
68 $STATUS_LINE->move (0, $HEIGHT - $ALT_ENTER_MESSAGE->{h} - $STATUS_LINE->{h});
69} 107}
70 108
71sub debug { 109sub debug {
72 $DEBUG_STATUS->set_text ($_[0]); 110 $DEBUG_STATUS->set_text ($_[0]);
73 $DEBUG_STATUS->move ($WIDTH - $DEBUG_STATUS->{w}, 0, $DEBUG_STATUS->{w}, $DEBUG_STATUS->{h}); 111 my ($w, $h) = $DEBUG_STATUS->size_request;
112 $DEBUG_STATUS->move ($WIDTH - $w, 0);
74} 113}
75 114
76sub start_game { 115sub start_game {
77 status "logging in..."; 116 status "logging in...";
78 117
79 my $mapsize = List::Util::min 32, List::Util::max 11, int $WIDTH * $CFG->{mapsize} * 0.01 / 32; 118 my $mapsize = List::Util::min 32, List::Util::max 11, int $WIDTH * $CFG->{mapsize} * 0.01 / 32;
80 119
81 $MAPCACHE = CFClient::db_table "mapcache_$CFG->{host}"; 120 $MAPCACHE = CFClient::db_table "mapcache_$CFG->{host}";
82
83 $MAP = new CFClient::Map $mapsize, $mapsize; 121 $MAP = new CFClient::Map $mapsize, $mapsize;
84 122
85 my ($host, $port) = split /:/, $CFG->{host}; 123 my ($host, $port) = split /:/, $CFG->{host};
86 124
87 $CONN = new conn 125 $CONN = eval {
126 new conn
88 host => $host, 127 host => $host,
89 port => $port || 13327, 128 port => $port || 13327,
90 user => $CFG->{user}, 129 user => $CFG->{user},
91 pass => $CFG->{password}, 130 pass => $CFG->{password},
92 mapw => $mapsize, 131 mapw => $mapsize,
93 maph => $mapsize, 132 maph => $mapsize,
133 ;
94 ; 134 };
95 135
136 if ($CONN) {
137 $LOGIN_BUTTON->set_text ("Logout");
138
96 status "login successful"; 139 status "login successful";
97 140
98 CFClient::lowdelay fileno $CONN->{fh}; 141 CFClient::lowdelay fileno $CONN->{fh};
142 } else {
143 status "unable to connect";
144 stop_game();
145 }
99} 146}
100 147
101sub stop_game { 148sub stop_game {
149 return unless $CONN;
150
151 status "connection closed";
152 $LOGIN_BUTTON->set_text ("Login");
153 $CONN->destroy;
154 $CONN = 0; # false, does not autovivify
155
156 undef $MAPCACHE;
102 undef $CONN; 157 undef $MAP;
103} 158}
104 159
105sub client_setup { 160sub client_setup {
106 my $dialog = new CFClient::UI::FancyFrame 161 my $dialog = new CFClient::UI::FancyFrame
107 title => "Client Setup", 162 title => "Client Setup",
283 audio_shutdown (); 338 audio_shutdown ();
284 audio_init (); 339 audio_init ();
285 } 340 }
286 ); 341 );
287 342
343 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Communication cmd");
344 $table->add (1, $row++, my $saycmd = new CFClient::UI::Entry
345 text => $CFG->{say_command},
346 tooltip => "This is the command that will be used if you write a line in the message window entry. "
347 ."Usually you want to enter something like 'say' or 'shout' or 'gsay' here. "
348 ."But you could also set it to 'tell <playername>' to only chat with that user.",
349 connect_changed => sub {
350 my ($self, $value) = @_;
351 $CFG->{say_command} = $value;
352 }
353 );
354
288 $dialog 355 $dialog
289} 356}
290 357
291sub set_stats_window_fontsize { 358sub set_stats_window_fontsize {
292 for (values %{$STATWIDS}) { 359 for (values %{$STATWIDS}) {
302# local $GAUGES->{win}{parent};#d# 369# local $GAUGES->{win}{parent};#d#
303# use PApp::Util; open D, ">:utf8", "d"; print D PApp::Util::dumpval $GAUGES->{win}; close D; 370# use PApp::Util; open D, ">:utf8", "d"; print D PApp::Util::dumpval $GAUGES->{win}; close D;
304} 371}
305 372
306sub make_gauge_window { 373sub make_gauge_window {
307 my $gh = int ($HEIGHT * $CFG->{gauge_size}); 374 my $gh = int $HEIGHT * $CFG->{gauge_size};
308# my $gw = int ($WIDTH * $CFG->{gauge_w_size});
309 375
310 my $win = new CFClient::UI::Frame ( 376 my $win = new CFClient::UI::Frame (
311 y => $HEIGHT - $gh, x => 0, user_w => $WIDTH, user_h => $gh 377 req_y => -1,
378 user_w => $WIDTH,
379 user_h => $gh,
312 ); 380 );
381
313 $win->add (my $vb = new CFClient::UI::VBox); 382 $win->add (my $hbox = new CFClient::UI::HBox
314 383 children => [
315 $vb->add (my $hbg = new CFClient::UI::HBox expand => 1); 384 (new CFClient::UI::HBox expand => 1),
385 (new CFClient::UI::VBox children => [
386 (new CFClient::UI::Empty expand => 1),
387 (new CFClient::UI::Frame bg => [0, 0, 0, 0.4], child => ($FLOORBOX = new CFClient::UI::VBox)),
388 ]),
389 (my $vbox = new CFClient::UI::VBox),
390 ],
391 );
316 392
393 $vbox->add (new CFClient::UI::HBox
394 expand => 1,
395 children => [
317 $hbg->add (new CFClient::UI::Empty expand => 1); 396 (new CFClient::UI::Empty expand => 1),
318 $hbg->add (my $hb = new CFClient::UI::HBox); 397 (my $hb = new CFClient::UI::HBox),
398 ],
399 );
400
319 $hb->add (my $hg = new CFClient::UI::Gauge type => 'hp', tooltip => "Health points"); 401 $hb->add (my $hg = new CFClient::UI::Gauge type => 'hp',
402 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.");
320 $hb->add (my $mg = new CFClient::UI::Gauge type => 'mana', tooltip => "Spellpoints"); 403 $hb->add (my $mg = new CFClient::UI::Gauge type => 'mana',
404 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.");
321 $hb->add (my $gg = new CFClient::UI::Gauge type => 'grace', tooltip => "Grace"); 405 $hb->add (my $gg = new CFClient::UI::Gauge type => 'grace',
406 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.");
322 $hb->add (my $fg = new CFClient::UI::Gauge type => 'food', tooltip => "Food"); 407 $hb->add (my $fg = new CFClient::UI::Gauge type => 'food',
408 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.");
323 409
324 $vb->add (my $exp = new CFClient::UI::Label valign => 0, align => 1, text => "XP: 0 LVL: 0"); 410 $vbox->add (my $exp = new CFClient::UI::Label valign => 0, align => 1, can_hover => 1, can_events => 1,
411 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.");
325 $vb->add (my $rng = new CFClient::UI::Label valign => 0, align => 1, text => "Rng:"); 412 $vbox->add (my $rng = new CFClient::UI::Label valign => 0, align => 1, can_hover => 1, can_events => 1,
413 tooltip => "Ranged attack - how you attack when you press shift-cursor (spell, skill, weapon etc.)");
326 414
327 $GAUGES = { 415 $GAUGES = {
328 exp => $exp, win => $win, range => $rng, 416 exp => $exp, win => $win, range => $rng,
329 food => $fg, mana => $mg, hp => $hg, grace => $gg 417 food => $fg, mana => $mg, hp => $hg, grace => $gg
330 }; 418 };
333 421
334 $win 422 $win
335} 423}
336 424
337sub make_stats_window { 425sub make_stats_window {
338 my $tgw = new CFClient::UI::FancyFrame (x => $WIDTH * 2/5, y => 0, title => "Stats"); 426 my $tgw = new CFClient::UI::FancyFrame x => $WIDTH * 2/5, y => 0, title => "Stats";
339 427
340 $tgw->add (my $vb = new CFClient::UI::VBox); 428 $tgw->add (new CFClient::UI::Window child => my $vb = new CFClient::UI::VBox);
341 $vb->add ($STATWIDS->{title} = new CFClient::UI::Label valign => 0, align => -1, text => "Title:", expand => 1); 429 $vb->add ($STATWIDS->{title} = new CFClient::UI::Label valign => 0, align => -1, text => "Title:", expand => 1);
342 $vb->add ($STATWIDS->{map} = new CFClient::UI::Label valign => 0, align => -1, text => "Map:", expand => 1); 430 $vb->add ($STATWIDS->{map} = new CFClient::UI::Label valign => 0, align => -1, text => "Map:", expand => 1);
343 431
344 $vb->add (my $hb = new CFClient::UI::HBox expand => 1); 432 $vb->add (my $hb = new CFClient::UI::HBox expand => 1);
345 433
346 $hb->add (my $tbl = new CFClient::UI::Table expand => 1); 434 $hb->add (my $tbl = new CFClient::UI::Table expand => 1);
347 435
348 $tbl->add (0, 0, $STATWIDS->{st_str} = new CFClient::UI::Label valign => 0, align => +1, template => "30"); 436 my $black = [0, 0, 0];
349 $tbl->add (0, 1, $STATWIDS->{st_dex} = new CFClient::UI::Label valign => 0, align => +1, template => "30");
350 $tbl->add (0, 2, $STATWIDS->{st_con} = new CFClient::UI::Label valign => 0, align => +1, template => "30");
351 $tbl->add (0, 3, $STATWIDS->{st_int} = new CFClient::UI::Label valign => 0, align => +1, template => "30");
352 $tbl->add (0, 4, $STATWIDS->{st_wis} = new CFClient::UI::Label valign => 0, align => +1, template => "30");
353 $tbl->add (0, 5, $STATWIDS->{st_pow} = new CFClient::UI::Label valign => 0, align => +1, template => "30");
354 $tbl->add (0, 6, $STATWIDS->{st_cha} = new CFClient::UI::Label valign => 0, align => +1, template => "30");
355 437
356 $tbl->add (1, 0, $STATWIDS->{st_str_lbl} = new CFClient::UI::Label valign => 0, align => -1, text => "Str"); 438 for (
357 $tbl->add (1, 1, $STATWIDS->{st_dex_lbl} = new CFClient::UI::Label valign => 0, align => -1, text => "Dex"); 439 [0, 0, st_str => "Str", 30, "Physical Strength, determines damage dealt with weapons, how much you can carry, and how often you can attack"],
358 $tbl->add (1, 2, $STATWIDS->{st_con_lbl} = new CFClient::UI::Label valign => 0, align => -1, text => "Con"); 440 [0, 1, st_dex => "Dex", 30, "Dexterity, your physical agility. Determines chance of being hit and affects armor class and speed"],
359 $tbl->add (1, 3, $STATWIDS->{st_int_lbl} = new CFClient::UI::Label valign => 0, align => -1, text => "Int"); 441 [0, 2, st_con => "Con", 30, "Constitution, physical health and toughness. Determines how many healthpoints you can have"],
360 $tbl->add (1, 4, $STATWIDS->{st_wis_lbl} = new CFClient::UI::Label valign => 0, align => -1, text => "Wis"); 442 [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"],
361 $tbl->add (1, 5, $STATWIDS->{st_pow_lbl} = new CFClient::UI::Label valign => 0, align => -1, text => "Pow"); 443 [0, 4, st_wis => "Wis", 30, "Wisdom, the ability to learn and use divine magic (prayers). Determines how many grace points you can have"],
362 $tbl->add (1, 6, $STATWIDS->{st_cha_lbl} = new CFClient::UI::Label valign => 0, align => -1, text => "Cha"); 444 [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"],
445 [0, 6, st_cha => "Cha", 30, "Charisma, how well you are received by NPCs. Affects buying and selling prices in shops."],
363 446
364 $tbl->add (2, 0, $STATWIDS->{st_wc} = new CFClient::UI::Label valign => 0, align => +1, template => "-120"); 447 [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."],
365 $tbl->add (2, 1, $STATWIDS->{st_ac} = new CFClient::UI::Label valign => 0, align => +1, template => "-120"); 448 [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."],
366 $tbl->add (2, 2, $STATWIDS->{st_dam} = new CFClient::UI::Label valign => 0, align => +1, template => "120"); 449 [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."],
367 $tbl->add (2, 3, $STATWIDS->{st_arm} = new CFClient::UI::Label valign => 0, align => +1, template => "120"); 450 [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."],
368 $tbl->add (2, 4, $STATWIDS->{st_spd} = new CFClient::UI::Label valign => 0, align => +1, template => "10.54"); 451 [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."],
369 $tbl->add (2, 5, $STATWIDS->{st_wspd} = new CFClient::UI::Label valign => 0, align => +1, template => "9"); 452 [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."],
453 ) {
454 my ($col, $row, $id, $label, $template, $tooltip) = @$_;
370 455
371 $tbl->add (3, 0, $STATWIDS->{st_wc_lbl} = new CFClient::UI::Label valign => 0, align => -1, text => "Wc"); 456 $tbl->add ($col , $row, $STATWIDS->{$id} = new CFClient::UI::Label
372 $tbl->add (3, 1, $STATWIDS->{st_ac_lbl} = new CFClient::UI::Label valign => 0, align => -1, text => "Ac"); 457 font => $FONT_FIXED, can_hover => 1, can_events => 1, valign => 0, align => +1, template => $template, tooltip => $tooltip);
373 $tbl->add (3, 2, $STATWIDS->{st_dam_lbl} = new CFClient::UI::Label valign => 0, align => -1, text => "Dam"); 458 $tbl->add ($col + 1, $row, $STATWIDS->{"$id\_lbl"} = new CFClient::UI::Label
374 $tbl->add (3, 3, $STATWIDS->{st_arm_lbl} = new CFClient::UI::Label valign => 0, align => -1, text => "Arm"); 459 font => $FONT_FIXED, can_hover => 1, can_events => 1, fg => $black, valign => 0, align => -1, text => $label, tooltip => $tooltip);
375 $tbl->add (3, 4, $STATWIDS->{st_spd_lbl} = new CFClient::UI::Label valign => 0, align => -1, text => "Sp"); 460 }
376 $tbl->add (3, 5, $STATWIDS->{st_wspd_lbl} = new CFClient::UI::Label valign => 0, align => -1, text => "WSp");
377 461
378 $hb->add (my $tbl2 = new CFClient::UI::Table expand => 1); 462 $hb->add (my $tbl2 = new CFClient::UI::Table expand => 1);
379 463
380 my $row = 0; 464 my $row = 0;
381 my $col = 0; 465 my $col = 0;
382 466
383 my %resist_names = ( 467 my %resist_names = (
384 slow => "Slow", 468 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.)",
385 holyw => "Holy Word", 469 holyw => "Holy Word (resistance you against getting the fear when someone whose god doesn't like you spells the holy word on you.)",
386 conf => "Confusion", 470 conf => "Confusion (If you are hit by confusion you will move into random directions, and likely into monsters.)",
387 fire => "Fire", 471 fire => "Fire (just your resistance to fire spells like burning hands, dragonbreath, meteor swarm fire, ...)",
388 depl => "Depletion", 472 depl => "Depletion (some monsters and other effects can cause stats depletion)",
389 magic => "Magic", 473 magic => "Magic (resistance to magic spells like magic missile or similar)",
390 drain => "Draining", 474 drain => "Draining (some monsters (e.g. vampires) and other effects can steal experience)",
391 acid => "Acid", 475 acid => "Acid (resistance to acid, acid hurts pretty much and also corrodes your weapons)",
392 pois => "Poison", 476 pois => "Poison (resistance to getting poisoned)",
393 para => "Paralysation", 477 para => "Paralysation (this resistance affects the chance you get paralysed)",
394 deat => "Death", 478 deat => "Death (resistance against death spells)",
395 phys => "Physical", 479 phys => "Physical (this is the resistance against physical attacks, like when a monster hit you in melee combat)",
396 blind => "Blind", 480 blind => "Blind (blind resistance affects the chance of a successful blinding attack)",
397 fear => "Fear", 481 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)",
398 tund => "Turn undead", 482 tund => "Turn undead",
399 elec => "Electricity", 483 elec => "Electricity (resistance againt electricity, spells like large lightning, small lightning, ...)",
400 cold => "Cold", 484 cold => "Cold (this is your resistance against cold spells like icestorm, snowstorm, ...)",
401 ghit => "Ghost hit", 485 ghit => "Ghost hit (special attack used by ghosts and ghost-like beings)",
402 ); 486 );
403 for (qw/slow holyw conf fire depl magic 487 for (qw/slow holyw conf fire depl magic
404 drain acid pois para deat phys 488 drain acid pois para deat phys
405 blind fear tund elec cold ghit/) 489 blind fear tund elec cold ghit/)
406 { 490 {
407 $tbl2->add ($col, $row, 491 $tbl2->add ($col, $row,
408 $STATWIDS->{"res_$_"} = 492 $STATWIDS->{"res_$_"} =
409 new CFClient::UI::Label 493 new CFClient::UI::Label
494 font => $FONT_FIXED,
410 template => "-100%", 495 template => "-100%",
411 align => +1, 496 align => +1,
412 valign => 0, 497 valign => 0,
498 can_events => 1,
499 can_hover => 1,
413 tooltip => $resist_names{$_} 500 tooltip => $resist_names{$_},
414 ); 501 );
415 $tbl2->add ($col + 1, $row, new CFClient::UI::Image 502 $tbl2->add ($col + 1, $row, new CFClient::UI::Image
503 font => $FONT_FIXED,
416 can_hover => 1, 504 can_hover => 1,
417 can_events => 1, 505 can_events => 1,
418 image => "ui/resist/resist_$_.png", 506 image => "ui/resist/resist_$_.png",
419 tooltip => $resist_names{$_} 507 tooltip => $resist_names{$_},
420 ); 508 );
421 509
422 $row++; 510 $row++;
423 if ($row % 6 == 0) { 511 if ($row % 6 == 0) {
424 $col += 2; 512 $col += 2;
503 591
504} 592}
505 593
506sub metaserver_dialog { 594sub metaserver_dialog {
507 my $dialog = new CFClient::UI::FancyFrame 595 my $dialog = new CFClient::UI::FancyFrame
508 title => "Metaserver", 596 title => "Server List",
509 child => (my $vbox = new CFClient::UI::VBox); 597 child => (my $vbox = new CFClient::UI::VBox);
510 598
511 $vbox->add ($dialog->{table} = new CFClient::UI::Table); 599 $vbox->add ($dialog->{table} = new CFClient::UI::Table);
512 600
513 $dialog 601 $dialog
514} 602}
603
604my $METASERVER_ATIME;
515 605
516sub update_metaserver { 606sub update_metaserver {
517 my ($HOST) = @_; 607 my ($HOST) = @_;
518 608
519 status "fetching metaserver list..."; 609 return if $METASERVER_ATIME > time;
610 $METASERVER_ATIME = time + 60;
611
612 my $table = $METASERVER->{table};
613 $table->clear;
614 $table->add (0, 0, my $label = new CFClient::UI::Label max_w => $WIDTH * 0.8, text => "fetching server list...");
520 615
521 my $buf; 616 my $buf;
522 617
523 my $fh = new IO::Socket::INET PeerHost => $META_SERVER, Blocking => 0; 618 my $fh = new IO::Socket::INET PeerHost => $META_SERVER, Blocking => 0;
619
620 unless ($fh) {
621 $label->set_text ("unable to contact metaserver: $!");
622 return;
623 }
524 624
525 Event->io (fd => $fh, poll => 'r', cb => sub { 625 Event->io (fd => $fh, poll => 'r', cb => sub {
526 my $res = sysread $fh, $buf, 8192, length $buf; 626 my $res = sysread $fh, $buf, 8192, length $buf;
527 627
528 if (!defined $res) { 628 if (!defined $res) {
529 $_[0]->w->cancel; 629 $_[0]->w->cancel;
530 status "metaserver: $!"; 630 $label->set_text ("error while retrieving server list: $!");
531 } elsif ($res == 0) { 631 } elsif ($res == 0) {
532 $_[0]->w->cancel; 632 $_[0]->w->cancel;
533 status "server list retrieved"; 633 status "server list retrieved";
534 634
535 my $table = $METASERVER->{table}; 635 utf8::decode $buf if utf8::valid $buf;
536 636
537 $table->clear; 637 $table->clear;
538 638
539 my @col = qw(Use #Users Host Uptime Version Description); 639 my @col = qw(Use #Users Host Uptime Version Description);
540 $table->add ($_, 0, new CFClient::UI::Label align => 0, fg => [1, 1, 0], text => $col[$_]) 640 $table->add ($_, 0, new CFClient::UI::Label align => 0, fg => [1, 1, 0], text => $col[$_])
564 $m = [$users, $host, $uptime, $version, $desc]; 664 $m = [$users, $host, $uptime, $version, $desc];
565 665
566 $y++; 666 $y++;
567 667
568 $table->add (0, $y, new CFClient::UI::VBox children => [ 668 $table->add (0, $y, new CFClient::UI::VBox children => [
569 (new CFClient::UI::Button text => " ", connect_activate => sub { 669 (new CFClient::UI::Button text => "Use", connect_activate => sub {
570 $HOST->set_text ($CFG->{host} = $host); 670 $HOST->set_text ($CFG->{host} = $host);
571 }), 671 }),
572 (new CFClient::UI::Empty expand => 1), 672 (new CFClient::UI::Empty expand => 1),
573 ]); 673 ]);
574 674
603 703
604 $METASERVER = metaserver_dialog; 704 $METASERVER = metaserver_dialog;
605 705
606 $vbox->add (new CFClient::UI::Flopper 706 $vbox->add (new CFClient::UI::Flopper
607 expand => 1, 707 expand => 1,
608 text => "Metaserver", 708 text => "Server List",
609 other => $METASERVER, 709 other => $METASERVER,
610 tooltip => "Show a list of avaible crossfire servers", 710 tooltip => "Show a list of available crossfire servers",
611 connect_open => sub { 711 connect_open => sub {
612 update_metaserver $HOST; 712 update_metaserver $HOST;
613 } 713 }
614 ); 714 );
615 } 715 }
630 hidden => 1, 730 hidden => 1,
631 tooltip => "The password for your character", 731 tooltip => "The password for your character",
632 connect_changed => sub { 732 connect_changed => sub {
633 my ($self, $value) = @_; 733 my ($self, $value) = @_;
634 $CFG->{password} = $value; 734 $CFG->{password} = $value;
635 }
636 );
637
638 $table->add (0, 6, new CFClient::UI::Label valign => 0, align => 1, text => "Def. say cmd");
639 $table->add (1, 6, my $saycmd = new CFClient::UI::Entry
640 text => $CFG->{say_command},
641 tooltip => "This is the command that will be used if you write a line in the message window entry. "
642 ."Usually you want to enter something like 'say' or 'shout' or 'gsay' here. "
643 ."But you could also set it to 'tell <playername>' to only chat with that user.",
644 connect_changed => sub {
645 my ($self, $value) = @_;
646 $CFG->{say_command} = $value;
647 } 735 }
648 ); 736 );
649 737
650 $table->add (0, 7, new CFClient::UI::Label valign => 0, align => 1, text => "Map Size"); 738 $table->add (0, 7, new CFClient::UI::Label valign => 0, align => 1, text => "Map Size");
651 $table->add (1, 7, new CFClient::UI::Slider 739 $table->add (1, 7, new CFClient::UI::Slider
658 746
659 $CFG->{mapsize} = $self->{range}[0] = $value = int $value; 747 $CFG->{mapsize} = $self->{range}[0] = $value = int $value;
660 }, 748 },
661 ); 749 );
662 750
663 $table->add (1, 8, new CFClient::UI::Button expand => 1, align => 0, text => "Login", connect_activate => sub { 751 $table->add (1, 8, $LOGIN_BUTTON = new CFClient::UI::Button
752 expand => 1,
753 align => 0,
754 text => "Login",
755 connect_activate => sub {
756 $CONN ? stop_game
664 start_game; 757 : start_game;
758 },
665 }); 759 );
666 760
667 $dialog 761 $dialog
668} 762}
669 763
670sub message_window { 764sub message_window {
671 my $window = new CFClient::UI::FancyFrame 765 my $window = new CFClient::UI::FancyFrame
672 title => "Messages", 766 title => "Messages",
673 border_bg => [1, 1, 1, 0.5], 767 border_bg => [1, 1, 1, 1],
674 bg => [0.3, 0.3, 0.3, 0.8], 768 bg => [0, 0, 0, 0.5],
675 user_w => int $::WIDTH / 3, 769 user_w => int $::WIDTH / 3,
676 user_h => int $::HEIGHT / 5, 770 user_h => int $::HEIGHT / 5,
677 child => (my $vbox = new CFClient::UI::VBox); 771 child => (my $vbox = new CFClient::UI::VBox);
678 772
679 $vbox->add ($LOGVIEW = new CFClient::UI::TextView 773 $vbox->add ($LOGVIEW = new CFClient::UI::TextView
719 }; 813 };
720 814
721 $window 815 $window
722} 816}
723 817
818sub make_inventory_window {
819 my $invwin = new CFClient::UI::FancyFrame user_w => 300, user_h => 300, title => "Inventory";
820 $invwin->add ($INV = new CFClient::UI::Inventory expand => 1);
821 $invwin
822}
823
724sub sdl_init { 824sub sdl_init {
725 CFClient::SDL_Init 825 CFClient::SDL_Init
726 and die "SDL::Init failed!\n"; 826 and die "SDL::Init failed!\n";
727} 827}
728 828
729sub video_init { 829sub video_init {
730 sdl_init; 830 sdl_init;
731 831
832 $CFG->{sdl_mode} = 0 if $CFG->{sdl_mode} >= @SDL_MODES;
833
732 ($WIDTH, $HEIGHT) = @{ $SDL_MODES[$CFG->{sdl_mode}] }; 834 ($WIDTH, $HEIGHT) = @{ $SDL_MODES[$CFG->{sdl_mode}] };
733 $FULLSCREEN = $CFG->{fullscreen}; 835 $FULLSCREEN = $CFG->{fullscreen};
734 $FAST = $CFG->{fast}; 836 $FAST = $CFG->{fast};
735 837
736 CFClient::SDL_SetVideoMode $WIDTH, $HEIGHT, $FULLSCREEN 838 CFClient::SDL_SetVideoMode $WIDTH, $HEIGHT, $FULLSCREEN
737 or die "SDL_SetVideoMode failed!\n"; 839 or die "SDL_SetVideoMode failed!\n";
738 840
739 $SDL_ACTIVE = 1; 841 $SDL_ACTIVE = 1;
740
741 $LAST_REFRESH = time - 0.01; 842 $LAST_REFRESH = time - 0.01;
742 843
743 CFClient::gl_init; 844 CFClient::gl_init;
744 845
745 $FONTSIZE = int $HEIGHT / 40 * $CFG->{gui_fontsize}; 846 $FONTSIZE = int $HEIGHT / 40 * $CFG->{gui_fontsize};
746 847
848 $CFClient::UI::ROOT->configure (0, 0, $WIDTH, $HEIGHT);#d#
849
747 ############################################################################# 850 #############################################################################
748 851
852 unless ($DEBUG_STATUS) {
853 # create the widgets
854
749 $DEBUG_STATUS = new CFClient::UI::Label padding => 0, z => 100; 855 $DEBUG_STATUS = new CFClient::UI::Label padding => 0, z => 100, req_x => -1;
750 $DEBUG_STATUS->show; 856 $DEBUG_STATUS->show;
751 857
752 $STATUS_LINE = new CFClient::UI::Label 858 $STATUSBOX = new CFClient::UI::Statusbox;
753 padding => 0, 859 $STATUSBOX->add ("Use <b>Alt-Enter</b> to toggle fullscreen mode", pri => -100, color => [1, 1, 1, 0.8]);
754 y => $HEIGHT - $FONTSIZE * 1.8;
755 $STATUS_LINE->show;
756 860
757 $ALT_ENTER_MESSAGE = new CFClient::UI::Label 861 (new CFClient::UI::Frame
758 padding => 0, 862 bg => [0, 0, 0, 0.4],
759 fontsize => 0.8, 863 req_y => -1,
760 markup => "Use <b>Alt-Enter</b> to toggle fullscreen mode"; 864 child => $STATUSBOX,
761 $ALT_ENTER_MESSAGE->show; 865 )->show;
762 $ALT_ENTER_MESSAGE->move (0, $HEIGHT - $ALT_ENTER_MESSAGE->{h});
763 866
764 $CFClient::UI::ROOT->add ($MAPWIDGET = new CFClient::MapWidget); 867 CFClient::UI::FancyFrame->new (
765 $MAPWIDGET->focus_in; 868 border_bg => [1, 1, 1, 192/255],
869 bg => [1, 1, 1, 0],
870 child => ($MAPMAP = new CFClient::MapWidget::MapMap),
871 )->show;
872
873 $MAPWIDGET = new CFClient::MapWidget;
766 $MAPWIDGET->connect (activate_console => sub { 874 $MAPWIDGET->connect (activate_console => sub {
767 my ($mapwidget, $preset) = @_; 875 my ($mapwidget, $preset) = @_;
768 876
769 if ($CONSOLE) { 877 if ($CONSOLE) {
770 $CONSOLE->{input}->{auto_activated} = 1; 878 $CONSOLE->{input}->{auto_activated} = 1;
771 $CONSOLE->{input}->focus_in; 879 $CONSOLE->{input}->focus_in;
772 880
773 if ($preset && $CONSOLE->{input}->get_text eq '') { 881 if ($preset && $CONSOLE->{input}->get_text eq '') {
774 $CONSOLE->{input}->set_text ($preset); 882 $CONSOLE->{input}->set_text ($preset);
883 }
775 } 884 }
776 } 885 });
777 }); 886 $MAPWIDGET->show;
887 $MAPWIDGET->focus_in;
778 888
779 $CFClient::UI::ROOT->add ($BUTTONBAR = new CFClient::UI::HBox); 889 $BUTTONBAR = new CFClient::UI::HBox;
780 890
781 $BUTTONBAR->add (new CFClient::UI::Flopper text => "Client Setup", other => client_setup); 891 $BUTTONBAR->add (new CFClient::UI::Flopper text => "Client Setup", other => client_setup);
782 $BUTTONBAR->add (new CFClient::UI::Flopper text => "Server Setup", other => server_setup); 892 $BUTTONBAR->add (new CFClient::UI::Flopper text => "Server Setup", other => server_setup);
783 $BUTTONBAR->add (new CFClient::UI::Flopper text => "Message Window", other => message_window); 893 $BUTTONBAR->add (new CFClient::UI::Flopper text => "Message Window", other => message_window);
784 894
785 $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 895 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
896
786 $BUTTONBAR->add (new CFClient::UI::Flopper text => "Stats Window", other => make_stats_window); 897 $BUTTONBAR->add (new CFClient::UI::Flopper text => "Stats Window", other => make_stats_window);
898 $BUTTONBAR->add (new CFClient::UI::Flopper text => "Inventory", other => make_inventory_window);
787 899
788 $BUTTONBAR->add (new CFClient::UI::Button text => "Save Config", connect_activate => sub { 900 $BUTTONBAR->add (new CFClient::UI::Button text => "Save Config", connect_activate => sub {
789 CFClient::write_cfg "$Crossfire::VARDIR/pclientrc"; 901 CFClient::write_cfg "$Crossfire::VARDIR/pclientrc";
790 status "Configuration Saved"; 902 status "Configuration Saved";
791 }); 903 });
792 904
905 $BUTTONBAR->show;
906
907 $STATUSBOX->add ("Set video mode $WIDTH×$HEIGHT", timeout => 10, fg => [1, 1, 1, 0.5]);
908
909 # delay till geometry is constant
910 $CFClient::UI::ROOT->on_post_alloc (startup => sub {
793 $BUTTONBAR->{children}[1]->emit ("activate"); # pop up server setup 911 $BUTTONBAR->{children}[1]->emit ("activate"); # pop up server setup
912 my $widget = $GAUGES->{win};
913 $widget->move (0, $HEIGHT - $widget->{h});#d# to in toplevel
914 });
915 force_refresh ();
916 }
794} 917}
795 918
796sub video_shutdown { 919sub video_shutdown {
797 $CFClient::UI::ROOT->{children} = [];
798 undef $SDL_ACTIVE; 920 undef $SDL_ACTIVE;
799} 921}
800 922
801my @bgmusic = qw(game1.ogg game2.ogg game3.ogg game5.ogg game6.ogg ross1.ogg ross2.ogg ross3.ogg ross4.ogg ross5.ogg); #d# 923my @bgmusic = qw(game1.ogg game2.ogg game3.ogg game5.ogg game6.ogg ross1.ogg ross2.ogg ross3.ogg ross4.ogg ross5.ogg); #d#
802my $bgmusic;#TODO#hack#d# 924my $bgmusic;#TODO#hack#d#
925
926sub audio_channel_finished {
927 my ($channel) = @_;
928
929 warn "channel $channel finished\n";#d#
930}
803 931
804sub audio_music_finished { 932sub audio_music_finished {
805 return unless $CFG->{bgm_enable}; 933 return unless $CFG->{bgm_enable};
806 934
807 # TODO: hack, do play loop and mood music 935 # TODO: hack, do play loop and mood music
811 push @bgmusic, shift @bgmusic; 939 push @bgmusic, shift @bgmusic;
812} 940}
813 941
814sub audio_init { 942sub audio_init {
815 if ($CFG->{audio_enable}) { 943 if ($CFG->{audio_enable}) {
816 if (open my $fh, "<:utf8", CFClient::find_rcfile "sounds/config") { 944 if (open my $fh, "<", CFClient::find_rcfile "sounds/config") {
817 $SDL_MIXER = !CFClient::Mix_OpenAudio; 945 $SDL_MIXER = !CFClient::Mix_OpenAudio;
818 CFClient::Mix_AllocateChannels 8; 946 CFClient::Mix_AllocateChannels 8;
819 CFClient::MixMusic::volume $CFG->{bgm_volume} * 128; 947 CFClient::MixMusic::volume $CFG->{bgm_volume} * 128;
820 948
821 audio_music_finished; 949 audio_music_finished;
848} 976}
849 977
850my %animate_object; 978my %animate_object;
851my $animate_timer; 979my $animate_timer;
852 980
853my $want_refresh;
854my $can_refresh;
855
856my $fps = 9; 981my $fps = 9;
857 982
858sub force_refresh { 983sub force_refresh {
859 $fps = $fps * 0.95 + 1 / ($NOW - $LAST_REFRESH) * 0.05; 984 $fps = $fps * 0.95 + 1 / (($NOW - $LAST_REFRESH) || 0.1) * 0.05;
860 debug sprintf "%3.2f", $fps; 985 debug sprintf "%3.2f", $fps;
861 986
862 $want_refresh = 0;
863 $can_refresh = 0;
864
865 $CFClient::UI::ROOT->draw; 987 $CFClient::UI::ROOT->draw;
866
867 CFClient::SDL_GL_SwapBuffers; 988 CFClient::SDL_GL_SwapBuffers;
868 989
990 $WANT_REFRESH = 0;
991 $CAN_REFRESH = 0;
869 $LAST_REFRESH = $NOW; 992 $LAST_REFRESH = $NOW;
870} 993}
871 994
872my $refresh_watcher = Event->timer (after => 0, hard => 1, interval => 1 / $MAX_FPS, cb => sub { 995my $refresh_watcher = Event->timer (after => 0, hard => 1, interval => 1 / $MAX_FPS, cb => sub {
873 $NOW = time; 996 $NOW = time;
875 ($SDL_CB{$_->{type}} || sub { warn "unhandled event $_->{type}" })->($_) 998 ($SDL_CB{$_->{type}} || sub { warn "unhandled event $_->{type}" })->($_)
876 for CFClient::SDL_PollEvent; 999 for CFClient::SDL_PollEvent;
877 1000
878 if (%animate_object) { 1001 if (%animate_object) {
879 $_->animate ($LAST_REFRESH - $NOW) for values %animate_object; 1002 $_->animate ($LAST_REFRESH - $NOW) for values %animate_object;
880 $want_refresh++; 1003 $WANT_REFRESH++;
881 } 1004 }
882 1005
883 if ($want_refresh) { 1006 if ($WANT_REFRESH) {
884 force_refresh; 1007 force_refresh;
885 } else { 1008 } else {
886 $can_refresh = 1; 1009 $CAN_REFRESH = 1;
887 } 1010 }
888}); 1011});
889
890sub refresh {
891 $want_refresh++;
892}
893 1012
894sub animation_start { 1013sub animation_start {
895 my ($widget) = @_; 1014 my ($widget) = @_;
896 $animate_object{$widget} = $widget; 1015 $animate_object{$widget} = $widget;
897} 1016}
976# at worst. 1095# at worst.
977sub conn::flood_fill { 1096sub conn::flood_fill {
978 my ($self, $gx, $gy, $path, $hash, $flags) = @_; 1097 my ($self, $gx, $gy, $path, $hash, $flags) = @_;
979 1098
980 # the server does not allow map paths > 6 1099 # the server does not allow map paths > 6
981 return if 6 <= length $path; 1100 return if 7 <= length $path;
982 1101
983 my ($x0, $y0, $x1, $y1) = @{$self->{neigh_rect}}; 1102 my ($x0, $y0, $x1, $y1) = @{$self->{neigh_rect}};
984 1103
985 for ( 1104 for (
986 [1, 0, -1], 1105 [1, 0, -1],
1029 1148
1030 $self->flush_map; 1149 $self->flush_map;
1031 1150
1032 my ($ox, $oy) = ($::MAP->ox, $::MAP->oy); 1151 my ($ox, $oy) = ($::MAP->ox, $::MAP->oy);
1033 1152
1034 my $mapmapw = 250; 1153 my $mapmapw = $MAPMAP->{w};
1035 my $mapmaph = 250; 1154 my $mapmaph = $MAPMAP->{h};
1036 1155
1037 $self->{neigh_rect} = [ 1156 $self->{neigh_rect} = [
1038 $ox - $mapmapw * 0.5, $oy - $mapmapw * 0.5, 1157 $ox - $mapmapw * 0.5, $oy - $mapmapw * 0.5,
1039 $ox + $mapmapw * 0.5 + $w, $oy + $mapmapw * 0.5 + $h, 1158 $ox + $mapmapw * 0.5 + $w, $oy + $mapmapw * 0.5 + $h,
1040 ]; 1159 ];
1083 } 1202 }
1084 1203
1085gotid: 1204gotid:
1086 $face->{id} = $id; 1205 $face->{id} = $id;
1087 $MAP->set_face ($facenum => $id); 1206 $MAP->set_face ($facenum => $id);
1207 $self->{faceid}[$facenum] = $id;#d#
1088 $TILECACHE->get ($id) 1208 $TILECACHE->get ($id)
1089} 1209}
1090 1210
1091sub conn::face_update { 1211sub conn::face_update {
1092 my ($self, $facenum, $face) = @_; 1212 my ($self, $facenum, $face) = @_;
1100 my ($self, $id, $data) = @_; 1220 my ($self, $id, $data) = @_;
1101 1221
1102 $self->{texture}[$id] ||= do { 1222 $self->{texture}[$id] ||= do {
1103 my $tex = 1223 my $tex =
1104 new_from_image CFClient::Texture 1224 new_from_image CFClient::Texture
1105 $data, minify => 1; 1225 $data, minify => 1, mipmap => 1;
1106 1226
1107 $MAP->set_texture ($id, @$tex{qw(name w h s t)}, @{$tex->{minified}}); 1227 $MAP->set_texture ($id, @$tex{qw(name w h s t)}, @{$tex->{minified}});
1108 $MAPWIDGET->update; 1228 $MAPWIDGET->update;
1109 1229
1110 $tex 1230 $tex
1208 [0.55, 0.41, 0.13], 1328 [0.55, 0.41, 0.13],
1209 [0.99, 0.77, 0.26], 1329 [0.99, 0.77, 0.26],
1210 [0.74, 0.65, 0.41], 1330 [0.74, 0.65, 0.41],
1211 ); 1331 );
1212 1332
1333 my $time = sprintf "%02d:%02d:%02d", (localtime time)[2,1,0];
1334
1335 $text =~ s/&/&amp;/g; $text =~ s/</&lt;/g;
1336 $text =~ s/\[b\](.*?)\[\/b\]/<b>\1<\/b>/g;
1337 $text =~ s/\[color=(.*?)\](.*?)\[\/color\]/<span foreground='\1'>\2<\/span>/g;
1338
1213 $LOGVIEW->add_paragraph ($color[$color], $text); 1339 $LOGVIEW->add_paragraph ($color[$color],
1340 join "\n", map "$time $_", split /\n/, $text);
1341
1342 $STATUSBOX->add ($text,
1343 group => $text,
1344 fg => $color[$color],
1345 timeout => 60,
1346 tooltip_font => $::FONT_FIXED,
1347 );
1348}
1349
1350sub conn::drawextinfo {
1351 my ($self, $color, $type, $subtype, $message) = @_;
1352
1353 $self->drawinfo ($color, $message);
1214} 1354}
1215 1355
1216sub conn::spell_add { 1356sub conn::spell_add {
1217 my ($self, $spell) = @_; 1357 my ($self, $spell) = @_;
1218 1358
1231 1371
1232 for my $skill (values %{$self->{skill_info}}) { 1372 for my $skill (values %{$self->{skill_info}}) {
1233 $MAPWIDGET->add_command ("ready_skill $skill", "Ready the skill '$skill'"); 1373 $MAPWIDGET->add_command ("ready_skill $skill", "Ready the skill '$skill'");
1234 $MAPWIDGET->add_command ("use_skill $skill", "Immediately use the skill '$skill'"); 1374 $MAPWIDGET->add_command ("use_skill $skill", "Immediately use the skill '$skill'");
1235 } 1375 }
1376
1377 $MAPWIDGET->add_command ("pet\\_mode defend", "Tell pets to stay close to you and defend you");
1378 $MAPWIDGET->add_command ("pet\\_mode arena", "Same as petmode attack, but also attack other players");
1379 $MAPWIDGET->add_command ("pet\\_mode sad", "Search &amp; Destroy - tell pets to roam about and attack enemies");
1380 $MAPWIDGET->add_command ("kill\\_pets", "kill your pets");
1381}
1382
1383sub conn::eof {
1384 stop_game;
1385}
1386
1387sub update_floorbox {
1388 $CFClient::UI::ROOT->on_refresh ($FLOORBOX => sub {
1389 return unless $CONN;
1390
1391 $FLOORBOX->clear;
1392 $FLOORBOX->add (new CFClient::UI::Empty expand => 1);
1393
1394 my $count = 4;
1395 for (@{ $CONN->{container}{0} }) {
1396 if (--$count) {
1397 $FLOORBOX->add (new CFClient::UI::InventoryItem item => $_);
1398 } else {
1399 $FLOORBOX->add (new CFClient::UI::Label text => "More...");
1400 last;
1401 }
1402 }
1403 });
1404
1405 $WANT_REFRESH++;
1236} 1406}
1237 1407
1238sub conn::container_add { 1408sub conn::container_add {
1239 my ($self, $id, $items) = @_; 1409 my ($self, $tag, $items) = @_;
1240 1410
1241 # 0 floor 1411 update_floorbox if $tag == 0;
1412
1413 $INV->set_items ($self->{container}{$self->{player}{tag}})
1414 if $tag == $self->{player}{tag};
1415
1242 # $self-<{player}{tag} => player inv 1416 # $self-<{player}{tag} => player inv
1243 #use PApp::Util; warn PApp::Util::dumpval $self->{container}{$self->{player}{tag}}; 1417 #use PApp::Util; warn PApp::Util::dumpval $self->{container}{$self->{player}{tag}};
1244} 1418}
1245 1419
1246sub conn::container_clear { 1420sub conn::container_clear {
1247 my ($self, $id) = @_; 1421 my ($self, $tag) = @_;
1422
1423 update_floorbox if $tag == 0;
1424
1425 $INV->set_items ($self->{container}{$tag})
1426 if $tag == $self->{player}{tag};
1427
1248# use PApp::Util; warn PApp::Util::dumpval $self->{container}{0}; 1428# use PApp::Util; warn PApp::Util::dumpval $self->{container}{0};
1429}
1430
1431sub conn::item_delete {
1432 my ($self, @items) = @_;
1433
1434 for (@items) {
1435 update_floorbox if $_->{container} == 0;
1436
1437 $INV->set_items ($self->{container}{$_->{container}})
1438 if $_->{container} == $self->{player}{tag};
1439 }
1440}
1441
1442sub conn::item_update {
1443 my ($self, $item) = @_;
1444
1445 update_floorbox if $item->{container} == 0;
1446
1447 $INV->set_items ($self->{container}{$item->{container}})
1448 if $item->{container} == $self->{player}{tag};
1249} 1449}
1250 1450
1251%SDL_CB = ( 1451%SDL_CB = (
1252 CFClient::SDL_QUIT => sub { 1452 CFClient::SDL_QUIT => sub {
1253 Event::unloop -1; 1453 Event::unloop -1;
1254 }, 1454 },
1255 CFClient::SDL_VIDEORESIZE => sub { 1455 CFClient::SDL_VIDEORESIZE => sub {
1256 }, 1456 },
1257 CFClient::SDL_VIDEOEXPOSE => \&refresh, 1457 CFClient::SDL_VIDEOEXPOSE => sub {
1458 $WANT_REFRESH++;
1459 },
1258 CFClient::SDL_ACTIVEEVENT => sub { 1460 CFClient::SDL_ACTIVEEVENT => sub {
1259# printf "active %x %x\n", $SDL_EV->active_gain, $SDL_EV->active_state;#d# 1461# printf "active %x %x\n", $SDL_EV->active_gain, $SDL_EV->active_state;#d#
1260 }, 1462 },
1261 CFClient::SDL_KEYDOWN => sub { 1463 CFClient::SDL_KEYDOWN => sub {
1262 if ($_[0]{mod} & CFClient::KMOD_ALT && $_[0]{sym} == 13) { 1464 if ($_[0]{mod} & CFClient::KMOD_ALT && $_[0]{sym} == 13) {
1266 video_init; 1468 video_init;
1267 } else { 1469 } else {
1268 CFClient::UI::feed_sdl_key_down_event ($_[0]); 1470 CFClient::UI::feed_sdl_key_down_event ($_[0]);
1269 } 1471 }
1270 }, 1472 },
1271 CFClient::SDL_KEYUP => \&CFClient::UI::feed_sdl_key_up_event, 1473 CFClient::SDL_KEYUP => \&CFClient::UI::feed_sdl_key_up_event,
1272 CFClient::SDL_MOUSEMOTION => \&CFClient::UI::feed_sdl_motion_event, 1474 CFClient::SDL_MOUSEMOTION => \&CFClient::UI::feed_sdl_motion_event,
1273 CFClient::SDL_MOUSEBUTTONDOWN => \&CFClient::UI::feed_sdl_button_down_event, 1475 CFClient::SDL_MOUSEBUTTONDOWN => \&CFClient::UI::feed_sdl_button_down_event,
1274 CFClient::SDL_MOUSEBUTTONUP => \&CFClient::UI::feed_sdl_button_up_event, 1476 CFClient::SDL_MOUSEBUTTONUP => \&CFClient::UI::feed_sdl_button_up_event,
1275 CFClient::SDL_USEREVENT => \&audio_music_finished, 1477 CFClient::SDL_USEREVENT => sub {
1478 if ($_[0]{code} == 1) {
1479 audio_channel_finished $_[0]{data1};
1480 } elsif ($_[0]{code} == 0) {
1481 audio_music_finished;
1482 }
1483 },
1276); 1484);
1277 1485
1278############################################################################# 1486#############################################################################
1279 1487
1280$SIG{INT} = $SIG{TERM} = sub { exit }; 1488$SIG{INT} = $SIG{TERM} = sub { exit };
1281 1489
1282$TILECACHE = CFClient::db_table "tilecache";
1283$FACEMAP = CFClient::db_table "facemap";
1284
1285CFClient::read_cfg "$Crossfire::VARDIR/pclientrc";
1286
1287my %DEF_CFG = (
1288 sdl_mode => 0,
1289 width => 640,
1290 height => 480,
1291 fullscreen => 0,
1292 fast => 0,
1293 map_scale => 0.5,
1294 fow_enable => 1,
1295 fow_intensity => 0.45,
1296 fow_smooth => 0,
1297 gui_fontsize => 1,
1298 log_fontsize => 1,
1299 gauge_fontsize => 1,
1300 gauge_size => 0.35,
1301 stat_fontsize => 1,
1302 mapsize => 100,
1303 host => "crossfire.schmorp.de",
1304 say_command => 'say',
1305 audio_enable => 1,
1306 bgm_enable => 1,
1307 bgm_volume => 0.25,
1308);
1309
1310while (my ($k, $v) = each %DEF_CFG) {
1311 $CFG->{$k} = $v unless exists $CFG->{$k};
1312}
1313
1314sdl_init;
1315
1316@SDL_MODES = reverse
1317 grep $_->[0] >= 640 && $_->[1] >= 480,
1318 CFClient::SDL_ListModes;
1319
1320@SDL_MODES or CFClient::fatal "Unable to find a usable video mode\n(hardware accelerated opengl fullscreen)";
1321
1322$CFG->{sdl_mode} = 0 if $CFG->{sdl_mode} > @SDL_MODES;
1323
1324{ 1490{
1491 local $SIG{__DIE__} = sub { CFClient::fatal $_[0] };
1492
1493 CFClient::read_cfg "$Crossfire::VARDIR/pclientrc";
1494
1495 $TILECACHE = CFClient::db_table "tilecache";
1496 $FACEMAP = CFClient::db_table "facemap";
1497
1498 my %DEF_CFG = (
1499 sdl_mode => 0,
1500 width => 640,
1501 height => 480,
1502 fullscreen => 0,
1503 fast => 0,
1504 map_scale => 0.5,
1505 fow_enable => 1,
1506 fow_intensity => 0.45,
1507 fow_smooth => 0,
1508 gui_fontsize => 1,
1509 log_fontsize => 1,
1510 gauge_fontsize=> 1,
1511 gauge_size => 0.35,
1512 stat_fontsize => 1,
1513 mapsize => 100,
1514 host => "crossfire.schmorp.de",
1515 say_command => 'say',
1516 audio_enable => 1,
1517 bgm_enable => 1,
1518 bgm_volume => 0.25,
1519 );
1520
1521 while (my ($k, $v) = each %DEF_CFG) {
1522 $CFG->{$k} = $v unless exists $CFG->{$k};
1523 }
1524
1525 sdl_init;
1526
1527 @SDL_MODES = reverse
1528 grep $_->[0] >= 640 && $_->[1] >= 480,
1529 CFClient::SDL_ListModes;
1530
1531 @SDL_MODES or CFClient::fatal "Unable to find a usable video mode\n(hardware accelerated opengl fullscreen)";
1532
1533 $CFG->{sdl_mode} = 0 if $CFG->{sdl_mode} > @SDL_MODES;
1534
1535 {
1325 my @fonts = map CFClient::find_rcfile "fonts/$_", qw( 1536 my @fonts = map CFClient::find_rcfile "fonts/$_", qw(
1326 DejaVuSans.ttf 1537 DejaVuSans.ttf
1327 DejaVuSansMono.ttf 1538 DejaVuSansMono.ttf
1328 DejaVuSans-Bold.ttf 1539 DejaVuSans-Bold.ttf
1329 DejaVuSansMono-Bold.ttf 1540 DejaVuSansMono-Bold.ttf
1330 DejaVuSans-Oblique.ttf 1541 DejaVuSans-Oblique.ttf
1331 DejaVuSansMono-Oblique.ttf 1542 DejaVuSansMono-Oblique.ttf
1332 DejaVuSans-BoldOblique.ttf 1543 DejaVuSans-BoldOblique.ttf
1333 DejaVuSansMono-BoldOblique.ttf 1544 DejaVuSansMono-BoldOblique.ttf
1334 ); 1545 );
1335 1546
1336 CFClient::add_font $_ for @fonts; 1547 CFClient::add_font $_ for @fonts;
1337 1548
1549 CFClient::pango_init;
1550
1338 $FONT_PROP = new_from_file CFClient::Font $fonts[0]; 1551 $FONT_PROP = new_from_file CFClient::Font $fonts[0];
1339 $FONT_FIXED = new_from_file CFClient::Font $fonts[1]; 1552 $FONT_FIXED = new_from_file CFClient::Font $fonts[1];
1340 1553
1341 $FONT_PROP->make_default; 1554 $FONT_PROP->make_default;
1342} 1555 }
1343 1556
1344video_init; 1557 video_init;
1345audio_init; 1558 audio_init;
1559}
1346 1560
1347Event::loop; 1561Event::loop;
1348 1562
1349END { CFClient::SDL_Quit } 1563END { CFClient::SDL_Quit }
1350 1564
1565=head1 pclient - Crossfire+ and Crossfire game client
1351 1566
1567Pclient is a Crossfire+ and Crossfire game client.
1568
1569=head2 Features
1570
1571=over 4
1572
1573=item Fullscreen Map
1574
1575PClient can uses a fullscreen map, which greatly enhances how much of the
1576game world you can see.
1577
1578=item Persistent Map Cache (Crossfire+ only)
1579
1580PClient can persistently cache all map data it received from the
1581server. This not only allows it to display an overview map, but also
1582ensures that once-explored areas will be available the next time you want
1583to explore more.
1584
1585=item Hardware acceleration
1586
1587Unlike most Crossfire clients, PClient take advantage of OpenGL hardware
1588acceleration. Most modern graphics cards have difficulties with 2D
1589acceleration, while 3D graphics is accelerated well.
1590
1591=item No arbitrary limits
1592
1593Unlike other Crossfire clients, pclient does not suffer from arbitrary
1594limits (like a fixed amount of face numbers). There are still limits, but
1595they are not arbitrarily low :)
1596
1597=back
1598
1599=head1 USAGE
1600
1601=head2 The Map
1602
1603The map is always displayed in the background, behind all other windows and UI elements.
1604
1605#TODO# middle-click scrolls
1606#
1607# keys:
1608#
1609# a apply
1610# keypad moves, kp_5 applies ranged attack to self
1611
1612Starting to type enters the I<completion mode>. In that mode, you can type
1613abbreviations or commands and have them executed as soon as they match a
1614valid command. This is best explained by a few examples:
1615
1616Typing B<climb> will display a list of commands with I<climb> in their
1617name, such as I<ready_skill climbing> and I<use_skill climbing>.
1618
1619You can abbreviate commands by typing only the first character of every
1620word. For example, typing I<iwor> will likely select I<invoke word of
1621recall>, while I<ccfo> will select I<cast create food>. Likewise, I<rscli>
1622will likely select I<ready_skill climbing> and I<usl> will give you
1623I<use_skill levitation>.
1624
1625=head2 The map overview
1626
1627#TODO#
1628
1629=head2 The Status area in the lower right corner
1630
1631#TODO#
1632
1633=head2 The I<Statistics>/I>Stats> window
1634
1635#TODO#
1636
1637=head1 FAQ
1638
1639=over 4
1640
1641=item The client is very sluggish and slow, what can I do about this?
1642
1643Most likely, you don't have accelerated OpenGL support. Try to find a
1644newer driver, or a driver from your hardware vendor, that features OpenGL
1645support.
1646
1647If this is not an option, the following Setup options reduce the load and
1648will likely make the client playable with sofwtare rendering (it will
1649still be slow, though):
1650
1651=over 4
1652
1653=item B<Video Mode> should be set as low as possible (e.g. 640x480)
1654
1655=item Enable B<Fast & Ugly> mode
1656
1657=item Disable B<Fog of War>
1658
1659=item Increase B<Map Scale>
1660
1661=back
1662
1663=back
1664
1665=head1 AUTHOR
1666
1667Marc Lehmann <crossfire@schmorp.de>, Robin Redeker <elmex@ta-sa.org>
1668
1669
1670

Diff Legend

Removed lines
+ Added lines
< Changed lines
> Changed lines