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.163 by root, Sun Apr 23 00:57:40 2006 UTC vs.
Revision 1.248 by root, Thu May 25 01:26:54 2006 UTC

1#!/opt/bin/perl 1#!/opt/bin/perl
2 2
3use strict; 3use strict;
4use utf8; 4use utf8;
5 5
6# do things only needed for single-binary version (par)
7BEGIN {
8 if (%PAR::LibCache) {
9 @INC = grep ref, @INC; # weed out all paths except pars loader refs
10
11 while (my ($filename, $zip) = each %PAR::LibCache) {
12 for ($zip->memberNames) {
13 next unless /^\/root\/(.*)/;
14 $zip->extractMember ($_, "$ENV{PAR_TEMP}/$1")
15 unless -e "$ENV{PAR_TEMP}/$1";
16 }
17 }
18
19 # TODO: pango-rc file, anybody?
20
21 unshift @INC, $ENV{PAR_TEMP};
22 }
23}
24
25# need to do it again because that pile of garbage called PAR nukes it before main
26unshift @INC, $ENV{PAR_TEMP}
27 if %PAR::LibCache;
28
6use Time::HiRes 'time'; 29use Time::HiRes 'time';
30use Pod::POM;
7use Event; 31use Event;
8 32
9use Crossfire; 33use Crossfire;
10use Crossfire::Protocol; 34use Crossfire::Protocol;
11 35
13 37
14use CFClient; 38use CFClient;
15use CFClient::UI; 39use CFClient::UI;
16use CFClient::MapWidget; 40use CFClient::MapWidget;
17 41
42$Event::DIED = sub {
43 # TODO: display dialog box or so
44 CFClient::error $_[1];
45};
46
47#$SIG{__WARN__} = sub { Carp::cluck $_[0] };#d#
48
18our $VERSION = '0.1'; 49our $VERSION = '0.1';
19 50
20my $MAX_FPS = 60; 51my $MAX_FPS = 60;
21my $MIN_FPS = 5; # unused as of yet 52my $MIN_FPS = 5; # unused as of yet
22 53
30our $NOW; 61our $NOW;
31 62
32our $CFG; 63our $CFG;
33our $CONN; 64our $CONN;
34our $FAST; # fast, low-quality mode, possibly useful for software-rendering 65our $FAST; # fast, low-quality mode, possibly useful for software-rendering
66
67our $WANT_REFRESH;
68our $CAN_REFRESH;
35 69
36our @SDL_MODES; 70our @SDL_MODES;
37our $WIDTH; 71our $WIDTH;
38our $HEIGHT; 72our $HEIGHT;
39our $FULLSCREEN; 73our $FULLSCREEN;
40our $FONTSIZE; 74our $FONTSIZE;
41 75
76our $FONT_PROP;
77our $FONT_FIXED;
78
42our $MAP; 79our $MAP;
80our $MAPMAP;
43our $MAPWIDGET; 81our $MAPWIDGET;
44our $BUTTONBAR; 82our $BUTTONBAR;
45our $LOGVIEW; 83our $LOGVIEW;
46our $CONSOLE; 84our $CONSOLE;
47our $METASERVER; 85our $METASERVER;
86our $LOGIN_BUTTON;
87our $QUIT_DIALOG;
48 88
89our $FLOORBOX;
49our $GAUGES; 90our $GAUGES;
50our $STATWIDS; 91our $STATWIDS;
51 92
52our $SDL_ACTIVE; 93our $SDL_ACTIVE;
53our %SDL_CB; 94our %SDL_CB;
55our $SDL_MIXER; 96our $SDL_MIXER;
56our @SOUNDS; # event => file mapping 97our @SOUNDS; # event => file mapping
57our %AUDIO_CHUNKS; # audio files 98our %AUDIO_CHUNKS; # audio files
58 99
59our $ALT_ENTER_MESSAGE; 100our $ALT_ENTER_MESSAGE;
60our $STATUS_LINE; 101our $STATUSBOX;
61our $DEBUG_STATUS; 102our $DEBUG_STATUS;
62 103
104our $INVWIN;
105our $INV;
106our $INVR;
107our $INVR_LBL;
108our $OPENCONT;
109
63sub status { 110sub status {
64 $STATUS_LINE->set_text ($_[0]); 111 $STATUSBOX->add (CFClient::UI::Label::escape $_[0], pri => -10, group => "status", timeout => 20, fg => [1, 1, 0, 1]);
65 $STATUS_LINE->move (0, $HEIGHT - $ALT_ENTER_MESSAGE->{h} - $STATUS_LINE->{h});
66} 112}
67 113
68sub debug { 114sub debug {
69 $DEBUG_STATUS->set_text ($_[0]); 115 $DEBUG_STATUS->set_text ($_[0]);
70 $DEBUG_STATUS->move ($WIDTH - $DEBUG_STATUS->{w}, 0, $DEBUG_STATUS->{w}, $DEBUG_STATUS->{h}); 116 my ($w, $h) = $DEBUG_STATUS->size_request;
117 $DEBUG_STATUS->move ($WIDTH - $w, 0);
71} 118}
72 119
73sub start_game { 120sub start_game {
74 status "logging in..."; 121 status "logging in...";
75 122
76 my $mapsize = List::Util::min 32, List::Util::max 11, int $WIDTH * $CFG->{mapsize} * 0.01 / 32; 123 my $mapsize = List::Util::min 32, List::Util::max 11, int $WIDTH * $CFG->{mapsize} * 0.01 / 32;
77 124
78 $MAPCACHE = CFClient::db_table "mapcache_$CFG->{host}"; 125 $MAPCACHE = CFClient::db_table "mapcache_$CFG->{host}";
79
80 $MAP = new CFClient::Map $mapsize, $mapsize; 126 $MAP = new CFClient::Map $mapsize, $mapsize;
81 127
82 my ($host, $port) = split /:/, $CFG->{host}; 128 my ($host, $port) = split /:/, $CFG->{host};
83 129
84 $CONN = new conn 130 $CONN = eval {
131 new conn
85 host => $host, 132 host => $host,
86 port => $port || 13327, 133 port => $port || 13327,
87 user => $CFG->{user}, 134 user => $CFG->{user},
88 pass => $CFG->{password}, 135 pass => $CFG->{password},
89 mapw => $mapsize, 136 mapw => $mapsize,
90 maph => $mapsize, 137 maph => $mapsize,
138 ;
91 ; 139 };
92 140
93 status "login successful"; 141 if ($CONN) {
94
95 CFClient::lowdelay fileno $CONN->{fh}; 142 CFClient::lowdelay fileno $CONN->{fh};
143
144 $LOGIN_BUTTON->set_text ("Logout");
145 status "login successful";
146
147 $BUTTONBAR->{children}[1]->emit ("activate")
148 if $BUTTONBAR->{children}[1]->{state};
149
150 } else {
151 status "unable to connect";
152 stop_game();
153 }
96} 154}
97 155
98sub stop_game { 156sub stop_game {
157 return unless $CONN;
158
159 status "connection closed";
160 $LOGIN_BUTTON->set_text ("Login");
161 $CONN->destroy;
162 $CONN = 0; # false, does not autovivify
163
164 $BUTTONBAR->{children}[1]->emit ("activate")
165 unless $BUTTONBAR->{children}[1]->{state};
166
167 undef $MAPCACHE;
99 undef $CONN; 168 undef $MAP;
100} 169}
101 170
102sub client_setup { 171sub client_setup {
103 my $dialog = new CFClient::UI::FancyFrame 172 my $dialog = new CFClient::UI::FancyFrame
104 title => "Client Setup", 173 title => "Client Setup",
106 $vbox->add (my $table = new CFClient::UI::Table expand => 1, col_expand => [0, 1]); 175 $vbox->add (my $table = new CFClient::UI::Table expand => 1, col_expand => [0, 1]);
107 176
108 $table->add (0, 0, new CFClient::UI::Label valign => 0, align => 1, text => "Video Mode"); 177 $table->add (0, 0, new CFClient::UI::Label valign => 0, align => 1, text => "Video Mode");
109 $table->add (1, 0, my $hbox = new CFClient::UI::HBox); 178 $table->add (1, 0, my $hbox = new CFClient::UI::HBox);
110 179
111 $hbox->add (my $mode_slider = new CFClient::UI::Slider expand => 1, req_w => 100, range => [$CFG->{sdl_mode}, 0, scalar @SDL_MODES, 1]); 180 $hbox->add (my $mode_slider = new CFClient::UI::Slider expand => 1, req_w => 100, range => [$CFG->{sdl_mode}, 0, $#SDL_MODES, 0, 1]);
112 $hbox->add (my $mode_label = new CFClient::UI::Label align => 0, valign => 0, height => 0.8, template => "9999x9999"); 181 $hbox->add (my $mode_label = new CFClient::UI::Label align => 0, valign => 0, height => 0.8, template => "9999x9999");
113 182
114 $mode_slider->connect (changed => sub { 183 $mode_slider->connect (changed => sub {
115 my ($self, $value) = @_; 184 my ($self, $value) = @_;
116 185
120 $mode_slider->emit (changed => $mode_slider->{range}[0]); 189 $mode_slider->emit (changed => $mode_slider->{range}[0]);
121 190
122 my $row = 1; 191 my $row = 1;
123 192
124 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Fullscreen"); 193 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Fullscreen");
125 $table->add (1, $row++, new CFClient::UI::CheckBox state => $CFG->{fullscreen}, connect_changed => sub { 194 $table->add (1, $row++, new CFClient::UI::CheckBox
195 state => $CFG->{fullscreen},
196 tooltip => "Bring the client into fullscreen mode",
197 connect_changed => sub {
126 my ($self, $value) = @_; 198 my ($self, $value) = @_;
127 $CFG->{fullscreen} = $value; 199 $CFG->{fullscreen} = $value;
200 }
128 }); 201 );
129 202
130 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Fast & Ugly"); 203 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Fast & Ugly");
131 $table->add (1, $row++, new CFClient::UI::CheckBox 204 $table->add (1, $row++, new CFClient::UI::CheckBox
132 state => $CFG->{fast}, 205 state => $CFG->{fast},
133 tooltip => "Lower the visual quality considerably to speed up rendering.", 206 tooltip => "Lower the visual quality considerably to speed up rendering.",
135 my ($self, $value) = @_; 208 my ($self, $value) = @_;
136 $CFG->{fast} = $value; 209 $CFG->{fast} = $value;
137 } 210 }
138 ); 211 );
139 212
213 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Map Scale");
214 $table->add (1, $row++, new CFClient::UI::Slider
215 range => [(log $CFG->{map_scale}) / (log 2), -3, 1, 0, 1],
216 tooltip => "Enlarge or shrink the displayed map",
217 connect_changed => sub {
218 my ($self, $value) = @_;
219 $CFG->{map_scale} = 2 ** $value;
220 }
221 );
222
140 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Fog of War"); 223 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Fog of War");
141 $table->add (1, $row++, new CFClient::UI::CheckBox 224 $table->add (1, $row++, new CFClient::UI::CheckBox
142 state => $CFG->{fow_enable}, 225 state => $CFG->{fow_enable},
143 tooltip => "Fog-of-War marks areas that cannot be seen by the player", 226 tooltip => "Fog-of-War marks areas that cannot be seen by the player",
144 connect_changed => sub { 227 connect_changed => sub {
147 } 230 }
148 ); 231 );
149 232
150 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "FoW Intensity"); 233 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "FoW Intensity");
151 $table->add (1, $row++, new CFClient::UI::Slider 234 $table->add (1, $row++, new CFClient::UI::Slider
152 range => [$CFG->{fow_intensity}, 0, 1 + 0.001, 0.001], 235 range => [$CFG->{fow_intensity}, 0, 1, 0, 1 / 256],
153 tooltip => "The higher the intensity, the lighter the Fog-of-War color", 236 tooltip => "The higher the intensity, the lighter the Fog-of-War color",
154 connect_changed => sub { 237 connect_changed => sub {
155 my ($self, $value) = @_; 238 my ($self, $value) = @_;
156 $CFG->{fow_intensity} = $value; 239 $CFG->{fow_intensity} = $value;
157 } 240 }
168 } 251 }
169 ); 252 );
170 253
171 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "GUI Fontsize"); 254 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "GUI Fontsize");
172 $table->add (1, $row++, new CFClient::UI::Slider 255 $table->add (1, $row++, new CFClient::UI::Slider
173 range => [$CFG->{gui_fontsize}, 0.5, 2, 0.1], 256 range => [$CFG->{gui_fontsize}, 0.5, 2, 0, 0.1],
174 tooltip => "The font size used by most GUI elements", 257 tooltip => "The font size used by most GUI elements",
175 connect_changed => sub { 258 connect_changed => sub { $CFG->{gui_fontsize} = $_[1] },
176 $CFG->{gui_fontsize} = 0.1 * int $_[1] * 10;
177# $FONTSIZE = int $HEIGHT / 40 * $CFG->{gui_fontsize};
178 }
179 ); 259 );
180 260
181 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Server Log Fontsize"); 261 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Server Log Fontsize");
182 $table->add (1, $row++, new CFClient::UI::Slider 262 $table->add (1, $row++, new CFClient::UI::Slider
183 range => [$CFG->{log_fontsize}, 0.5, 2, 0.1], 263 range => [$CFG->{log_fontsize}, 0.5, 2, 0, 0.1],
184 tooltip => "The font size used by the server log window only", 264 tooltip => "The font size used by the server log window only",
185 connect_changed => sub { 265 connect_changed => sub { $LOGVIEW->set_fontsize ($CFG->{log_fontsize} = $_[1]) },
186 $LOGVIEW->set_fontsize ($CFG->{log_fontsize} = 0.1 * int $_[1] * 10);
187 }
188 ); 266 );
189 267
190 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Stats Fontsize"); 268 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Stats Fontsize");
191 269
192 $table->add (1, $row++, new CFClient::UI::Slider 270 $table->add (1, $row++, new CFClient::UI::Slider
193 range => [$CFG->{stat_fontsize}, 0.5, 2, 0.1], 271 range => [$CFG->{stat_fontsize}, 0.5, 2, 0, 0.1],
194 tooltip => "The font size used by the statistics window only", 272 tooltip => "The font size used by the statistics window only",
195 connect_changed => sub { 273 connect_changed => sub {
196 $CFG->{stat_fontsize} = 0.1 * int $_[1] * 10; 274 $CFG->{stat_fontsize} = $_[1];
197 &set_stats_window_fontsize; 275 &set_stats_window_fontsize;
198 } 276 }
199 ); 277 );
200 278
201 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Gaugesize"); 279 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Gauge fontsize");
202 $table->add (1, $row++, new CFClient::UI::Slider range => [$CFG->{gauge_size}, 0.2, 0.8, 0.02], connect_changed => sub { 280 $table->add (1, $row++, new CFClient::UI::Slider
281 range => [$CFG->{gauge_fontsize}, 0.5, 2, 0, 0.1],
282 tooltip => "Adjusts the fontsize of the gauges at the bottom right",
283 connect_changed => sub {
203 $CFG->{gauge_size} = $_[1]; 284 $CFG->{gauge_fontsize} = $_[1];
204 my $h = int ($HEIGHT * $CFG->{gauge_size}); 285 &set_gauge_window_fontsize;
205 $GAUGES->{win}->set_size ($WIDTH, $h); 286 }
206 $GAUGES->{win}->{y} = $HEIGHT - $h;
207 $GAUGES->{win}->{x} = 0;
208 $GAUGES->{win}->update;
209 }); 287 );
210 288
211 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Gauge size"); 289 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Gauge size");
212 $table->add (1, $row++, new CFClient::UI::Slider 290 $table->add (1, $row++, new CFClient::UI::Slider
213 range => [$CFG->{gauge_size}, 0.2, 0.8, 0.02], 291 range => [$CFG->{gauge_size}, 0.2, 0.8],
214 tooltip => "Adjust the size of the stats gauges at the bottom right", 292 tooltip => "Adjust the size of the stats gauges at the bottom right",
215 connect_changed => sub { 293 connect_changed => sub {
216 $CFG->{gauge_size} = $_[1]; 294 $CFG->{gauge_size} = $_[1];
217 my $h = int ($HEIGHT * $CFG->{gauge_size}); 295 $GAUGES->{win}->set_size ($WIDTH, int $HEIGHT * $CFG->{gauge_size});
218 $GAUGES->{win}->set_size ($WIDTH, $h);
219 $GAUGES->{win}->{y} = $HEIGHT - $h;
220 $GAUGES->{win}->{x} = 0;
221 $GAUGES->{win}->update;
222 }
223 );
224
225 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Gauge fontsize");
226 $table->add (1, $row++, new CFClient::UI::Slider
227 range => [$CFG->{gauge_fontsize}, 0.5, 2.0, 0.1],
228 connect_changed => sub {
229 $CFG->{gauge_fontsize} = 0.1 * int $_[1] * 10;
230 &set_gauge_window_fontsize;
231 $GAUGES->{win}->check_size;
232 $GAUGES->{win}->update;
233 } 296 }
234 ); 297 );
235 298
236 $table->add (1, $row++, new CFClient::UI::Button 299 $table->add (1, $row++, new CFClient::UI::Button
237 expand => 1, align => 0, text => "Apply", 300 expand => 1, align => 0, text => "Apply",
238 tooltip => "Apply the video settings (unless they are auto-apply)", 301 tooltip => "Apply the video settings",
239 connect_activate => sub { 302 connect_activate => sub {
240 video_shutdown (); 303 video_shutdown ();
241 video_init (); 304 video_init ();
242 } 305 }
243 ); 306 );
262 connect_changed => sub { 325 connect_changed => sub {
263 $CFG->{bgm_enable} = $_[1]; 326 $CFG->{bgm_enable} = $_[1];
264 } 327 }
265 ); 328 );
266 $hbox->add (new CFClient::UI::Slider 329 $hbox->add (new CFClient::UI::Slider
267 expand => 1, range => [$CFG->{bgm_volume}, 0, 1, 0.1], 330 expand => 1, range => [$CFG->{bgm_volume}, 0, 1, 0, 1/128],
268 tooltip => "The volume of the background music", 331 tooltip => "The volume of the background music",
269 connect_changed => sub { 332 connect_changed => sub {
270 $CFG->{bgm_volume} = $_[1]; 333 $CFG->{bgm_volume} = $_[1];
271 CFClient::MixMusic::volume $_[1] * 128; 334 CFClient::MixMusic::volume $_[1] * 128;
272 } 335 }
273 ); 336 );
274 337
275 $table->add (1, $row++, new CFClient::UI::Button 338 $table->add (1, $row++, new CFClient::UI::Button
276 expand => 1, align => 0, text => "Apply", 339 expand => 1, align => 0, text => "Apply",
277 tooltip => "Apply the audio settings that are not auto-apply", 340 tooltip => "Apply the audio settings",
278 connect_activate => sub { 341 connect_activate => sub {
279 audio_shutdown (); 342 audio_shutdown ();
280 audio_init (); 343 audio_init ();
281 } 344 }
282 ); 345 );
283 346
347 $table->add (0, $row, new CFClient::UI::Label valign => 0, align => 1, text => "Chat Command");
348 $table->add (1, $row++, my $saycmd = new CFClient::UI::Entry
349 text => $CFG->{say_command},
350 tooltip => "This is the command that will be used if you write a line in the message window entry or press <b>\"</b> in the map window. "
351 . "Usually you want to enter something like 'say' or 'shout' or 'gsay' here. "
352 . "But you could also set it to <b>tell <i>playername</i></b> to only chat with that user.",
353 connect_changed => sub {
354 my ($self, $value) = @_;
355 $CFG->{say_command} = $value;
356 }
357 );
358
284 $dialog 359 $dialog
285} 360}
286 361
287sub set_stats_window_fontsize { 362sub set_stats_window_fontsize {
288 for (values %{$STATWIDS}) { 363 for (values %{$STATWIDS}) {
295 $_->set_fontsize ($::CFG->{gauge_fontsize}); 370 $_->set_fontsize ($::CFG->{gauge_fontsize});
296 } 371 }
297} 372}
298 373
299sub make_gauge_window { 374sub make_gauge_window {
300 my $gh = int ($HEIGHT * $CFG->{gauge_size}); 375 my $gh = int $HEIGHT * $CFG->{gauge_size};
301# my $gw = int ($WIDTH * $CFG->{gauge_w_size});
302 376
303 my $win = new CFClient::UI::Frame ( 377 my $win = new CFClient::UI::Frame (
304 y => $HEIGHT - $gh, x => 0, req_w => $WIDTH, req_h => $gh 378 req_y => -1,
379 user_w => $WIDTH,
380 user_h => $gh,
305 ); 381 );
382
306 $win->add (my $vb = new CFClient::UI::VBox); 383 $win->add (my $hbox = new CFClient::UI::HBox
307 384 children => [
308 $vb->add (my $hbg = new CFClient::UI::HBox expand => 1); 385 (new CFClient::UI::HBox expand => 1),
386 (new CFClient::UI::VBox children => [
387 (new CFClient::UI::Empty expand => 1),
388 (new CFClient::UI::Frame bg => [0, 0, 0, 0.4], child => ($FLOORBOX = new CFClient::UI::VBox)),
389 ]),
390 (my $vbox = new CFClient::UI::VBox),
391 ],
392 );
309 393
310 394 $vbox->add (new CFClient::UI::HBox
395 expand => 1,
396 children => [
311 $hbg->add (new CFClient::UI::Empty expand => 1); 397 (new CFClient::UI::Empty expand => 1),
312 $hbg->add (my $hb = new CFClient::UI::HBox); 398 (my $hb = new CFClient::UI::HBox),
399 ],
400 );
401
313 $hb->add (my $hg = new CFClient::UI::Gauge type => 'hp'); 402 $hb->add (my $hg = new CFClient::UI::Gauge type => 'hp',
403 tooltip => "<b>Health points</b>. Measures of how much damage you can take before dying. Hit points are determined from your level and are influenced by the value of your Con. Hp value may range between 1 to beyond 500 and higher values indicate a greater ability to withstand punishment.");
314 $hb->add (my $mg = new CFClient::UI::Gauge type => 'mana'); 404 $hb->add (my $mg = new CFClient::UI::Gauge type => 'mana',
405 tooltip => "<b>Spell points</b>. Measures of how much \"fuel\" you have for casting spells and incantations. Mana is calculated from your level and your Pow. Mana values can range between 1 to beyond 500 (glowing crystals can increase the current spell points beyond your normal maximum). Higher values indicate greater amounts of mana.");
315 $hb->add (my $gg = new CFClient::UI::Gauge type => 'grace'); 406 $hb->add (my $gg = new CFClient::UI::Gauge type => 'grace',
407 tooltip => "<b>Grace points</b> - how favored you are by your god. In game terms, how much divine magic you can cast. Your level, Wis and Pow effect what the value of grace is. Prayong on an altar of your god can increase this value beyond your normal maximum. Grace can take on large positive and negative values. Positive values indicate favor by the gods.");
316 $hb->add (my $fg = new CFClient::UI::Gauge type => 'food'); 408 $hb->add (my $fg = new CFClient::UI::Gauge type => 'food',
409 tooltip => "<b>Food</b>. Ranges between 0 (starving) and 999 (satiated). At a value of 0 the character begins to die. Some magic can speed up or slow down the character digestion. Healing wounds will speed up digestion too.");
317 410
318 $vb->add (my $exp = new CFClient::UI::Label valign => 0, align => 1, text => "XP:"); 411 $vbox->add (my $exp = new CFClient::UI::Label valign => 0, align => 1, can_hover => 1, can_events => 1,
319# $vb->add (my $lvl = new CFClient::UI::Label valign => 0, align => 1, text => "Lvl:"); 412 tooltip => "<b>Experience points and overall level</b> - experience is increased as a reward for appropriate action (such as killing monsters) and may decrease as a result of a magical attack or dying. Level is directly derived from the experience value. As the level of the character increases, the character becomes able to succeed at more difficult tasks. A character's level starts at a value of 0 and may range up beyond 100.");
320 $vb->add (my $rng = new CFClient::UI::Label valign => 0, align => 1, text => "Rng:"); 413 $vbox->add (my $rng = new CFClient::UI::Label valign => 0, align => 1, can_hover => 1, can_events => 1,
321 414 tooltip => "<b>Ranged attack</b> - how you attack when you press shift-cursor (spell, skill, weapon etc.)");
322 415
323 $GAUGES = { 416 $GAUGES = {
324 exp => $exp,# lvl => $lvl,
325 win => $win, range => $rng, 417 exp => $exp, win => $win, range => $rng,
326 food => $fg, mana => $mg, hp => $hg, grace => $gg 418 food => $fg, mana => $mg, hp => $hg, grace => $gg
327 }; 419 };
420
421 &set_gauge_window_fontsize;
422
328 $win 423 $win
329} 424}
330 425
331sub make_stats_window { 426sub make_stats_window {
332 my $tgw = new CFClient::UI::FancyFrame (x => $WIDTH * 2/5, y => 0, title => "Stats"); 427 my $tgw = new CFClient::UI::FancyFrame title => "Stats";
333 428
334 $tgw->add (my $vb = new CFClient::UI::VBox); 429 $tgw->add (new CFClient::UI::Window child => my $vb = new CFClient::UI::VBox);
335 $vb->add (my $uhb = new CFClient::UI::HBox);
336 $uhb->add ($STATWIDS->{title} = new CFClient::UI::Label valign => 0, align => -1, text => "Title:", expand => 1); 430 $vb->add ($STATWIDS->{title} = new CFClient::UI::Label valign => 0, align => -1, text => "Title:", expand => 1,
431 can_hover => 1, can_events => 1,
432 tooltip => "Your name and title. You can change your title by using the <b>title</b> command, if supported by the server.");
337 $uhb->add ($STATWIDS->{map} = new CFClient::UI::Label valign => 0, align => -1, text => "Map:", expand => 1); 433 $vb->add ($STATWIDS->{map} = new CFClient::UI::Label valign => 0, align => -1, text => "Map:", expand => 1,
434 can_hover => 1, can_events => 1,
435 tooltip => "The map you are currently on (if supported by the server).");
338 436
339 $vb->add (my $hb = new CFClient::UI::HBox expand => 1); 437 $vb->add (my $hb = new CFClient::UI::HBox expand => 1);
340
341 $hb->add (my $tbl = new CFClient::UI::Table expand => 1); 438 $hb->add (my $tbl = new CFClient::UI::Table expand => 1);
342 439
343 $tbl->add (0, 0, $STATWIDS->{st_str_lbl} = new CFClient::UI::Label valign => 0, align => +1, text => "Str"); 440 my $color2 = [1, 1, 0];
344 $tbl->add (0, 1, $STATWIDS->{st_dex_lbl} = new CFClient::UI::Label valign => 0, align => +1, text => "Dex");
345 $tbl->add (0, 2, $STATWIDS->{st_con_lbl} = new CFClient::UI::Label valign => 0, align => +1, text => "Con");
346 $tbl->add (0, 3, $STATWIDS->{st_int_lbl} = new CFClient::UI::Label valign => 0, align => +1, text => "Int");
347 $tbl->add (0, 4, $STATWIDS->{st_wis_lbl} = new CFClient::UI::Label valign => 0, align => +1, text => "Wis");
348 $tbl->add (0, 5, $STATWIDS->{st_pow_lbl} = new CFClient::UI::Label valign => 0, align => +1, text => "Pow");
349 $tbl->add (0, 6, $STATWIDS->{st_cha_lbl} = new CFClient::UI::Label valign => 0, align => +1, text => "Cha");
350 441
351 $tbl->add (1, 0, $STATWIDS->{st_str} = new CFClient::UI::Label valign => 0, align => +1, text => ""); 442 for (
352 $tbl->add (1, 1, $STATWIDS->{st_dex} = new CFClient::UI::Label valign => 0, align => +1, text => ""); 443 [0, 0, st_str => "Str", 30, "<b>Physical Strength</b>, determines damage dealt with weapons, how much you can carry, and how often you can attack"],
353 $tbl->add (1, 2, $STATWIDS->{st_con} = new CFClient::UI::Label valign => 0, align => +1, text => ""); 444 [0, 1, st_dex => "Dex", 30, "<b>Dexterity</b>, your physical agility. Determines chance of being hit and affects armor class and speed"],
354 $tbl->add (1, 3, $STATWIDS->{st_int} = new CFClient::UI::Label valign => 0, align => +1, text => ""); 445 [0, 2, st_con => "Con", 30, "<b>Constitution</b>, physical health and toughness. Determines how many healthpoints you can have"],
355 $tbl->add (1, 4, $STATWIDS->{st_wis} = new CFClient::UI::Label valign => 0, align => +1, text => ""); 446 [0, 3, st_int => "Int", 30, "<b>Intelligence</b>, your ability to learn and use skills and incantations (both prayers and magic) and determines how much spell points you can have"],
356 $tbl->add (1, 5, $STATWIDS->{st_pow} = new CFClient::UI::Label valign => 0, align => +1, text => ""); 447 [0, 4, st_wis => "Wis", 30, "<b>Wisdom</b>, the ability to learn and use divine magic (prayers). Determines how many grace points you can have"],
357 $tbl->add (1, 6, $STATWIDS->{st_cha} = new CFClient::UI::Label valign => 0, align => +1, text => ""); 448 [0, 5, st_pow => "Pow", 30, "<b>Power</b>, your magical potential. Influences the strength of spell effects, and also how much your spell and grace points increase when leveling up"],
449 [0, 6, st_cha => "Cha", 30, "<b>Charisma</b>, how well you are received by NPCs. Affects buying and selling prices in shops."],
358 450
359 $tbl->add (2, 0, $STATWIDS->{st_wc_lbl} = new CFClient::UI::Label valign => 0, align => +1, text => "Wc"); 451 [2, 0, st_wc => "Wc", -120, "<b>Weapon Class</b>, effectiveness of melee/missile attacks. Lower is more potent. Current weapon, level and Str are some things which effect the value of Wc. The value of Wc may range between 25 and -72."],
360 $tbl->add (2, 1, $STATWIDS->{st_ac_lbl} = new CFClient::UI::Label valign => 0, align => +1, text => "Ac"); 452 [2, 1, st_ac => "Ac", -120, "<b>Armour Class</b>, how protected you are from being hit by any attack. Lower values are better. Ac is based on your race and is modified by the Dex and current armour worn. For characters that cannot wear armour, Ac improves as their level increases."],
361 $tbl->add (2, 2, $STATWIDS->{st_dam_lbl} = new CFClient::UI::Label valign => 0, align => +1, text => "Dam"); 453 [2, 2, st_dam => "Dam", 120, "<b>Damage</b>, how much damage your melee/missile attack inflicts. Higher values indicate a greater amount of damage will be inflicted with each attack."],
362 $tbl->add (2, 3, $STATWIDS->{st_arm_lbl} = new CFClient::UI::Label valign => 0, align => +1, text => "Arm"); 454 [2, 3, st_arm => "Arm", 120, "<b>Armour</b>, how much damage (from physical attacks) will be subtracted from successful hits made upon you. This value ranges between 0 to 99%. Current armour worn primarily determines Arm value."],
363 $tbl->add (2, 4, $STATWIDS->{st_spd_lbl} = new CFClient::UI::Label valign => 0, align => +1, text => "Sp"); 455 [2, 4, st_spd => "Spd", 10.54, "<b>Speed</b>, how fast you can move. The value of speed may range between nearly 0 (\"very slow\") to higher than 5 (\"lightning fast\"). Base speed is determined from the Dex and modified downward proportionally by the amount of weight carried which exceeds the Max Carry limit. The armour worn also sets the upper limit on speed."],
364 $tbl->add (2, 5, $STATWIDS->{st_wspd_lbl} = new CFClient::UI::Label valign => 0, align => +1, text => "WSp"); 456 [2, 5, st_wspd => "WSp", 10.54, "<b>Weapon Speed</b>, how many attacks you may make per unit of time (0.120s). Higher values indicate faster attack speed. Current weapon and Dex effect the value of weapon speed."],
457 ) {
458 my ($col, $row, $id, $label, $template, $tooltip) = @$_;
365 459
366 $tbl->add (3, 0, $STATWIDS->{st_wc} = new CFClient::UI::Label valign => 0, align => +1, text => ""); 460 $tbl->add ($col , $row, $STATWIDS->{$id} = new CFClient::UI::Label
367 $tbl->add (3, 1, $STATWIDS->{st_ac} = new CFClient::UI::Label valign => 0, align => +1, text => ""); 461 font => $FONT_FIXED, can_hover => 1, can_events => 1, valign => 0, align => +1, template => $template, tooltip => $tooltip);
368 $tbl->add (3, 2, $STATWIDS->{st_dam} = new CFClient::UI::Label valign => 0, align => +1, text => ""); 462 $tbl->add ($col + 1, $row, $STATWIDS->{"$id\_lbl"} = new CFClient::UI::Label
369 $tbl->add (3, 3, $STATWIDS->{st_arm} = new CFClient::UI::Label valign => 0, align => +1, text => ""); 463 font => $FONT_FIXED, can_hover => 1, can_events => 1, fg => $color2, valign => 0, align => -1, text => $label, tooltip => $tooltip);
370 $tbl->add (3, 4, $STATWIDS->{st_spd} = new CFClient::UI::Label valign => 0, align => +1, text => ""); 464 }
371 $tbl->add (3, 5, $STATWIDS->{st_wspd} = new CFClient::UI::Label valign => 0, align => +1, text => "");
372 465
373 $hb->add (my $tbl2 = new CFClient::UI::Table expand => 1); 466 $hb->add (my $tbl2 = new CFClient::UI::Table expand => 1);
374 467
375 my $row = 0; 468 my $row = 0;
376 my $col = 0; 469 my $col = 0;
377 470
471 my %resist_names = (
472 slow => "<b>Slow</b> (slows you down when you are hit by the spell. Monsters will have an opportunity to come near you faster and hit you more often.)",
473 holyw => "<b>Holy Word</b> (resistance you against getting the fear when someone whose god doesn't like you spells the holy word on you.)",
474 conf => "<b>Confusion</b> (If you are hit by confusion you will move into random directions, and likely into monsters.)",
475 fire => "<b>Fire</b> (just your resistance to fire spells like burning hands, dragonbreath, meteor swarm fire, ...)",
476 depl => "<b>Depletion</b> (some monsters and other effects can cause stats depletion)",
477 magic => "<b>Magic</b> (resistance to magic spells like magic missile or similar)",
478 drain => "<b>Draining</b> (some monsters (e.g. vampires) and other effects can steal experience)",
479 acid => "<b>Acid</b> (resistance to acid, acid hurts pretty much and also corrodes your weapons)",
480 pois => "<b>Poison</b> (resistance to getting poisoned)",
481 para => "<b>Paralysation</b> (this resistance affects the chance you get paralysed)",
482 deat => "<b>Death</b> (resistance against death spells)",
483 phys => "<b>Physical</b> (this is the resistance against physical attacks, like when a monster hit you in melee combat)",
484 blind => "<b>Blind</b> (blind resistance affects the chance of a successful blinding attack)",
485 fear => "<b>Fear</b> (this attack will drive you away from monsters who cast this and hit you successfully, being resistant to this helps a lot when fighting those monsters)",
486 tund => "<b>Turn undead</b> (affects your resistancy to various forms of 'turn undead' spells. Only relevant when you are, in fact, undead...",
487 elec => "<b>Electricity</b> (resistance against electricity, spells like large lightning, small lightning, ...)",
488 cold => "<b>Cold</b> (this is your resistance against cold spells like icestorm, snowstorm, ...)",
489 ghit => "<b>Ghost hit</b> (special attack used by ghosts and ghost-like beings)",
490 );
378 for (qw/slow holyw conf fire depl magic 491 for (qw/slow holyw conf fire depl magic
379 drain acid pois para deat phys 492 drain acid pois para deat phys
380 blind fear tund elec cold ghit/) 493 blind fear tund elec cold ghit/)
381 { 494 {
382 $tbl2->add ($col , $row, 495 $tbl2->add ($col, $row,
383 $STATWIDS->{"res_$_"} = 496 $STATWIDS->{"res_$_"} =
384 new CFClient::UI::Label text => "0", align => +1, valign => 0 497 new CFClient::UI::Label
498 font => $FONT_FIXED,
499 template => "-100%",
500 align => +1,
501 valign => 0,
502 can_events => 1,
503 can_hover => 1,
504 tooltip => $resist_names{$_},
385 ); 505 );
386 $tbl2->add ($col + 1, $row, new CFClient::UI::Image image => "ui/resist/resist_$_.png"); 506 $tbl2->add ($col + 1, $row, new CFClient::UI::Image
507 font => $FONT_FIXED,
508 can_hover => 1,
509 can_events => 1,
510 image => "ui/resist/resist_$_.png",
511 tooltip => $resist_names{$_},
512 );
387 513
388 $row++; 514 $row++;
389 if ($row % 6 == 0) { 515 if ($row % 6 == 0) {
390 $col += 2; 516 $col += 2;
391 $row = 0; 517 $row = 0;
396 update_stats_window ({}); 522 update_stats_window ({});
397 523
398 $tgw 524 $tgw
399} 525}
400 526
527sub formsep {
528 reverse join ",", grep length, split /(...)/, reverse $_[0] * 1
529}
530
401sub update_stats_window { 531sub update_stats_window {
402 my ($stats) = @_; 532 my ($stats) = @_;
403 533
404 # i love text protocols!!! 534 # i love text protocols!!!
405 my $hp = $stats->{1} * 1; 535 my $hp = $stats->{Crossfire::Protocol::CS_STAT_HP} * 1;
406 my $hp_m = $stats->{2} * 1; 536 my $hp_m = $stats->{Crossfire::Protocol::CS_STAT_MAXHP} * 1;
407 my $sp = $stats->{3} * 1; 537 my $sp = $stats->{Crossfire::Protocol::CS_STAT_SP} * 1;
408 my $sp_m = $stats->{4} * 1; 538 my $sp_m = $stats->{Crossfire::Protocol::CS_STAT_MAXSP} * 1;
409 my $fo = $stats->{18} * 1; 539 my $fo = $stats->{Crossfire::Protocol::CS_STAT_FOOD} * 1;
410 my $fo_m = 999; 540 my $fo_m = 999;
411 my $gr = $stats->{23} * 1; 541 my $gr = $stats->{Crossfire::Protocol::CS_STAT_GRACE} * 1;
412 my $gr_m = $stats->{24} * 1; 542 my $gr_m = $stats->{Crossfire::Protocol::CS_STAT_MAXGRACE} * 1;
413 543
414 $GAUGES->{hp} ->set_value ($hp, $hp_m); 544 $GAUGES->{hp} ->set_value ($hp, $hp_m);
415 $GAUGES->{mana} ->set_value ($sp, $sp_m); 545 $GAUGES->{mana} ->set_value ($sp, $sp_m);
416 $GAUGES->{food} ->set_value ($fo, $fo_m); 546 $GAUGES->{food} ->set_value ($fo, $fo_m);
417 $GAUGES->{grace} ->set_value ($gr, $gr_m); 547 $GAUGES->{grace} ->set_value ($gr, $gr_m);
418 $GAUGES->{exp} ->set_text ("XP: " . ($stats->{11} || $stats->{28}) * 1 548 $GAUGES->{exp} ->set_text ("Exp: " . (formsep $stats->{Crossfire::Protocol::CS_STAT_EXP64})
419 ." LVL: " . $stats->{12} * 1); 549 . " (lvl " . ($stats->{Crossfire::Protocol::CS_STAT_LEVEL} * 1) . ")");
420 my $rng = $stats->{20}; 550 my $rng = $stats->{Crossfire::Protocol::CS_STAT_RANGE};
421 $rng =~ s/^Range: //; # thank you so much dear server 551 $rng =~ s/^Range: //; # thank you so much dear server
422 $GAUGES->{range} ->set_text ("Rng: " . $rng); 552 $GAUGES->{range} ->set_text ("Rng: " . $rng);
423# $GAUGES->{lvl} ->set_text ("LVL: " . $stats->{12}); 553 my $title = $stats->{Crossfire::Protocol::CS_STAT_TITLE};
554 $title =~ s/^Player: //;
424 $STATWIDS->{title} ->set_text ("Title: " . $stats->{21}); 555 $STATWIDS->{title} ->set_text ("Title: " . $title);
425 556
426 if (0) { # this code can vanish, just wanted to preserver it for a checkin
427 $STATWIDS->{st_str} ->set_text (sprintf "S%d", $stats->{5});
428 $STATWIDS->{st_dex} ->set_text (sprintf "D%d", $stats->{8});
429 $STATWIDS->{st_con} ->set_text (sprintf "Co%d", $stats->{9});
430 $STATWIDS->{st_int} ->set_text (sprintf "I%d", $stats->{6});
431 $STATWIDS->{st_wis} ->set_text (sprintf "W%d", $stats->{7});
432 $STATWIDS->{st_pow} ->set_text (sprintf "P%d", $stats->{22});
433 $STATWIDS->{st_cha} ->set_text (sprintf "Ch%d", $stats->{10});
434 $STATWIDS->{st_wc} ->set_text (sprintf "Wc%d", $stats->{13});
435 $STATWIDS->{st_ac} ->set_text (sprintf "Ac%d", $stats->{14});
436 $STATWIDS->{st_dam} ->set_text (sprintf "Dam%d", $stats->{15});
437 $STATWIDS->{st_arm} ->set_text (sprintf "Arm%d", $stats->{16});
438 $STATWIDS->{st_spd} ->set_text (sprintf "Sp%.1f", $stats->{17});
439 $STATWIDS->{st_wspd}->set_text (sprintf "WSp%.1f", $stats->{19});
440 } else {
441 $STATWIDS->{st_str} ->set_text (sprintf "%d", $stats->{5}); 557 $STATWIDS->{st_str} ->set_text (sprintf "%d", $stats->{5});
442 $STATWIDS->{st_dex} ->set_text (sprintf "%d", $stats->{8}); 558 $STATWIDS->{st_dex} ->set_text (sprintf "%d", $stats->{8});
443 $STATWIDS->{st_con} ->set_text (sprintf "%d", $stats->{9}); 559 $STATWIDS->{st_con} ->set_text (sprintf "%d", $stats->{9});
444 $STATWIDS->{st_int} ->set_text (sprintf "%d", $stats->{6}); 560 $STATWIDS->{st_int} ->set_text (sprintf "%d", $stats->{6});
445 $STATWIDS->{st_wis} ->set_text (sprintf "%d", $stats->{7}); 561 $STATWIDS->{st_wis} ->set_text (sprintf "%d", $stats->{7});
446 $STATWIDS->{st_pow} ->set_text (sprintf "%d", $stats->{22}); 562 $STATWIDS->{st_pow} ->set_text (sprintf "%d", $stats->{22});
447 $STATWIDS->{st_cha} ->set_text (sprintf "%d", $stats->{10}); 563 $STATWIDS->{st_cha} ->set_text (sprintf "%d", $stats->{10});
448 $STATWIDS->{st_wc} ->set_text (sprintf "%d", $stats->{13}); 564 $STATWIDS->{st_wc} ->set_text (sprintf "%d", $stats->{13});
449 $STATWIDS->{st_ac} ->set_text (sprintf "%d", $stats->{14}); 565 $STATWIDS->{st_ac} ->set_text (sprintf "%d", $stats->{14});
450 $STATWIDS->{st_dam} ->set_text (sprintf "%d", $stats->{15}); 566 $STATWIDS->{st_dam} ->set_text (sprintf "%d", $stats->{15});
451 $STATWIDS->{st_arm} ->set_text (sprintf "%d", $stats->{16}); 567 $STATWIDS->{st_arm} ->set_text (sprintf "%d", $stats->{16});
452 $STATWIDS->{st_spd} ->set_text (sprintf "%.1f", $stats->{17}); 568 $STATWIDS->{st_spd} ->set_text (sprintf "%.1f", $stats->{Crossfire::Protocol::CS_STAT_SPEED});
453 $STATWIDS->{st_wspd}->set_text (sprintf "%.1f", $stats->{19}); 569 $STATWIDS->{st_wspd}->set_text (sprintf "%.1f", $stats->{Crossfire::Protocol::CS_STAT_WEAP_SP});
454 }
455 570
456 my %tbl = ( 571 my %tbl = (
457 phys => 100, 572 phys => 100,
458 magic => 101, 573 magic => 101,
459 fire => 102, 574 fire => 102,
466 pois => 109, 581 pois => 109,
467 slow => 110, 582 slow => 110,
468 para => 111, 583 para => 111,
469 tund => 112, 584 tund => 112,
470 fear => 113, 585 fear => 113,
586 depl => 113,
471 deat => 115, 587 deat => 115,
472 holyw => 116, 588 holyw => 116,
473 blind => 117 589 blind => 117
474 ); 590 );
475 591
479 595
480} 596}
481 597
482sub metaserver_dialog { 598sub metaserver_dialog {
483 my $dialog = new CFClient::UI::FancyFrame 599 my $dialog = new CFClient::UI::FancyFrame
484 title => "Metaserver", 600 title => "Server List",
485 child => (my $vbox = new CFClient::UI::VBox); 601 child => (my $vbox = new CFClient::UI::VBox);
486 602
487 $vbox->add ($dialog->{table} = new CFClient::UI::Table); 603 $vbox->add ($dialog->{table} = new CFClient::UI::Table);
488 604
489 $dialog 605 $dialog
490} 606}
607
608my $METASERVER_ATIME;
491 609
492sub update_metaserver { 610sub update_metaserver {
493 my ($HOST) = @_; 611 my ($HOST) = @_;
494 612
495 status "fetching metaserver list..."; 613 return if $METASERVER_ATIME > time;
614 $METASERVER_ATIME = time + 60;
615
616 my $table = $METASERVER->{table};
617 $table->clear;
618 $table->add (0, 0, my $label = new CFClient::UI::Label max_w => $WIDTH * 0.8, text => "fetching server list...");
496 619
497 my $buf; 620 my $buf;
498 621
499 my $fh = new IO::Socket::INET PeerHost => $META_SERVER, Blocking => 0; 622 my $fh = new IO::Socket::INET PeerHost => $META_SERVER, Blocking => 0;
623
624 unless ($fh) {
625 $label->set_text ("unable to contact metaserver: $!");
626 return;
627 }
500 628
501 Event->io (fd => $fh, poll => 'r', cb => sub { 629 Event->io (fd => $fh, poll => 'r', cb => sub {
502 my $res = sysread $fh, $buf, 8192, length $buf; 630 my $res = sysread $fh, $buf, 8192, length $buf;
503 631
504 if (!defined $res) { 632 if (!defined $res) {
505 $_[0]->w->cancel; 633 $_[0]->w->cancel;
506 status "metaserver: $!"; 634 $label->set_text ("error while retrieving server list: $!");
507 } elsif ($res == 0) { 635 } elsif ($res == 0) {
508 $_[0]->w->cancel; 636 $_[0]->w->cancel;
509 status "server list retrieved"; 637 status "server list retrieved";
510 638
511 my $table = $METASERVER->{table}; 639 utf8::decode $buf if utf8::valid $buf;
512 640
513 $table->clear; 641 $table->clear;
514 642
515 my @col = qw(Use #Users Host Uptime Version Description); 643 my @col = qw(Use #Users Host Uptime Version Description);
516 $table->add ($_, 0, new CFClient::UI::Label align => 0, fg => [1, 1, 0], text => $col[$_]) 644 $table->add ($_, 0, new CFClient::UI::Label align => 0, fg => [1, 1, 0], text => $col[$_])
540 $m = [$users, $host, $uptime, $version, $desc]; 668 $m = [$users, $host, $uptime, $version, $desc];
541 669
542 $y++; 670 $y++;
543 671
544 $table->add (0, $y, new CFClient::UI::VBox children => [ 672 $table->add (0, $y, new CFClient::UI::VBox children => [
545 (new CFClient::UI::Button text => " ", connect_activate => sub { 673 (new CFClient::UI::Button text => "Use", connect_activate => sub {
546 $HOST->set_text ($CFG->{host} = $host); 674 $HOST->set_text ($CFG->{host} = $host);
547 }), 675 }),
548 (new CFClient::UI::Empty expand => 1), 676 (new CFClient::UI::Empty expand => 1),
549 ]); 677 ]);
550 678
551 $table->add ($_ + 1, $y, new CFClient::UI::Label align => $align[$_], text => $m->[$_], fontsize => 0.8) 679 $table->add ($_ + 1, $y, new CFClient::UI::Label
680 ellipsise => 0, align => $align[$_], text => $m->[$_], fontsize => 0.8)
552 for 0 .. $#$m; 681 for 0 .. $#$m;
553 } 682 }
554 } 683 }
555 }); 684 });
556} 685}
564 $table->add (0, 2, new CFClient::UI::Label valign => 0, align => 1, text => "Host:Port"); 693 $table->add (0, 2, new CFClient::UI::Label valign => 0, align => 1, text => "Host:Port");
565 694
566 { 695 {
567 $table->add (1, 2, my $vbox = new CFClient::UI::VBox); 696 $table->add (1, 2, my $vbox = new CFClient::UI::VBox);
568 697
569 $vbox->add (my $HOST = new CFClient::UI::Entry expand => 1, text => $CFG->{host}, connect_changed => sub { 698 $vbox->add (
699 my $HOST = new CFClient::UI::Entry
700 expand => 1,
701 text => $CFG->{host},
702 tooltip => "The hostname or ip address of the Crossfire(+) server to connect to",
703 connect_changed => sub {
704 my ($self, $value) = @_;
705 $CFG->{host} = $value;
706 }
707 );
708
709 $METASERVER = metaserver_dialog;
710
711 $vbox->add (new CFClient::UI::Flopper
712 expand => 1,
713 text => "Server List",
714 other => $METASERVER,
715 tooltip => "Show a list of available crossfire servers",
716 connect_open => sub {
717 update_metaserver $HOST;
718 }
719 );
720 }
721
722 $table->add (0, 4, new CFClient::UI::Label valign => 0, align => 1, text => "Username");
723 $table->add (1, 4, new CFClient::UI::Entry
724 text => $CFG->{user},
725 tooltip => "The name of your character on the server",
726 connect_changed => sub {
570 my ($self, $value) = @_; 727 my ($self, $value) = @_;
571 $CFG->{host} = $value;
572 });
573
574 $METASERVER = metaserver_dialog;
575
576 $vbox->add (new CFClient::UI::Flopper expand => 1, text => "Metaserver", other => $METASERVER, connect_open => sub {
577 update_metaserver $HOST;
578 });
579 }
580
581 $table->add (0, 4, new CFClient::UI::Label valign => 0, align => 1, text => "Username");
582 $table->add (1, 4, new CFClient::UI::Entry text => $CFG->{user}, connect_changed => sub {
583 my ($self, $value) = @_;
584 $CFG->{user} = $value; 728 $CFG->{user} = $value;
729 }
585 }); 730 );
586 731
587 $table->add (0, 5, new CFClient::UI::Label valign => 0, align => 1, text => "Password"); 732 $table->add (0, 5, new CFClient::UI::Label valign => 0, align => 1, text => "Password");
588 $table->add (1, 5, new CFClient::UI::Entry text => $CFG->{password}, hidden => 1, connect_changed => sub { 733 $table->add (1, 5, new CFClient::UI::Entry
734 text => $CFG->{password},
735 hidden => 1,
736 tooltip => "The password for your character",
737 connect_changed => sub {
589 my ($self, $value) = @_; 738 my ($self, $value) = @_;
590 $CFG->{password} = $value; 739 $CFG->{password} = $value;
740 }
591 }); 741 );
592
593 $table->add (0, 6, new CFClient::UI::Label valign => 0, align => 1, text => "Def. say cmd");
594 $table->add (1, 6, my $saycmd = new CFClient::UI::Entry text => $CFG->{say_command}, connect_changed => sub {
595 my ($self, $value) = @_;
596 $CFG->{say_command} = $value;
597 });
598 742
599 $table->add (0, 7, new CFClient::UI::Label valign => 0, align => 1, text => "Map Size"); 743 $table->add (0, 7, new CFClient::UI::Label valign => 0, align => 1, text => "Map Size");
600 $table->add (1, 7, new CFClient::UI::Slider 744 $table->add (1, 7, new CFClient::UI::Slider
601 req_w => 100, 745 req_w => 100,
602 range => [$CFG->{mapsize}, 10, 100 + 1, 1], 746 range => [$CFG->{mapsize}, 10, 100, 0, 1],
747 tooltip => "This is the size of the portion of the map update the server sends you. "
748 . "If you set this to a high value you will be able to see further for example.",
603 connect_changed => sub { 749 connect_changed => sub {
604 my ($self, $value) = @_; 750 my ($self, $value) = @_;
605 751
606 $CFG->{mapsize} = $self->{range}[0] = $value = int $value; 752 $CFG->{mapsize} = $self->{range}[0] = $value = int $value;
607 }, 753 },
608 ); 754 );
609 755
610 $table->add (1, 8, new CFClient::UI::Button expand => 1, align => 0, text => "Login", connect_activate => sub { 756 $table->add (0, 8, new CFClient::UI::Label valign => 0, align => 1, text => "Output-Count");
611 start_game; 757 $table->add (1, 8, new CFClient::UI::Entry
758 text => $CFG->{output_count},
759 tooltip => "Should be set to 1 unless you know what you are doing",
760 connect_changed => sub { $CFG->{output_count} = $_[1] },
612 }); 761 );
762
763 $table->add (0, 9, new CFClient::UI::Label valign => 0, align => 1, text => "Output-Sync");
764 $table->add (1, 9, new CFClient::UI::Entry
765 text => $CFG->{output_sync},
766 tooltip => "Should be set to 1 unless you know what you are doing",
767 connect_changed => sub { $CFG->{output_sync} = $_[1] },
768 );
769
770 $table->add (1, 10, $LOGIN_BUTTON = new CFClient::UI::Button
771 expand => 1,
772 align => 0,
773 text => "Login",
774 connect_activate => sub {
775 $CONN ? stop_game
776 : start_game;
777 },
778 );
613 779
614 $dialog 780 $dialog
615} 781}
616 782
617sub message_window { 783sub message_window {
618 my $window = new CFClient::UI::FancyFrame 784 my $window = new CFClient::UI::FancyFrame
619 title => "Messages", 785 title => "Messages",
620 border_bg => [1, 1, 1, 0.5], 786 border_bg => [1, 1, 1, 1],
621 bg => [0.3, 0.3, 0.3, 0.8], 787 bg => [0, 0, 0, 0.75],
622 user_w => int $::WIDTH / 3, 788 user_w => int $::WIDTH / 3,
623 user_h => int $::HEIGHT / 5, 789 user_h => int $::HEIGHT / 5,
624 child => (my $vbox = new CFClient::UI::VBox); 790 child => (my $vbox = new CFClient::UI::VBox);
625 791
626 $vbox->add ($LOGVIEW = new CFClient::UI::TextView 792 $vbox->add ($LOGVIEW);
627 expand => 1,
628 fontsize => $::CFG->{log_fontsize},
629 );
630 793
631 $vbox->add (my $input = new CFClient::UI::Entry 794 $vbox->add (my $input = new CFClient::UI::Entry
795 tooltip => "<b>Chat Box</b>. If you enter a text and press return/enter here, the current <i>communication command</i> "
796 . "from the client setup will be prepended (e.g. <b>shout</b>, <b>chat</b>...). "
797 . "If you prepend a slash (/), you will submit a command instead (similar to IRC). "
798 . "A better way to submit commands (and the occasional chat command) is often the map command completer.",
632 connect_focus_in => sub { 799 connect_focus_in => sub {
633 my ($input, $prev_focus) = @_; 800 my ($input, $prev_focus) = @_;
634 801
635 delete $input->{refocus_map}; 802 delete $input->{refocus_map};
636 803
665 }; 832 };
666 833
667 $window 834 $window
668} 835}
669 836
837sub open_quit_dialog {
838 unless ($QUIT_DIALOG) {
839
840 $QUIT_DIALOG = new CFClient::UI::FancyFrame title => "Really Quit?";
841
842 $QUIT_DIALOG->add (my $vb = new CFClient::UI::VBox expand => 1);
843
844 $vb->add (new CFClient::UI::Label
845 text => "You should find a savebed and apply it first!",
846 max_w => $WIDTH * 0.25,
847 ellipsize => 0,
848 );
849 $vb->add (my $hb = new CFClient::UI::HBox expand => 1);
850 $hb->add (new CFClient::UI::Button
851 text => "Ok",
852 expand => 1,
853 connect_activate => sub { $QUIT_DIALOG->hide },
854 );
855 $hb->add (new CFClient::UI::Button
856 text => "Quit anyway",
857 expand => 1,
858 connect_activate => sub { exit },
859 );
860
861 $QUIT_DIALOG->show_centered;
862 } else {
863 $QUIT_DIALOG->show_centered;
864 }
865}
866
867sub make_inventory_window {
868 my $invwin = new CFClient::UI::FancyFrame
869 user_w => $WIDTH * (7/8), user_h => $HEIGHT * (7/8), title => "Inventory";
870
871 $invwin->add (my $hb = new CFClient::UI::HBox expand => 1);
872
873 $hb->add (my $vb1 = new CFClient::UI::VBox expand => 1);
874 $vb1->add (my $lbl = new CFClient::UI::Label);
875 $lbl->set_text ("Player");
876 $vb1->add ($INV = new CFClient::UI::Inventory expand => 1);
877
878 $hb->add (my $vb2 = new CFClient::UI::VBox expand => 1);
879 $vb2->add ($INVR_LBL = new CFClient::UI::Label);
880 $INVR_LBL->set_text ("Floor");
881 $vb2->add ($INVR = new CFClient::UI::Inventory expand => 1);
882
883 $invwin
884}
885
886sub make_help_window {
887 my $win = new CFClient::UI::FancyFrame
888 user_w => $WIDTH * (7/8), user_h => $HEIGHT * (7/8), title => "Documentation";
889
890 $win->add (my $vbox = new CFClient::UI::VBox);
891
892 $vbox->add (my $buttons = new CFClient::UI::HBox);
893 $vbox->add (my $viewer = new CFClient::UI::TextView expand => 1, fontsize => 0.8);
894
895 for (
896 [intro => "Introduction"],
897 [manual => "Manual"],
898 [command_help => "Commands"],
899 [skill_help => "Skills"],
900 ) {
901 my ($pod, $label) = @$_;
902
903 $buttons->add (new CFClient::UI::Button
904 text => $label,
905 connect_activate => sub {
906 my $parser = new Pod::POM;
907 my $pom = $parser->parse_file (CFClient::find_rcfile "pod/$pod.pod");
908
909 $viewer->clear;
910
911 $viewer->add_paragraph ([1, 1, 1, 1], $_->[1], $_->[0])
912 for @{ CFClient::pod_to_pango_list $pom };
913
914 $viewer->set_offset (0);
915 },
916 );
917 }
918
919 $viewer->add_paragraph ([1, 1, 0, 1], "<big>Use one of the buttons above to display a document.</big>");
920
921 $win
922}
923
670sub sdl_init { 924sub sdl_init {
671 CFClient::SDL_Init 925 CFClient::SDL_Init
672 and die "SDL::Init failed!\n"; 926 and die "SDL::Init failed!\n";
673} 927}
674 928
675sub video_init { 929sub video_init {
676 sdl_init; 930 sdl_init;
677 931
932 $CFG->{sdl_mode} = 0 if $CFG->{sdl_mode} >= @SDL_MODES;
933
934 my ($old_w, $old_h) = ($WIDTH, $HEIGHT);
935
678 ($WIDTH, $HEIGHT) = @{ $SDL_MODES[$CFG->{sdl_mode}] }; 936 ($WIDTH, $HEIGHT) = @{ $SDL_MODES[$CFG->{sdl_mode}] };
679 $FULLSCREEN = $CFG->{fullscreen}; 937 $FULLSCREEN = $CFG->{fullscreen};
680 $FAST = $CFG->{fast}; 938 $FAST = $CFG->{fast};
681 939
682 CFClient::SDL_SetVideoMode $WIDTH, $HEIGHT, $FULLSCREEN 940 CFClient::SDL_SetVideoMode $WIDTH, $HEIGHT, $FULLSCREEN
683 or die "SDL_SetVideoMode failed!\n"; 941 or die "SDL_SetVideoMode failed: " . (CFClient::SDL_GetError) . "\n";
684 942
685 $SDL_ACTIVE = 1; 943 $SDL_ACTIVE = 1;
686
687 $LAST_REFRESH = time - 0.01; 944 $LAST_REFRESH = time - 0.01;
688 945
689 CFClient::gl_init; 946 CFClient::gl_init;
690 947
691 $FONTSIZE = int $HEIGHT / 40 * $CFG->{gui_fontsize}; 948 $FONTSIZE = int $HEIGHT / 40 * $CFG->{gui_fontsize};
692 949
950 $CFClient::UI::ROOT->configure (0, 0, $WIDTH, $HEIGHT);#d#
951
693 ############################################################################# 952 #############################################################################
694 953
954 if ($DEBUG_STATUS) {
955 CFClient::UI::rescale_widgets $WIDTH / $old_w, $HEIGHT / $old_h;
956 } else {
957 # create the widgets
958
695 $DEBUG_STATUS = new CFClient::UI::Label padding => 0, z => 100; 959 $DEBUG_STATUS = new CFClient::UI::Label padding => 0, z => 100, req_x => -1;
696 $DEBUG_STATUS->show; 960 $DEBUG_STATUS->show;
697 961
698 $STATUS_LINE = new CFClient::UI::Label 962 $STATUSBOX = new CFClient::UI::Statusbox;
699 padding => 0, 963 $STATUSBOX->add ("Use <b>Alt-Enter</b> to toggle fullscreen mode", pri => -100, color => [1, 1, 1, 0.8]);
700 y => $HEIGHT - $FONTSIZE * 1.8;
701 $STATUS_LINE->show;
702 964
703 $ALT_ENTER_MESSAGE = new CFClient::UI::Label 965 (new CFClient::UI::Frame
704 padding => 0, 966 bg => [0, 0, 0, 0.4],
705 fontsize => 0.8, 967 req_y => -1,
706 markup => "Use <b>Alt-Enter</b> to toggle fullscreen mode"; 968 child => $STATUSBOX,
707 $ALT_ENTER_MESSAGE->show; 969 )->show;
708 $ALT_ENTER_MESSAGE->move (0, $HEIGHT - $ALT_ENTER_MESSAGE->{h});
709 970
710 $CFClient::UI::ROOT->add ($MAPWIDGET = new CFClient::MapWidget); 971 CFClient::UI::FancyFrame->new (
711 $MAPWIDGET->focus_in; 972 border_bg => [1, 1, 1, 192/255],
973 bg => [1, 1, 1, 0],
974 child => ($MAPMAP = new CFClient::MapWidget::MapMap
975 tooltip => "<b>Map</b>. On servers that support this feature, this will display an overview of the surrounding areas.",
976 ),
977 )->show;
978
979 $MAPWIDGET = new CFClient::MapWidget;
712 $MAPWIDGET->connect (activate_console => sub { 980 $MAPWIDGET->connect (activate_console => sub {
713 my ($mapwidget, $preset) = @_; 981 my ($mapwidget, $preset) = @_;
714 982
715 if ($CONSOLE) { 983 if ($CONSOLE) {
716 $CONSOLE->{input}->{auto_activated} = 1; 984 $CONSOLE->{input}->{auto_activated} = 1;
717 $CONSOLE->{input}->focus_in; 985 $CONSOLE->{input}->focus_in;
718 986
719 if ($preset && $CONSOLE->{input}->get_text eq '') { 987 if ($preset && $CONSOLE->{input}->get_text eq '') {
720 $CONSOLE->{input}->set_text ($preset); 988 $CONSOLE->{input}->set_text ($preset);
989 }
721 } 990 }
722 } 991 });
723 }); 992 $MAPWIDGET->show;
993 $MAPWIDGET->focus_in;
724 994
725 $CFClient::UI::ROOT->add ($BUTTONBAR = new CFClient::UI::HBox); 995 $LOGVIEW = new CFClient::UI::TextView
996 expand => 1,
997 font => $FONT_FIXED,
998 fontsize => $::CFG->{log_fontsize},
999 can_hover => 1,
1000 can_events => 1,
1001 tooltip => "<b>Server Log</b>. This text viewer contains all the messages sent by the server.",
1002 ;
726 1003
1004 $BUTTONBAR = new CFClient::UI::HBox;
1005
727 $BUTTONBAR->add (new CFClient::UI::Flopper text => "Client Setup", other => client_setup); 1006 $BUTTONBAR->add (new CFClient::UI::Flopper text => "Client Setup", other => client_setup,
1007 tooltip => "Toggles a dialog where you can configure various aspects of the client, such as graphics mode, performance, and audio options.");
728 $BUTTONBAR->add (new CFClient::UI::Flopper text => "Server Setup", other => server_setup); 1008 $BUTTONBAR->add (new CFClient::UI::Flopper text => "Server Setup", other => server_setup,
1009 tooltip => "Toggles a dialog where you can configure the server to play on, your username, password and other server-related options.");
729 $BUTTONBAR->add (new CFClient::UI::Flopper text => "Message Window", other => message_window); 1010 $BUTTONBAR->add (new CFClient::UI::Flopper text => "Message Window", other => message_window,
1011 tooltip => "Toggles the server message log, where the client collects <i>all</i> messages from the server.");
730 1012
731 $BUTTONBAR->add (new CFClient::UI::Button text => "Save Config", connect_activate => sub {
732 CFClient::write_cfg "$Crossfire::VARDIR/pclientrc";
733 status "Configuration Saved";
734 });
735
736 $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 1013 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
1014
737 $BUTTONBAR->add (new CFClient::UI::Flopper text => "Stats Window", other => make_stats_window); 1015 $BUTTONBAR->add (new CFClient::UI::Flopper text => "Stats Window", other => make_stats_window,
1016 tooltip => "Toggles the statistics window, where all your Stats and Resistances are beign displaye at all times.");
1017 $BUTTONBAR->add (new CFClient::UI::Flopper text => "Inventory", other => make_inventory_window,
1018 tooltip => "Toggles the inventory window, where you can manage your loot (or treaures :).");
738 1019
1020 $BUTTONBAR->add (new CFClient::UI::Button
1021 text => "Save Config",
1022 tooltip => "Saves the options chosen in the client setting, server settings and the window layout to be restored on later runs.",
1023 connect_activate => sub {
1024 CFClient::write_cfg "$Crossfire::VARDIR/pclientrc";
1025 status "Configuration Saved";
1026 },
1027 );
1028
1029 $BUTTONBAR->add (new CFClient::UI::Flopper text => "Help!", other => make_help_window,
1030 tooltip => "View Documentation");
1031
1032 $BUTTONBAR->add (new CFClient::UI::Button
1033 text => "Quit",
1034 tooltip => "Terminates the program",
1035 connect_activate => sub {
1036 if ($CONN) {
1037 open_quit_dialog;
1038 } else {
1039 exit;
1040 }
1041 },
1042 );
1043
1044 $BUTTONBAR->show;
1045
1046 $STATUSBOX->add ("Set video mode $WIDTH×$HEIGHT", timeout => 10, fg => [1, 1, 1, 0.5]);
1047
1048 # delay till geometry is constant
1049 $CFClient::UI::ROOT->on_post_alloc (startup => sub {
739 $BUTTONBAR->{children}[1]->emit ("activate"); # pop up server setup 1050 $BUTTONBAR->{children}[1]->emit ("activate"); # pop up server setup
740 1051 my $widget = $GAUGES->{win};
741 1052 $widget->move (0, $HEIGHT - $widget->{h});#d# to in toplevel
1053 });
1054 force_refresh ();
1055 }
742} 1056}
743 1057
744sub video_shutdown { 1058sub video_shutdown {
745 $CFClient::UI::ROOT->{children} = [];
746 undef $SDL_ACTIVE; 1059 undef $SDL_ACTIVE;
747} 1060}
748 1061
749my @bgmusic = qw(game1.ogg game2.ogg game3.ogg game5.ogg game6.ogg ross1.ogg ross2.ogg ross3.ogg ross4.ogg ross5.ogg); #d# 1062my @bgmusic = qw(game1.ogg game2.ogg game3.ogg game5.ogg game6.ogg ross1.ogg ross2.ogg ross3.ogg ross4.ogg ross5.ogg); #d#
750my $bgmusic;#TODO#hack#d# 1063my $bgmusic;#TODO#hack#d#
1064
1065sub audio_channel_finished {
1066 my ($channel) = @_;
1067
1068 #warn "channel $channel finished\n";#d#
1069}
751 1070
752sub audio_music_finished { 1071sub audio_music_finished {
753 return unless $CFG->{bgm_enable}; 1072 return unless $CFG->{bgm_enable};
754 1073
755 # TODO: hack, do play loop and mood music 1074 # TODO: hack, do play loop and mood music
759 push @bgmusic, shift @bgmusic; 1078 push @bgmusic, shift @bgmusic;
760} 1079}
761 1080
762sub audio_init { 1081sub audio_init {
763 if ($CFG->{audio_enable}) { 1082 if ($CFG->{audio_enable}) {
764 if (open my $fh, "<:utf8", CFClient::find_rcfile "sounds/config") { 1083 if (open my $fh, "<", CFClient::find_rcfile "sounds/config") {
765 $SDL_MIXER = !CFClient::Mix_OpenAudio; 1084 $SDL_MIXER = !CFClient::Mix_OpenAudio;
1085
1086 unless ($SDL_MIXER) {
1087 status "Unable to open sound device: there will be no sound";
1088 return;
1089 }
1090
766 CFClient::Mix_AllocateChannels 8; 1091 CFClient::Mix_AllocateChannels 8;
767 CFClient::MixMusic::volume $CFG->{bgm_volume} * 128; 1092 CFClient::MixMusic::volume $CFG->{bgm_volume} * 128;
768 1093
769 audio_music_finished; 1094 audio_music_finished;
770 1095
796} 1121}
797 1122
798my %animate_object; 1123my %animate_object;
799my $animate_timer; 1124my $animate_timer;
800 1125
801my $want_refresh;
802my $can_refresh;
803
804my $fps = 9; 1126my $fps = 9;
805 1127
1128my %demo;#d#
1129
806sub force_refresh { 1130sub force_refresh {
807 $fps = $fps * 0.95 + 1 / ($NOW - $LAST_REFRESH) * 0.05; 1131 $fps = $fps * 0.95 + 1 / (($NOW - $LAST_REFRESH) || 0.1) * 0.05;
808 debug sprintf "%3.2f", $fps; 1132 debug sprintf "%3.2f", $fps;
809 1133
810 $want_refresh = 0;
811 $can_refresh = 0;
812
813 $CFClient::UI::ROOT->draw; 1134 $CFClient::UI::ROOT->draw;
814 1135
1136 $WANT_REFRESH = 0;
1137 $CAN_REFRESH = 0;
1138 $LAST_REFRESH = $NOW;
1139
11400 && do {
1141 # some weird model-drawing code, just a joke right now
1142 use CFClient::OpenGL;
1143
1144 $demo{t}{eye_auv} ||= new_from_file CFClient::Texture "eye2.png" or die;
1145 $demo{t}{body_auv} ||= new_from_file CFClient::Texture "body_auv3.png" or die;
1146 $demo{r} ||= do {
1147 my $mod = Compress::LZF::sthaw do { local $/; open my $fh, "<:raw:perlio", "dread.lz3"; <$fh> };
1148 $mod->{v} = pack "f*", @{$mod->{v}};
1149 $_ = [scalar @$_, pack "S!*", @$_]
1150 for values %{$mod->{g}};
1151 $mod
1152 };
1153
1154 my $r = $demo{r} or die;
1155
1156 glDepthMask 1;
1157 glClear GL_DEPTH_BUFFER_BIT;
1158 glEnable GL_TEXTURE_2D;
1159 glEnable GL_DEPTH_TEST;
1160 glEnable GL_CULL_FACE;
1161 glShadeModel $::FAST ? GL_FLAT : GL_SMOOTH;
1162
1163 glMatrixMode GL_PROJECTION;
1164 glLoadIdentity;
1165 glFrustum -1 * ($::WIDTH / $::HEIGHT), 1 * ($::WIDTH / $::HEIGHT), 1, -1, 1, 10000;
1166 #glOrtho 0, $::WIDTH, 0, $::HEIGHT, -10000, 10000;
1167 glMatrixMode GL_MODELVIEW;
1168 glLoadIdentity;
1169
1170 glPushMatrix;
1171 glTranslate 0, 0, -800;
1172 glScale 1, -1, 1;
1173 glRotate $NOW * 1000 % 36000 / 5, 0, 1, 0;
1174 glRotate $NOW * 1000 % 36000 / 6, 1, 0, 0;
1175 glRotate $NOW * 1000 % 36000 / 7, 0, 0, 1;
1176 glScale 50, 50, 50;
1177
1178 glInterleavedArrays GL_T2F_N3F_V3F, 0, $r->{v};
1179 while (my ($k, $v) = each %{$r->{g}}) {
1180 glBindTexture GL_TEXTURE_2D, ($demo{t}{$k}{name} or die);
1181 glDrawElements GL_TRIANGLES, $v->[0], GL_UNSIGNED_SHORT, $v->[1];
1182 }
1183
1184 glPopMatrix;
1185
1186 glShadeModel GL_FLAT;
1187 glDisable GL_DEPTH_TEST;
1188 glDisable GL_TEXTURE_2D;
1189 glDepthMask 0;
1190
1191 $WANT_REFRESH++;
1192};
1193
815 CFClient::SDL_GL_SwapBuffers; 1194 CFClient::SDL_GL_SwapBuffers;
816
817 $LAST_REFRESH = $NOW;
818} 1195}
819 1196
820my $refresh_watcher = Event->timer (after => 0, hard => 1, interval => 1 / $MAX_FPS, cb => sub { 1197my $refresh_watcher = Event->timer (after => 0, hard => 1, interval => 1 / $MAX_FPS, cb => sub {
821 $NOW = time; 1198 $NOW = time;
822 1199
823 ($SDL_CB{$_->{type}} || sub { warn "unhandled event $_->{type}" })->($_) 1200 ($SDL_CB{$_->{type}} || sub { warn "unhandled event $_->{type}" })->($_)
824 for CFClient::SDL_PollEvent; 1201 for CFClient::SDL_PollEvent;
825 1202
826 if (%animate_object) { 1203 if (%animate_object) {
827 $_->animate ($LAST_REFRESH - $NOW) for values %animate_object; 1204 $_->animate ($LAST_REFRESH - $NOW) for values %animate_object;
828 $want_refresh++; 1205 $WANT_REFRESH++;
829 } 1206 }
830 1207
831 if ($want_refresh) { 1208 if ($WANT_REFRESH) {
832 force_refresh; 1209 force_refresh;
833 } else { 1210 } else {
834 $can_refresh = 1; 1211 $CAN_REFRESH = 1;
835 } 1212 }
836}); 1213});
837
838sub refresh {
839 $want_refresh++;
840}
841 1214
842sub animation_start { 1215sub animation_start {
843 my ($widget) = @_; 1216 my ($widget) = @_;
844 $animate_object{$widget} = $widget; 1217 $animate_object{$widget} = $widget;
845} 1218}
849 delete $animate_object{$widget}; 1222 delete $animate_object{$widget};
850} 1223}
851 1224
852@conn::ISA = Crossfire::Protocol::; 1225@conn::ISA = Crossfire::Protocol::;
853 1226
1227sub conn::new {
1228 my $class = shift;
1229
1230 my $self = $class->Crossfire::Protocol::new (@_);
1231
1232 $MAPWIDGET->clr_commands;
1233
1234 my $parser = new Pod::POM;
1235 my $pod = $parser->parse_file (CFClient::find_rcfile "pod/command_help.pod");
1236
1237 for my $head2 ($pod->head2) {
1238 $head2->title =~ /^(\S+) (?:\s+ \( ([^\)]*) \) )?/x
1239 or next;
1240
1241 my $cmd = $1;
1242 my @args = split /\|/, $2;
1243 @args = (".*") unless @args;
1244
1245 my $text = CFClient::pod_to_pango $head2->content;
1246
1247 for my $arg (@args) {
1248 $arg = $arg eq ".*" ? "" : " $arg";
1249
1250 $MAPWIDGET->add_command ("$cmd$arg", $text);
1251 }
1252 }
1253
1254 $self->{noface} = new_from_file CFClient::Texture
1255 CFClient::find_rcfile "noface.png", minify => 1, mipmap => 1;
1256
1257 $self
1258}
1259
854sub conn::stats_update { 1260sub conn::stats_update {
855 my ($self, $stats) = @_; 1261 my ($self, $stats) = @_;
1262
1263 if (my $exp = $stats->{Crossfire::Protocol::CS_STAT_EXP64}) {
1264 my $diff = $exp - $self->{prev_exp};
1265 $STATUSBOX->add ("$diff experience gained", group => "experience $diff", fg => [0.5, 1, 0.5, 0.8], timeout => 5)
1266 if exists $self->{prev_exp} && $diff;
1267 $self->{prev_exp} = $exp;
1268 }
856 1269
857 update_stats_window ($stats); 1270 update_stats_window ($stats);
858} 1271}
859 1272
860sub conn::user_send { 1273sub conn::user_send {
915 $self->set_texture ($id => $data); 1328 $self->set_texture ($id => $data);
916 } 1329 }
917 } 1330 }
918} 1331}
919 1332
1333# hardcode /world/world_xxx_xxx map names, the savings are enourmous,
1334# (server resource,s latency, bandwidth), so this hack is warranted.
1335# the right fix is to make real tiled maps with an overview file
1336sub conn::send_mapinfo {
1337 my ($self, $data, $cb) = @_;
1338
1339 if ($self->{map_info}[0] =~ m%^/world/world_(\d\d\d)_(\d\d\d)$%) {
1340 my ($wx, $wy) = ($1, $2);
1341
1342 if ($data =~ /^spatial ([1-4]+)$/) {
1343 my @dx = (0, 0, 1, 0, -1);
1344 my @dy = (0, -1, 0, 1, 0);
1345 my ($dx, $dy);
1346
1347 for (split //, $1) {
1348 $dx += $dx[$_];
1349 $dy += $dy[$_];
1350 }
1351
1352 $cb->(spatial => 15,
1353 $self->{map_info}[1] - $MAP->ox + $dx * 50,
1354 $self->{map_info}[2] - $MAP->oy + $dy * 50,
1355 50, 50,
1356 sprintf "/world/world_%03d_%03d", $wx + $dx, $wy + $dy
1357 );
1358
1359 return;
1360 }
1361 }
1362
1363 $self->Crossfire::Protocol::send_mapinfo ($data, $cb);
1364}
1365
920# this method does a "flood fill" into every tile direction 1366# this method does a "flood fill" into every tile direction
921# it assumes that tiles are arranged in a rectangular grid, 1367# it assumes that tiles are arranged in a rectangular grid,
922# i.e. a map is the same as the left of the right map etc. 1368# i.e. a map is the same as the left of the right map etc.
923# failure to comply are harmless and result in display errors 1369# failure to comply are harmless and result in display errors
924# at worst. 1370# at worst.
925sub conn::flood_fill { 1371sub conn::flood_fill {
926 my ($self, $gx, $gy, $path, $hash, $flags) = @_; 1372 my ($self, $block, $gx, $gy, $path, $hash, $flags) = @_;
927 1373
928 # the server does not allow map paths > 6 1374 # the server does not allow map paths > 6
929 return if 6 <= length $path; 1375 return if 7 <= length $path;
930 1376
931 my ($x0, $y0, $x1, $y1) = @{$self->{neigh_rect}}; 1377 my ($x0, $y0, $x1, $y1) = @{$self->{neigh_rect}};
932 1378
933 for ( 1379 for (
934 [1, 0, -1], 1380 [1, 3, 0, -1],
935 [2, 1, 0], 1381 [2, 4, 1, 0],
936 [3, 0, 1], 1382 [3, 1, 0, 1],
937 [4, -1, 0], 1383 [4, 2, -1, 0],
938 ) { 1384 ) {
939 my ($tile, $dx, $dy) = @$_; 1385 my ($tile, $tile2, $dx, $dy) = @$_;
1386
1387 next if $block & (1 << $tile);
1388 my $block = $block | (1 << $tile2);
940 1389
941 my $gx = $gx + $dx; 1390 my $gx = $gx + $dx;
942 my $gy = $gy + $dy; 1391 my $gy = $gy + $dy;
943 1392
944 next unless $flags & (1 << ($tile - 1)); 1393 next unless $flags & (1 << ($tile - 1));
946 1395
947 my $neigh = $self->{neigh_map}{$hash} ||= []; 1396 my $neigh = $self->{neigh_map}{$hash} ||= [];
948 if (my $info = $neigh->[$tile]) { 1397 if (my $info = $neigh->[$tile]) {
949 my ($flags, $x, $y, $w, $h, $hash) = @$info; 1398 my ($flags, $x, $y, $w, $h, $hash) = @$info;
950 1399
951 $self->flood_fill ($gx, $gy, "$path$tile", $hash, $flags) 1400 $self->flood_fill ($block, $gx, $gy, "$path$tile", $hash, $flags)
952 if $x >= $x0 && $x + $w < $x1 && $y >= $y0 && $y + $h < $y1; 1401 if $x >= $x0 && $x + $w < $x1 && $y >= $y0 && $y + $h < $y1;
953 1402
954 } else { 1403 } else {
955 $self->send_mapinfo ("spatial $path$tile", sub { 1404 $self->send_mapinfo ("spatial $path$tile", sub {
956 my ($mode, $flags, $x, $y, $w, $h, $hash) = @_; 1405 my ($mode, $flags, $x, $y, $w, $h, $hash) = @_;
957 1406
958 return if $mode ne "spatial"; 1407 return if $mode ne "spatial";
959 1408
960 $x += $MAP->ox; 1409 $x += $MAP->ox;
961 $y += $MAP->oy; 1410 $y += $MAP->oy;
962 1411
963 $self->load_map ($hash, $x, $y) 1412 $self->load_map ($hash, $x, $y)
964 unless $self->{neigh_map}{$hash}[5]++;#d# 1413 unless $self->{neigh_map}{$hash}[5]++;#d#
965 1414
966 $neigh->[$tile] = [$flags, $x, $y, $w, $h, $hash]; 1415 $neigh->[$tile] = [$flags, $x, $y, $w, $h, $hash];
967 1416
968 $self->flood_fill ($gx, $gy, "$path$tile", $hash, $flags) 1417 $self->flood_fill ($block, $gx, $gy, "$path$tile", $hash, $flags)
969 if $x >= $x0 && $x + $w < $x1 && $y >= $y0 && $y + $h < $y1; 1418 if $x >= $x0 && $x + $w < $x1 && $y >= $y0 && $y + $h < $y1;
970 }); 1419 });
971 } 1420 }
972 } 1421 }
973} 1422}
977 1426
978 $self->flush_map; 1427 $self->flush_map;
979 1428
980 my ($ox, $oy) = ($::MAP->ox, $::MAP->oy); 1429 my ($ox, $oy) = ($::MAP->ox, $::MAP->oy);
981 1430
982 my $mapmapw = 250; 1431 my $mapmapw = $MAPMAP->{w};
983 my $mapmaph = 250; 1432 my $mapmaph = $MAPMAP->{h};
984 1433
985 $self->{neigh_rect} = [ 1434 $self->{neigh_rect} = [
986 $ox - $mapmapw * 0.5, $oy - $mapmapw * 0.5, 1435 $ox - $mapmapw * 0.5, $oy - $mapmapw * 0.5,
987 $ox + $mapmapw * 0.5 + $w, $oy + $mapmapw * 0.5 + $h, 1436 $ox + $mapmapw * 0.5 + $w, $oy + $mapmapw * 0.5 + $h,
988 ]; 1437 ];
989 1438
990 delete $self->{neigh_grid}; 1439 delete $self->{neigh_grid};
991 $self->flood_fill (0, 0, "", $hash, $flags);
992 1440
993 $x += $ox; 1441 $x += $ox;
994 $y += $oy; 1442 $y += $oy;
995 1443
996 $self->{map_info} = [$hash, $x, $y, $w, $h]; 1444 $self->{map_info} = [$hash, $x, $y, $w, $h];
997 1445
998 my $map = $self->{map_info}[0];
999 $map =~ s/^.*?\/([^\/]+)$/\1/; 1446 (my $map = $hash) =~ s/^.*?\/([^\/]+)$/\1/;
1000 $STATWIDS->{map}->set_text ("Map: " . $map); 1447 $STATWIDS->{map}->set_text ("Map: " . $map);
1001 1448
1002 $self->load_map ($hash, $x, $y); 1449 $self->load_map ($hash, $x, $y);
1450 $self->flood_fill (0, 0, 0, "", $hash, $flags);
1003} 1451}
1004 1452
1005sub conn::face_find { 1453sub conn::face_find {
1006 my ($self, $facenum, $face) = @_; 1454 my ($self, $facenum, $face) = @_;
1007 1455
1009 1457
1010 my $id = $FACEMAP->get ($hash); 1458 my $id = $FACEMAP->get ($hash);
1011 1459
1012 unless ($id) { 1460 unless ($id) {
1013 # create new id for face 1461 # create new id for face
1014 # i love transactions 1462 # I love transactions
1015 for (1..100) { 1463 for (1..100) {
1016 my $txn = $CFClient::DB_ENV->txn_begin; 1464 my $txn = $CFClient::DB_ENV->txn_begin;
1017 my $status = $FACEMAP->db_get (id => $id, BerkeleyDB::DB_RMW); 1465 my $status = $FACEMAP->db_get (id => $id, BerkeleyDB::DB_RMW);
1018 if ($status == 0 || $status == BerkeleyDB::DB_NOTFOUND) { 1466 if ($status == 0 || $status == BerkeleyDB::DB_NOTFOUND) {
1019 $id++; 1467 $id = ($id || 16) + 1;
1020 if ($FACEMAP->put (id => $id) == 0 1468 if ($FACEMAP->put (id => $id) == 0
1021 && $FACEMAP->put ($hash => $id) == 0) { 1469 && $FACEMAP->put ($hash => $id) == 0) {
1022 $txn->txn_commit; 1470 $txn->txn_commit;
1023 1471
1024 goto gotid; 1472 goto gotid;
1031 } 1479 }
1032 1480
1033gotid: 1481gotid:
1034 $face->{id} = $id; 1482 $face->{id} = $id;
1035 $MAP->set_face ($facenum => $id); 1483 $MAP->set_face ($facenum => $id);
1484 $self->{faceid}[$facenum] = $id;#d#
1485
1036 $TILECACHE->get ($id) 1486 $TILECACHE->get ($id) || do {
1487 my $tex = $self->{noface};
1488 $MAP->set_texture ($id, @$tex{qw(name w h s t)}, @{$tex->{minified}});
1489 undef
1490 };
1037} 1491}
1038 1492
1039sub conn::face_update { 1493sub conn::face_update {
1040 my ($self, $facenum, $face) = @_; 1494 my ($self, $facenum, $face) = @_;
1041 1495
1048 my ($self, $id, $data) = @_; 1502 my ($self, $id, $data) = @_;
1049 1503
1050 $self->{texture}[$id] ||= do { 1504 $self->{texture}[$id] ||= do {
1051 my $tex = 1505 my $tex =
1052 new_from_image CFClient::Texture 1506 new_from_image CFClient::Texture
1053 $data, minify => 1; 1507 $data, minify => 1, mipmap => 1;
1054 1508
1055 $MAP->set_texture ($id, @$tex{qw(name w h s t)}, @{$tex->{minified}}); 1509 $MAP->set_texture ($id, @$tex{qw(name w h s t)}, @{$tex->{minified}});
1056 $MAPWIDGET->update; 1510 $MAPWIDGET->update;
1057 1511
1058 $tex 1512 $tex
1070 1524
1071 $chunk->play; 1525 $chunk->play;
1072# warn "sound $x,$y,$soundnum,$type\n";#d# 1526# warn "sound $x,$y,$soundnum,$type\n";#d#
1073} 1527}
1074 1528
1529my $LAST_QUERY; # server is stupid, stupid, stupid
1530
1075sub conn::query { 1531sub conn::query {
1076 my ($self, $flags, $prompt) = @_; 1532 my ($self, $flags, $prompt) = @_;
1077 1533
1078 #TODO, display dialog with relevant information 1534 $prompt = $LAST_QUERY unless length $prompt;
1079 warn "<<<<QUERY:$flags:$prompt>>>\n";#d# 1535 $LAST_QUERY = $prompt;
1536
1537 my $dialog = new CFClient::UI::FancyFrame
1538 title => "Query",
1539 child => my $vbox = new CFClient::UI::VBox;
1540
1541 $vbox->add (new CFClient::UI::Label
1542 max_w => $::WIDTH * 0.4,
1543 text => $prompt);
1544
1545 if ($flags & Crossfire::Protocol::CS_QUERY_YESNO) {
1546 $vbox->add (my $hbox = new CFClient::HBox);
1547 $hbox->add (new CFClient::Button
1548 text => "No",
1549 connect_activate => sub {
1550 $self->send ("reply n");
1551 $dialog->destroy;
1552 $MAPWIDGET->focus_in;
1553 }
1554 );
1555 $hbox->add (new CFClient::Button
1556 text => "Yes",
1557 connect_activate => sub {
1558 $self->send ("reply y");
1559 $dialog->destroy;
1560 $MAPWIDGET->focus_in;
1561 },
1562 );
1563
1564 $dialog->focus_in;
1565
1566 } elsif ($flags & Crossfire::Protocol::CS_QUERY_SINGLECHAR) {
1567 $dialog->{tooltip} = "Press a key (click on the entry to make sure it has keyboard focus)";
1568 $vbox->add (my $entry = new CFClient::UI::Entry
1569 connect_changed => sub {
1570 $self->send ("reply $_[1]");
1571 $dialog->destroy;
1572 $MAPWIDGET->focus_in;
1573 },
1574 );
1575
1576 $entry->focus_in;
1577
1578 } else {
1579 $dialog->{tooltip} = "Enter the reply and press return (click on the entry to make sure it has keyboard focus)";
1580
1581 $vbox->add (my $entry = new CFClient::UI::Entry
1582 $flags & Crossfire::Protocol::CS_QUERY_HIDEINPUT ? (hiddenchar => "*") : (),
1583 connect_activate => sub {
1584 $self->send ("reply $_[1]");
1585 $dialog->destroy;
1586 $MAPWIDGET->focus_in;
1587 },
1588 );
1589
1590 $entry->focus_in;
1591 }
1592
1593 $dialog->show_centered;
1080} 1594}
1081 1595
1082sub conn::drawinfo { 1596sub conn::drawinfo {
1083 my ($self, $color, $text) = @_; 1597 my ($self, $color, $text) = @_;
1084 1598
1096 [0.55, 0.41, 0.13], 1610 [0.55, 0.41, 0.13],
1097 [0.99, 0.77, 0.26], 1611 [0.99, 0.77, 0.26],
1098 [0.74, 0.65, 0.41], 1612 [0.74, 0.65, 0.41],
1099 ); 1613 );
1100 1614
1615 my $time = sprintf "%02d:%02d:%02d", (localtime time)[2,1,0];
1616
1617 $text = CFClient::UI::Label::escape $text;
1618 $text =~ s/\[b\](.*?)\[\/b\]/<b>\1<\/b>/g;
1619 $text =~ s/\[color=(.*?)\](.*?)\[\/color\]/<span foreground='\1'>\2<\/span>/g;
1620
1101 $LOGVIEW->add_paragraph ($color[$color], $text); 1621 $LOGVIEW->add_paragraph ($color[$color],
1622 join "\n", map "$time $_", split /\n/, $text);
1623
1624 $STATUSBOX->add ($text,
1625 group => $text,
1626 fg => $color[$color],
1627 timeout => 60,
1628 tooltip_font => $::FONT_FIXED,
1629 );
1630}
1631
1632sub conn::drawextinfo {
1633 my ($self, $color, $type, $subtype, $message) = @_;
1634
1635 $self->drawinfo ($color, $message);
1102} 1636}
1103 1637
1104sub conn::spell_add { 1638sub conn::spell_add {
1105 my ($self, $spell) = @_; 1639 my ($self, $spell) = @_;
1106 1640
1641 # TODO
1642 # create a widget dynamically, using spell face (CF::Protocol downloads them)
1107 $MAPWIDGET->add_command ("invoke $spell->{name}", $spell->{message}, sub { 1643 $MAPWIDGET->add_command ("invoke $spell->{name}", CFClient::UI::Label::escape $spell->{message});
1108 });
1109 $MAPWIDGET->add_command ("cast $spell->{name}", $spell->{message}, sub { 1644 $MAPWIDGET->add_command ("cast $spell->{name}", CFClient::UI::Label::escape $spell->{message});
1110 });
1111} 1645}
1112 1646
1113sub conn::spell_delete { 1647sub conn::spell_delete {
1114 my ($self, $spell) = @_; 1648 my ($self, $spell) = @_;
1115} 1649}
1116 1650
1117sub conn::addme_success { 1651sub conn::addme_success {
1118 my ($self) = @_; 1652 my ($self) = @_;
1119 1653
1654 $self->send ("command output-sync $CFG->{output_sync}");
1655 $self->send ("command output-count $CFG->{output_count}");
1656
1657 my $parser = new Pod::POM;
1658 my $pod = $parser->parse_file (CFClient::find_rcfile "pod/skill_help.pod");
1659
1660 my %skill_tooltip;
1661
1662 for my $head2 ($pod->head2) {
1663 $skill_tooltip{$head2->title} = CFClient::pod_to_pango $head2->content;
1664 }
1665
1120 for my $skill (values %{$self->{skill_info}}) { 1666 for my $skill (values %{$self->{skill_info}}) {
1121 $MAPWIDGET->add_command ("ready_skill $skill", "", sub { 1667 $MAPWIDGET->add_command ("ready_skill $skill",
1122 }); 1668 (CFClient::UI::Label::escape "Ready the skill '$skill'\n\n")
1669 . $skill_tooltip{$skill});
1123 $MAPWIDGET->add_command ("use_skill $skill", "", sub { 1670 $MAPWIDGET->add_command ("use_skill $skill",
1671 (CFClient::UI::Label::escape "Immediately use the skill '$skill'\n\n")
1672 . $skill_tooltip{$skill});
1673 }
1674}
1675
1676sub conn::eof {
1677 $MAPWIDGET->clr_commands;
1678
1679 stop_game;
1680}
1681
1682sub update_floorbox {
1683 $CFClient::UI::ROOT->on_refresh ($FLOORBOX => sub {
1684 return unless $CONN;
1685
1686 $FLOORBOX->clear;
1687 $FLOORBOX->add (new CFClient::UI::Empty expand => 1);
1688
1689 my $count = 7;
1690 for (@{ $CONN->{container}{0} }) {
1691 if (--$count) {
1692 $FLOORBOX->add (new CFClient::UI::InventoryItem item => $_);
1693 } else {
1694 $FLOORBOX->add (new CFClient::UI::Label text => "More...");
1695 last;
1696 }
1124 }); 1697 }
1698 });
1699
1700 $WANT_REFRESH++;
1701}
1702
1703sub conn::container_add {
1704 my ($self, $tag, $items) = @_;
1705
1706 #d# print "container_add: container $tag ($self->{player}{tag})\n";
1707
1708 if ($tag == 0) {
1709 update_floorbox;
1710 $OPENCONT = 0;
1711 $INVR_LBL->set_text ("Floor");
1712 $INVR->set_items ($self->{container}{0});
1713 } elsif ($tag == $self->{player}{tag}) {
1714 $INVR_LBL->set_text ("Player");
1715 $INV->set_items ($self->{container}{$self->{player}{tag}})
1716 } else {
1717 $OPENCONT = $tag;
1718 $INVR_LBL->set_text (CFClient::UI::InventoryItem::_item_to_desc ($self->{item}->{$OPENCONT}));
1719 $INVR->set_items ($self->{container}{$tag});
1720 }
1721
1722 # $self-<{player}{tag} => player inv
1723 #use PApp::Util; warn PApp::Util::dumpval $self->{container}{$self->{player}{tag}};
1724}
1725
1726sub conn::container_clear {
1727 my ($self, $tag) = @_;
1728
1729 #d# print "container_clear: container $tag ($self->{player}{tag})\n";
1730
1731 if ($tag == 0) {
1732 update_floorbox;
1733 $OPENCONT = 0;
1734 $INVR_LBL->set_text ("Floor");
1735 $INVR->set_items ($self->{container}{0});
1736 } elsif ($tag == $self->{player}{tag}) {
1737 $INVR_LBL->set_text ("Player");
1738 $INV->set_items ($self->{container}{$tag})
1739 } else {
1740 $OPENCONT = $tag;
1741 $INVR_LBL->set_text (CFClient::UI::InventoryItem::_item_to_desc ($self->{item}->{$OPENCONT}));
1742 $INVR->set_items ($self->{container}{$tag});
1743 }
1744
1745# use PApp::Util; warn PApp::Util::dumpval $self->{container}{0};
1746}
1747
1748sub conn::item_delete {
1749 my ($self, @items) = @_;
1750
1751 for (@items) {
1752 #d# print "item_delete: $_->{tag} from $_->{container} ($self->{player}{tag})\n";
1753
1754 if ($_->{container} == 0) {
1755 update_floorbox;
1756 $OPENCONT = 0;
1757 $INVR_LBL->set_text ("Floor");
1758 $INVR->set_items ($self->{container}{0});
1759 } elsif ($_->{container} == $self->{player}{tag}) {
1760 $INVR_LBL->set_text ("Player");
1761 $INV->set_items ($self->{container}{$self->{player}{tag}})
1762 } else {
1763 $OPENCONT = $_->{container};
1764 $INVR_LBL->set_text (CFClient::UI::InventoryItem::_item_to_desc ($self->{item}->{$OPENCONT}));
1765 $INVR->set_items ($self->{container}{$_->{container}});
1766 }
1767 }
1768}
1769
1770sub conn::item_update {
1771 my ($self, $item) = @_;
1772
1773 #d# print "item_update: $item->{tag} in $item->{container} ($self->{player}{tag}) ($OPENCONT)\n";
1774
1775 if ($item->{tag} == $OPENCONT && not ($item->{flags} & Crossfire::Protocol::F_OPEN)) {
1776 $OPENCONT = 0;
1777 $INVR_LBL->set_text ("Floor");
1778 $INVR->set_items ($self->{container}{0});
1779
1780 $item->{widget}->update_item
1781 if $item->{widget};
1782 } else {
1783 if ($item->{container} == 0) {
1784 update_floorbox;
1785 $OPENCONT = 0;
1786 $INVR_LBL->set_text ("Floor");
1787 $INVR->set_items ($self->{container}{0});
1788 } elsif ($item->{container} == $self->{player}{tag}) {
1789 $INV->set_items ($self->{container}{$item->{container}})
1790 }
1125 } 1791 }
1126} 1792}
1127 1793
1128%SDL_CB = ( 1794%SDL_CB = (
1129 CFClient::SDL_QUIT => sub { 1795 CFClient::SDL_QUIT => sub {
1130 Event::unloop -1; 1796 Event::unloop -1;
1131 }, 1797 },
1132 CFClient::SDL_VIDEORESIZE => sub { 1798 CFClient::SDL_VIDEORESIZE => sub {
1133 }, 1799 },
1134 CFClient::SDL_VIDEOEXPOSE => \&refresh, 1800 CFClient::SDL_VIDEOEXPOSE => sub {
1801 CFClient::UI::full_refresh;
1802 },
1135 CFClient::SDL_ACTIVEEVENT => sub { 1803 CFClient::SDL_ACTIVEEVENT => sub {
1136# printf "active %x %x\n", $SDL_EV->active_gain, $SDL_EV->active_state;#d# 1804# printf "active %x %x\n", $SDL_EV->active_gain, $SDL_EV->active_state;#d#
1137 }, 1805 },
1138 CFClient::SDL_KEYDOWN => sub { 1806 CFClient::SDL_KEYDOWN => sub {
1139 if ($_[0]{mod} & CFClient::KMOD_ALT && $_[0]{sym} == 13) { 1807 if ($_[0]{mod} & CFClient::KMOD_ALT && $_[0]{sym} == 13) {
1143 video_init; 1811 video_init;
1144 } else { 1812 } else {
1145 CFClient::UI::feed_sdl_key_down_event ($_[0]); 1813 CFClient::UI::feed_sdl_key_down_event ($_[0]);
1146 } 1814 }
1147 }, 1815 },
1148 CFClient::SDL_KEYUP => \&CFClient::UI::feed_sdl_key_up_event, 1816 CFClient::SDL_KEYUP => \&CFClient::UI::feed_sdl_key_up_event,
1149 CFClient::SDL_MOUSEMOTION => \&CFClient::UI::feed_sdl_motion_event, 1817 CFClient::SDL_MOUSEMOTION => \&CFClient::UI::feed_sdl_motion_event,
1150 CFClient::SDL_MOUSEBUTTONDOWN => \&CFClient::UI::feed_sdl_button_down_event, 1818 CFClient::SDL_MOUSEBUTTONDOWN => \&CFClient::UI::feed_sdl_button_down_event,
1151 CFClient::SDL_MOUSEBUTTONUP => \&CFClient::UI::feed_sdl_button_up_event, 1819 CFClient::SDL_MOUSEBUTTONUP => \&CFClient::UI::feed_sdl_button_up_event,
1152 CFClient::SDL_USEREVENT => \&audio_music_finished, 1820 CFClient::SDL_USEREVENT => sub {
1821 if ($_[0]{code} == 1) {
1822 audio_channel_finished $_[0]{data1};
1823 } elsif ($_[0]{code} == 0) {
1824 audio_music_finished;
1825 }
1826 },
1153); 1827);
1154 1828
1155############################################################################# 1829#############################################################################
1156 1830
1157$SIG{INT} = $SIG{TERM} = sub { exit }; 1831$SIG{INT} = $SIG{TERM} = sub { exit };
1158 1832
1159$TILECACHE = CFClient::db_table "tilecache";
1160$FACEMAP = CFClient::db_table "facemap";
1161
1162CFClient::read_cfg "$Crossfire::VARDIR/pclientrc";
1163
1164my %DEF_CFG = (
1165 sdl_mode => 0,
1166 width => 640,
1167 height => 480,
1168 fullscreen => 0,
1169 fast => 0,
1170 fow_enable => 1,
1171 fow_intensity => 0.45,
1172 fow_smooth => 0,
1173 gui_fontsize => 1,
1174 log_fontsize => 1,
1175 gauge_fontsize => 1,
1176 gauge_size => 0.35,
1177 stat_fontsize => 1,
1178 mapsize => 100,
1179 host => "crossfire.schmorp.de",
1180 say_command => 'say',
1181 audio_enable => 1,
1182 bgm_enable => 1,
1183 bgm_volume => 0.25,
1184);
1185
1186while (my ($k, $v) = each %DEF_CFG) {
1187 $CFG->{$k} = $v unless exists $CFG->{$k};
1188}
1189
1190sdl_init;
1191
1192@SDL_MODES = reverse
1193 grep $_->[0] >= 640 && $_->[1] >= 480,
1194 CFClient::SDL_ListModes;
1195
1196@SDL_MODES or CFClient::fatal "Unable to find a usable video mode\n(hardware accelerated opengl fullscreen)";
1197
1198$CFG->{sdl_mode} = 0 if $CFG->{sdl_mode} > @SDL_MODES;
1199
1200{ 1833{
1201 my @fonts = map CFClient::find_rcfile $_, qw(uifont.ttf uifontb.ttf uifonti.ttf uifontbi.ttf); 1834 local $SIG{__DIE__} = sub { CFClient::fatal $_[0] };
1202 1835
1836 CFClient::read_cfg "$Crossfire::VARDIR/pclientrc";
1837
1838 $TILECACHE = CFClient::db_table "tilecache";
1839 $FACEMAP = CFClient::db_table "facemap";
1840
1841 my %DEF_CFG = (
1842 sdl_mode => 0,
1843 width => 640,
1844 height => 480,
1845 fullscreen => 0,
1846 fast => 0,
1847 map_scale => 1,
1848 fow_enable => 1,
1849 fow_intensity => 0.45,
1850 fow_smooth => 0,
1851 gui_fontsize => 1,
1852 log_fontsize => 1,
1853 gauge_fontsize=> 1,
1854 gauge_size => 0.35,
1855 stat_fontsize => 1,
1856 mapsize => 100,
1857 host => "crossfire.schmorp.de",
1858 say_command => 'say',
1859 audio_enable => 1,
1860 bgm_enable => 1,
1861 bgm_volume => 0.25,
1862 output_sync => 1,
1863 output_count => 1,
1864 );
1865
1866 while (my ($k, $v) = each %DEF_CFG) {
1867 $CFG->{$k} = $v unless exists $CFG->{$k};
1868 }
1869
1870 sdl_init;
1871
1872 @SDL_MODES = reverse
1873 grep $_->[0] >= 640 && $_->[1] >= 480,
1874 CFClient::SDL_ListModes;
1875
1876 @SDL_MODES or CFClient::fatal "Unable to find a usable video mode\n(hardware accelerated opengl fullscreen)";
1877
1878 $CFG->{sdl_mode} = 0 if $CFG->{sdl_mode} > @SDL_MODES;
1879
1880 {
1881 my @fonts = map CFClient::find_rcfile "fonts/$_", qw(
1882 DejaVuSans.ttf
1883 DejaVuSansMono.ttf
1884 DejaVuSans-Bold.ttf
1885 DejaVuSansMono-Bold.ttf
1886 DejaVuSans-Oblique.ttf
1887 DejaVuSansMono-Oblique.ttf
1888 DejaVuSans-BoldOblique.ttf
1889 DejaVuSansMono-BoldOblique.ttf
1890 );
1891
1203 CFClient::add_font $_ for @fonts; 1892 CFClient::add_font $_ for @fonts;
1204 CFClient::set_font $fonts[0]; 1893
1205} 1894 CFClient::pango_init;
1206 1895
1896 $FONT_PROP = new_from_file CFClient::Font $fonts[0];
1897 $FONT_FIXED = new_from_file CFClient::Font $fonts[1];
1898
1899 $FONT_PROP->make_default;
1900 }
1901
1902# compare mono (ft) vs. rgba (cairo)
1903# ft - 1.8s, cairo 3s, even in alpha-only mode
1904# for my $rgba (0..1) {
1905# my $t1 = Time::HiRes::time;
1906# for (1..1000) {
1907# my $layout = CFClient::Layout->new ($rgba);
1908# $layout->set_text ("hallo" x 100);
1909# $layout->render;
1910# }
1911# my $t2 = Time::HiRes::time;
1912# warn $t2-$t1;
1913# }
1914
1207video_init; 1915 video_init;
1208audio_init; 1916 audio_init;
1917}
1209 1918
1210Event::loop; 1919Event::loop;
1211 1920
1212END { CFClient::SDL_Quit } 1921END { CFClient::SDL_Quit }
1213 1922
1923=head1 NAME
1214 1924
1925pclient - A Crossfire+ and Crossfire game client
1926
1927=head1 SYNOPSIS
1928
1929Just run it - no commandline arguments are supported.
1930
1931=head1 USAGE
1932
1933Pclient utilises OpenGL for all UI elements and the game. It is supposed to be used
1934fullscreen and interactively.
1935
1936=head1 AUTHOR
1937
1938Marc Lehmann <crossfire@schmorp.de>, Robin Redeker <elmex@ta-sa.org>
1939
1940
1941

Diff Legend

Removed lines
+ Added lines
< Changed lines
> Changed lines