ViewVC Help
View File | Revision Log | Show Annotations | Download File
/cvs/deliantra/Deliantra-Client/bin/pclient
Revision: 1.107
Committed: Sat Apr 15 00:04:59 2006 UTC (18 years, 1 month ago) by root
Branch: MAIN
Changes since 1.106: +1 -1 lines
Log Message:
*** empty log message ***

File Contents

# User Rev Content
1 root 1.1 #!/opt/bin/perl
2 root 1.25
3 root 1.2 use strict;
4 root 1.25 use utf8;
5 root 1.2
6 root 1.87 use Time::HiRes 'time';
7     use Event;
8 root 1.13
9     use SDL;
10     use SDL::App;
11     use SDL::Event;
12     use SDL::Surface;
13     use SDL::OpenGL;
14    
15 elmex 1.11 use Crossfire;
16 root 1.2 use Crossfire::Protocol;
17    
18 root 1.67 use CFClient;
19 root 1.72 use CFClient::UI;
20 elmex 1.10
21 root 1.63 our $VERSION = '0.1';
22    
23 root 1.96 my $MAX_FPS = 60;
24 root 1.90 my $MIN_FPS = 5; # unused as of yet
25 root 1.63
26 root 1.19 our $FACECACHE;
27    
28 root 1.87 our $LAST_REFRESH;
29     our $NOW;
30    
31 elmex 1.10 our $CFG;
32 root 1.13 our $CONN;
33 root 1.85 our $FAST; # fast, low-quality mode, possibly useful for software-rendering
34 root 1.2
35 root 1.75 our @SDL_MODES;
36 root 1.30 our $WIDTH;
37     our $HEIGHT;
38     our $FULLSCREEN;
39 root 1.99 our $FONTSIZE;
40 root 1.30
41 root 1.95 our $MAP;
42 root 1.69 our $MAPWIDGET;
43 root 1.57
44 root 1.86 our $SDL_ACTIVE;
45 root 1.58 our $SDL_EV;
46 root 1.13 our %SDL_CB;
47 root 1.18
48 root 1.30 our $ALT_ENTER_MESSAGE;
49 root 1.51 our $STATUS_LINE;
50 root 1.64 our $DEBUG_STATUS;
51 root 1.98 our $BUTTONBAR;
52 root 1.99 our $LOGVIEW;
53 elmex 1.102 our $CONSOLE;
54 root 1.30
55 root 1.82 sub status {
56     $STATUS_LINE->set_text ($_[0]);
57     my ($w, $h) = $STATUS_LINE->size_request;
58     $STATUS_LINE->size_allocate (0, $HEIGHT - $ALT_ENTER_MESSAGE->{h} - $h, $w, $h);
59     }
60    
61     sub debug {
62     $DEBUG_STATUS->set_text ($_[0]);
63     my ($w, $h) = $DEBUG_STATUS->size_request;
64     $DEBUG_STATUS->size_allocate ($WIDTH - $w, 0, $w, $h);
65     }
66    
67 root 1.84 sub start_game {
68 root 1.85 status "logging in...";
69    
70 root 1.106 my $mapsize = List::Util::min 32, List::Util::max 11, int $WIDTH * $CFG->{mapsize} * 0.01 / 32;
71 root 1.84
72 root 1.95 $MAP = new CFClient::Map $mapsize, $mapsize;
73    
74 root 1.84 $CONN = new conn
75     host => $CFG->{host},
76     port => $CFG->{port},
77     user => $CFG->{user},
78     pass => $CFG->{password},
79     mapw => $mapsize,
80     maph => $mapsize,
81     ;
82    
83 root 1.85 status "login successful";
84    
85 root 1.84 CFClient::lowdelay fileno $CONN->{fh};
86     }
87    
88     sub stop_game {
89     undef $CONN;
90     }
91    
92 root 1.81 sub config_dialog {
93 root 1.99 my $dialog = new CFClient::UI::FancyFrame
94 root 1.81 child => (my $vbox = new CFClient::UI::VBox);
95 root 1.82 $vbox->add (new CFClient::UI::Label align => 0, text => "Client Setup");
96 root 1.81 $vbox->add (my $table = new CFClient::UI::Table expand => 1, col_expand => [0, 1]);
97    
98     $table->add (0, 0, new CFClient::UI::Label align => 1, text => "Video Mode");
99     $table->add (1, 0, my $hbox = new CFClient::UI::HBox);
100    
101     $hbox->add (my $mode_slider = new CFClient::UI::Slider expand => 1, req_w => 100, range => [$CFG->{sdl_mode}, 0, scalar @SDL_MODES, 1]);
102     $hbox->add (my $mode_label = new CFClient::UI::Label height => $FONTSIZE * 0.8);
103    
104     $mode_slider->connect (changed => sub {
105     my ($self, $value) = @_;
106    
107     $CFG->{sdl_mode} = $self->{range}[0] = $value = int $value;
108     $mode_label->set_text (sprintf "%dx%d", @{$SDL_MODES[$value]});
109     });
110     $mode_slider->emit (changed => $mode_slider->{range}[0]);
111 root 1.82
112 root 1.85 $table->add (0, 1, new CFClient::UI::Label align => 1, text => "Fullscreen");
113     $table->add (1, 1, new CFClient::UI::CheckBox state => $CFG->{fullscreen}, connect_changed => sub {
114     my ($self, $value) = @_;
115     $CFG->{fullscreen} = $value;
116     });
117    
118 root 1.89 $table->add (0, 2, new CFClient::UI::Label align => 1, text => "Fast & Ugly");
119     $table->add (1, 2, new CFClient::UI::CheckBox state => $CFG->{fast}, connect_changed => sub {
120     my ($self, $value) = @_;
121     $CFG->{fast} = $value;
122     });
123    
124 root 1.91 $table->add (0, 3, new CFClient::UI::Label align => 1, text => "Fog of War");
125 root 1.97 $table->add (1, 3, new CFClient::UI::CheckBox state => $CFG->{fow_enable}, connect_changed => sub {
126     my ($self, $value) = @_;
127     $CFG->{fow_enable} = $value;
128     });
129    
130     $table->add (0, 4, new CFClient::UI::Label align => 1, text => "FoW Intensity");
131     $table->add (1, 4, new CFClient::UI::Slider range => [$CFG->{fow_intensity}, 0, 1 + 0.001, 0.001], connect_changed => sub {
132 root 1.90 my ($self, $value) = @_;
133     $CFG->{fow_intensity} = $value;
134     });
135    
136 root 1.97 $table->add (0, 5, new CFClient::UI::Label align => 1, text => "FoW Smooth");
137     $table->add (1, 5, new CFClient::UI::CheckBox state => $CFG->{fow_smooth}, connect_changed => sub {
138 root 1.91 my ($self, $value) = @_;
139     $CFG->{fow_smooth} = $value;
140     });
141    
142 root 1.105 $table->add (0, 5, new CFClient::UI::Label align => 1, text => "Log Fontsize");
143     $table->add (1, 5, new CFClient::UI::Slider range => [$CFG->{log_fontsize}, 8, 30, 1], connect_changed => sub {
144     my ($self, $value) = @_;
145     $LOGVIEW->set_fontsize ($CFG->{log_fontsize} = int $value);
146     });
147    
148 root 1.97 $table->add (1, 6, new CFClient::UI::Button expand => 1, align => 0, text => "Apply", connect_activate => sub {
149 root 1.83 destroy_screen ();
150     init_screen ();
151 root 1.82 });
152 root 1.81
153 root 1.89 $vbox->add (new CFClient::UI::Label align => 0, text => "Server Setup");
154 root 1.82 $vbox->add (my $table = new CFClient::UI::Table expand => 1, col_expand => [0, 1]);
155     $table->add (0, 2, new CFClient::UI::Label align => 1, text => "Host");
156 elmex 1.101 $table->add (1, 2, my $host = new CFClient::UI::Entry text => $CFG->{host}, connect_changed => sub {
157     my ($self, $value) = @_;
158     $CFG->{host} = $value;
159     });
160    
161 root 1.82 $table->add (0, 3, new CFClient::UI::Label align => 1, text => "Port");
162 elmex 1.101 $table->add (1, 3, my $port = new CFClient::UI::Entry text => $CFG->{port}, connect_changed => sub {
163     my ($self, $value) = @_;
164     $CFG->{port} = $value;
165     });
166 root 1.81
167 root 1.82 $table->add (0, 4, new CFClient::UI::Label align => 1, text => "Username");
168 elmex 1.101 $table->add (1, 4, my $user = new CFClient::UI::Entry text => $CFG->{user}, connect_changed => sub {
169     my ($self, $value) = @_;
170     $CFG->{user} = $value;
171     });
172 root 1.81
173 root 1.82 $table->add (0, 5, new CFClient::UI::Label align => 1, text => "Password");
174 elmex 1.101 $table->add (1, 5, my $pass = new CFClient::UI::Entry text => $CFG->{password}, hidden => 1, connect_changed => sub {
175     my ($self, $value) = @_;
176     $CFG->{password} = $value;
177     });
178    
179     $table->add (0, 6, new CFClient::UI::Label align => 1, text => "Def. say cmd");
180     $table->add (1, 6, my $saycmd = new CFClient::UI::Entry text => $CFG->{say_command}, connect_changed => sub {
181     my ($self, $value) = @_;
182     $CFG->{say_command} = $value;
183     });
184 root 1.81
185 elmex 1.101 $table->add (0, 7, new CFClient::UI::Label align => 1, text => "Map Size");
186     $table->add (1, 7, new CFClient::UI::Slider
187 root 1.81 req_w => 100,
188     range => [$CFG->{mapsize}, 10, 100 + 1, 1],
189     connect_changed => sub {
190     my ($self, $value) = @_;
191    
192     $CFG->{mapsize} = $self->{range}[0] = $value = int $value;
193     },
194     );
195    
196 elmex 1.101 $table->add (1, 8, new CFClient::UI::Button expand => 1, align => 0, text => "Login", connect_activate => sub {
197 root 1.84 start_game;
198 root 1.82 });
199    
200 root 1.81 $vbox->add (my $hbox = new CFClient::UI::HBox);
201 root 1.82
202 root 1.81 $hbox->add (new CFClient::UI::Button expand => 1, align => 0, text => "Save", connect_activate => sub {
203 root 1.84 CFClient::write_cfg "$Crossfire::VARDIR/pclientrc";
204 root 1.82 status "Configuration Saved";
205 root 1.81 });
206 root 1.98
207     $dialog
208 root 1.81 }
209 root 1.58
210 root 1.99 sub console_window {
211     my $window = new CFClient::UI::FancyFrame
212     border_bg => [1, 1, 1, 0.5],
213     bg => [0.3, 0.3, 0.3, 0.8],
214 root 1.105 user_w => $::WIDTH/4,
215     user_h => $::HEIGHT,
216 root 1.99 child => (my $vbox = new CFClient::UI::VBox);
217    
218 root 1.105 $vbox->add ($LOGVIEW = new CFClient::UI::TextView
219     expand => 1,
220     fontsize => $::CFG->{log_fontsize},
221     );
222    
223 elmex 1.100 $vbox->add (my $input = new CFClient::UI::LineEntry);
224     $input->connect (activate => sub {
225     my ($input, $text) = @_;
226     $input->set_text ('');
227    
228     if ($text =~ /^\/(.*)/) {
229     $::CONN->user_send ("command $1");
230     } else {
231 elmex 1.101 my $say_cmd = $::CFG->{say_command} || 'say';
232     $::CONN->user_send ("command $say_cmd $text");
233 elmex 1.100 }
234     1
235     });
236 elmex 1.102 $input->connect (escape => sub {
237     $MAPWIDGET->focus_in
238     });
239     $input->focus_in;
240    
241     $CONSOLE = {
242     window => $window,
243     input => $input
244     };
245 root 1.99
246     $window
247     }
248    
249 root 1.89 sub sdl_init {
250 root 1.99 SDL::Init SDL_INIT_AUDIO | SDL_INIT_VIDEO
251 root 1.89 and die "SDL::Init failed!\n";
252     }
253    
254 root 1.13 sub init_screen {
255 root 1.89 sdl_init;
256    
257 root 1.84 ($WIDTH, $HEIGHT) = @{ $SDL_MODES[$CFG->{sdl_mode}] };
258     $FULLSCREEN = $CFG->{fullscreen};
259 root 1.89 $FAST = $CFG->{fast};
260 root 1.84
261 root 1.86 SDL::GLSetAttribute SDL_GL_RED_SIZE, 5;
262     SDL::GLSetAttribute SDL_GL_GREEN_SIZE, 5;
263     SDL::GLSetAttribute SDL_GL_BLUE_SIZE, 5;
264     SDL::GLSetAttribute SDL_GL_ALPHA_SIZE, 0;
265    
266     SDL::GLSetAttribute SDL_GL_ACCUM_RED_SIZE, 0;
267     SDL::GLSetAttribute SDL_GL_ACCUM_GREEN_SIZE, 0;
268     SDL::GLSetAttribute SDL_GL_ACCUM_BLUE_SIZE, 0;
269     SDL::GLSetAttribute SDL_GL_ACCUM_ALPHA_SIZE, 0;
270    
271     SDL::GLSetAttribute SDL_GL_DOUBLEBUFFER, 1;
272     SDL::GLSetAttribute SDL_GL_BUFFER_SIZE, 15;
273     SDL::GLSetAttribute SDL_GL_DEPTH_SIZE, 0;
274    
275     SDL::SetVideoMode $WIDTH, $HEIGHT, 0,
276     SDL_HWSURFACE | SDL_ANYFORMAT | SDL_OPENGL | SDL_DOUBLEBUF
277     | ($FULLSCREEN ? SDL_FULLSCREEN : 0)
278     or die "SDL::SetVideoMode failed!\n";
279    
280     SDL::WMSetCaption "Crossfire+ Client", "Crossfire+";
281 root 1.2
282 root 1.58 $SDL_EV = new SDL::Event;
283     $SDL_EV->set_unicode (1);
284    
285 root 1.86 $SDL_ACTIVE = 1;
286    
287 root 1.87 $LAST_REFRESH = time - 0.01;
288 root 1.45
289 root 1.67 CFClient::gl_init;
290 root 1.30
291 root 1.77 $FONTSIZE = int $HEIGHT / 40;
292 root 1.39
293 root 1.52 #############################################################################
294    
295 root 1.99 $DEBUG_STATUS = new CFClient::UI::Label padding => 0, z => 100;
296 root 1.72 $CFClient::UI::TOPLEVEL->add ($DEBUG_STATUS);
297 root 1.52
298 root 1.72 $STATUS_LINE = new CFClient::UI::Label
299 root 1.77 padding => 0,
300 root 1.99 y => $HEIGHT * 44 / 45 - $FONTSIZE;
301 root 1.72 $CFClient::UI::TOPLEVEL->add ($STATUS_LINE);
302 root 1.51
303 root 1.72 $ALT_ENTER_MESSAGE = new CFClient::UI::Label
304 root 1.77 padding => 0,
305 root 1.99 y => $HEIGHT * 44 / 45,
306     height => $HEIGHT / 45,
307 root 1.68 text => "Use <b>Alt-Enter</b> to toggle fullscreen mode";
308 root 1.72 $CFClient::UI::TOPLEVEL->add ($ALT_ENTER_MESSAGE);
309 root 1.30
310 root 1.98 $CFClient::UI::TOPLEVEL->add ($MAPWIDGET = new CFClient::UI::MapWidget);
311 root 1.69 $MAPWIDGET->focus_in;
312 elmex 1.102 $MAPWIDGET->connect (activate_console => sub {
313 elmex 1.103 my ($mapwidget, $preset) = @_;
314    
315 elmex 1.102 if ($CONSOLE) {
316     $CONSOLE->{input}->focus_in;
317 elmex 1.103
318     if ($preset && $CONSOLE->{input}->get_text eq '') {
319     $CONSOLE->{input}->set_text ($preset);
320     }
321 elmex 1.102 }
322     });
323 root 1.81
324 root 1.98 $CFClient::UI::TOPLEVEL->add ($BUTTONBAR = new CFClient::UI::HBox);
325    
326 root 1.99 $BUTTONBAR->add (my $setup = new CFClient::UI::Flopper x => 0, y => 0, text => "Setup", other => config_dialog, state => 1);
327     $BUTTONBAR->add (my $setup = new CFClient::UI::Flopper x => 0, y => 0, text => "Console", other => console_window);
328 root 1.2 }
329    
330 root 1.62 sub destroy_screen {
331 root 1.81 $CFClient::UI::TOPLEVEL->{children} = [];
332 root 1.86 undef $SDL_ACTIVE;
333 root 1.62 undef $SDL_EV;
334     SDL::Quit;
335     }
336    
337 root 1.87 my %animate_object;
338     my $animate_timer;
339    
340     my $want_refresh;
341     my $can_refresh;
342    
343     my $fps = 9;
344    
345 root 1.30 sub force_refresh {
346 root 1.87 $fps = $fps * 0.95 + 1 / ($NOW - $LAST_REFRESH) * 0.05;
347     debug sprintf "%3.2f", $fps;
348    
349 root 1.96 $want_refresh = 0;
350 root 1.87 $can_refresh = 0;
351    
352 root 1.72 $CFClient::UI::TOPLEVEL->draw;
353 root 1.1
354 root 1.2 SDL::GLSwapBuffers;
355 root 1.87
356     $LAST_REFRESH = $NOW;
357 root 1.1 }
358    
359 root 1.87 my $refresh_watcher = Event->timer (after => 0, hard => 1, interval => 1 / $MAX_FPS, cb => sub {
360     $NOW = time;
361    
362     ($SDL_CB{$SDL_EV->type} || sub { warn "unhandled event ", $SDL_EV->type })->()
363     while $SDL_EV->poll;
364    
365     if (%animate_object) {
366     $_->animate ($LAST_REFRESH - $NOW) for values %animate_object;
367     $want_refresh++;
368     }
369    
370     if ($want_refresh) {
371     force_refresh;
372     } else {
373     $can_refresh = 1;
374     }
375     });
376 root 1.64
377 root 1.30 sub refresh {
378 root 1.87 $want_refresh++;
379 root 1.30 }
380    
381 root 1.45 sub animation_start {
382     my ($widget) = @_;
383 root 1.87 $animate_object{$widget} = $widget;
384 root 1.45 }
385    
386     sub animation_stop {
387     my ($widget) = @_;
388 root 1.87 delete $animate_object{$widget};
389 root 1.45 }
390    
391 root 1.2 @conn::ISA = Crossfire::Protocol::;
392 root 1.1
393 root 1.89 sub conn::user_send {
394 root 1.88 my ($self, $command) = @_;
395    
396 root 1.89 $self->send ($command);
397 root 1.88 status $command;
398     }
399    
400 root 1.94 sub conn::feed_map1a {
401     my ($self, $data) = @_;
402    
403 root 1.95 # $self->Crossfire::Protocol::feed_map1a ($data);
404 root 1.1
405 root 1.95 $MAP->scroll (delete $self->{delayed_scroll_x}, delete $self->{delayed_scroll_y});
406     $MAP->map1a_update ($data);
407 root 1.69 $MAPWIDGET->update;
408 root 1.1 }
409    
410 root 1.95 #sub conn::map_update {
411     # my ($self, $dirty) = @_;
412     #
413     # $MAPWIDGET->update;
414     #}
415 root 1.1
416 root 1.2 sub conn::map_clear {
417 root 1.1 my ($self) = @_;
418    
419 root 1.95 $MAP->clear;
420    
421 root 1.45 # refresh;
422 root 1.1 }
423    
424 root 1.19 sub conn::face_find {
425     my ($self, $face) = @_;
426    
427     $FACECACHE->{"$face->{chksum},$face->{name}"}
428     }
429    
430 root 1.2 sub conn::face_update {
431 root 1.95 my ($self, $facenum, $face) = @_;
432 root 1.19
433     $FACECACHE->{"$face->{chksum},$face->{name}"} = $face->{image};
434 root 1.1
435 root 1.95 my $tex = $face->{texture} = new_from_image CFClient::Texture delete $face->{image};
436    
437 root 1.107 $MAP->set_texture ($facenum, @$tex{qw(name w h s t)}, @{$tex->{minified}});
438 root 1.95 $MAPWIDGET->update;
439 root 1.1 }
440    
441 root 1.33 sub conn::query {
442     my ($self, $flags, $prompt) = @_;
443    
444 root 1.99 #TODO
445 root 1.33 warn "<<<<QUERY:$flags:$prompt>>>\n";#d#
446     }
447    
448 root 1.99 sub conn::drawinfo {
449     my ($self, $color, $text) = @_;
450    
451     my @color = (
452     [1.00, 1.00, 1.00], #[0.00, 0.00, 0.00],
453     [1.00, 1.00, 1.00],
454     [0.00, 0.00, 0.55],
455     [1.00, 0.00, 0.00],
456     [1.00, 0.54, 0.00],
457     [0.11, 0.56, 1.00],
458     [0.93, 0.46, 0.00],
459     [0.18, 0.54, 0.34],
460     [0.56, 0.73, 0.56],
461     [0.80, 0.80, 0.80],
462     [0.55, 0.41, 0.13],
463     [0.99, 0.77, 0.26],
464     [0.74, 0.65, 0.41],
465     );
466    
467     $LOGVIEW->add_paragraph ($color[$color], $text);
468     }
469    
470 root 1.87 %SDL_CB = (
471     SDL_QUIT() => sub {
472     Event::unloop -1;
473     },
474     SDL_VIDEORESIZE() => sub {
475     },
476     SDL_VIDEOEXPOSE() => sub {
477     refresh;
478     },
479     SDL_KEYDOWN() => sub {
480     if ($SDL_EV->key_mod & KMOD_ALT && $SDL_EV->key_sym == SDLK_RETURN) {
481     # alt-enter
482     destroy_screen;
483 root 1.99 $CFG->{fullscreen} = !$CFG->{fullscreen};
484 root 1.87 init_screen;
485     } else {
486     CFClient::UI::feed_sdl_key_down_event ($SDL_EV);
487 elmex 1.23 }
488 root 1.87 },
489     SDL_KEYUP() => sub {
490     CFClient::UI::feed_sdl_key_up_event ($SDL_EV);
491     },
492     SDL_MOUSEMOTION() => sub {
493     CFClient::UI::feed_sdl_motion_event ($SDL_EV);
494     },
495     SDL_MOUSEBUTTONDOWN() => sub {
496     CFClient::UI::feed_sdl_button_down_event ($SDL_EV);
497     },
498     SDL_MOUSEBUTTONUP() => sub {
499     CFClient::UI::feed_sdl_button_up_event ($SDL_EV);
500     },
501     SDL_ACTIVEEVENT() => sub {
502     # printf "active %x %x\n", $SDL_EV->active_gain, $SDL_EV->active_state;#d#
503     },
504     );
505 elmex 1.23
506 root 1.1 #############################################################################
507    
508 root 1.67 CFClient::read_cfg "$Crossfire::VARDIR/pclientrc";
509 elmex 1.10
510 root 1.90 my %DEF_CFG = (
511 root 1.105 sdl_mode => 0,
512 root 1.90 width => 640,
513     height => 480,
514 root 1.105 fullscreen => 0,
515 root 1.90 fast => 0,
516 root 1.97 fow_enable => 1,
517 root 1.90 fow_intensity => 0.45,
518 root 1.92 fow_smooth => 0,
519 root 1.105 log_fontsize => 14,
520 root 1.90 mapsize => 100,
521     host => "crossfire.schmorp.de",
522     port => 13327,
523 elmex 1.101 say_command => 'say',
524 root 1.90 );
525    
526     while (my ($k, $v) = each %DEF_CFG) {
527     $CFG->{$k} = $v unless exists $CFG->{$k};
528     }
529 elmex 1.12
530 root 1.89 sdl_init;
531 root 1.87
532 root 1.93 @SDL_MODES = reverse
533     grep $_->[0] >= 640 && $_->[1] >= 480,
534     map [SDL::RectW ($_), SDL::RectH ($_)],
535     @{ SDL::ListModes 0, SDL_FULLSCREEN | SDL_HWSURFACE | SDL_OPENGL };
536 root 1.87
537 root 1.89 @SDL_MODES or CFClient::fatal "Unable to find a usable video mode\n(hardware accelerated opengl fullscreen)";
538    
539     $CFG->{sdl_mode} = 0 if $CFG->{sdl_mode} > @SDL_MODES;
540    
541 root 1.87 init_screen;
542    
543 root 1.65 {
544 root 1.67 my @fonts = map CFClient::find_rcfile $_, qw(uifont.ttf uifontb.ttf uifonti.ttf uifontbi.ttf);
545 root 1.65
546 root 1.67 CFClient::add_font $_ for @fonts;
547     CFClient::set_font $fonts[0];
548 root 1.65 }
549 root 1.40
550 root 1.24 $FACECACHE = eval { Crossfire::load_ref "$Crossfire::VARDIR/pclient.faces" } || {};
551    
552 root 1.87 Event::loop;
553 root 1.19
554 root 1.68 Crossfire::save_ref $FACECACHE, "$Crossfire::VARDIR/pclient.faces";
555 root 1.82