ViewVC Help
View File | Revision Log | Show Annotations | Download File
/cvs/deliantra/Deliantra-Client/bin/pclient
Revision: 1.111
Committed: Sat Apr 15 12:46:48 2006 UTC (18 years, 1 month ago) by root
Branch: MAIN
Changes since 1.110: +26 -20 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.111 sub client_setup {
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.111
153     $dialog
154     }
155    
156     sub server_setup {
157     my $dialog = new CFClient::UI::FancyFrame
158     child => (my $vbox = new CFClient::UI::VBox);
159 root 1.81
160 root 1.89 $vbox->add (new CFClient::UI::Label align => 0, text => "Server Setup");
161 root 1.82 $vbox->add (my $table = new CFClient::UI::Table expand => 1, col_expand => [0, 1]);
162     $table->add (0, 2, new CFClient::UI::Label align => 1, text => "Host");
163 elmex 1.101 $table->add (1, 2, my $host = new CFClient::UI::Entry text => $CFG->{host}, connect_changed => sub {
164     my ($self, $value) = @_;
165     $CFG->{host} = $value;
166     });
167    
168 root 1.82 $table->add (0, 3, new CFClient::UI::Label align => 1, text => "Port");
169 elmex 1.101 $table->add (1, 3, my $port = new CFClient::UI::Entry text => $CFG->{port}, connect_changed => sub {
170     my ($self, $value) = @_;
171     $CFG->{port} = $value;
172     });
173 root 1.81
174 root 1.82 $table->add (0, 4, new CFClient::UI::Label align => 1, text => "Username");
175 elmex 1.101 $table->add (1, 4, my $user = new CFClient::UI::Entry text => $CFG->{user}, connect_changed => sub {
176     my ($self, $value) = @_;
177     $CFG->{user} = $value;
178     });
179 root 1.81
180 root 1.82 $table->add (0, 5, new CFClient::UI::Label align => 1, text => "Password");
181 elmex 1.101 $table->add (1, 5, my $pass = new CFClient::UI::Entry text => $CFG->{password}, hidden => 1, connect_changed => sub {
182     my ($self, $value) = @_;
183     $CFG->{password} = $value;
184     });
185    
186     $table->add (0, 6, new CFClient::UI::Label align => 1, text => "Def. say cmd");
187     $table->add (1, 6, my $saycmd = new CFClient::UI::Entry text => $CFG->{say_command}, connect_changed => sub {
188     my ($self, $value) = @_;
189     $CFG->{say_command} = $value;
190     });
191 root 1.81
192 elmex 1.101 $table->add (0, 7, new CFClient::UI::Label align => 1, text => "Map Size");
193     $table->add (1, 7, new CFClient::UI::Slider
194 root 1.81 req_w => 100,
195     range => [$CFG->{mapsize}, 10, 100 + 1, 1],
196     connect_changed => sub {
197     my ($self, $value) = @_;
198    
199     $CFG->{mapsize} = $self->{range}[0] = $value = int $value;
200     },
201     );
202    
203 elmex 1.101 $table->add (1, 8, new CFClient::UI::Button expand => 1, align => 0, text => "Login", connect_activate => sub {
204 root 1.84 start_game;
205 root 1.82 });
206    
207 root 1.98 $dialog
208 root 1.81 }
209 root 1.58
210 root 1.111 sub message_window {
211 root 1.99 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.109 #SDL::Init SDL_INIT_AUDIO | SDL_INIT_VIDEO | SDL_INIT_NOPARACHUTE
251     SDL::Init SDL_INIT_AUDIO | SDL_INIT_VIDEO
252 root 1.89 and die "SDL::Init failed!\n";
253     }
254    
255 root 1.13 sub init_screen {
256 root 1.89 sdl_init;
257    
258 root 1.84 ($WIDTH, $HEIGHT) = @{ $SDL_MODES[$CFG->{sdl_mode}] };
259     $FULLSCREEN = $CFG->{fullscreen};
260 root 1.89 $FAST = $CFG->{fast};
261 root 1.84
262 root 1.86 SDL::GLSetAttribute SDL_GL_RED_SIZE, 5;
263     SDL::GLSetAttribute SDL_GL_GREEN_SIZE, 5;
264     SDL::GLSetAttribute SDL_GL_BLUE_SIZE, 5;
265     SDL::GLSetAttribute SDL_GL_ALPHA_SIZE, 0;
266    
267     SDL::GLSetAttribute SDL_GL_ACCUM_RED_SIZE, 0;
268     SDL::GLSetAttribute SDL_GL_ACCUM_GREEN_SIZE, 0;
269     SDL::GLSetAttribute SDL_GL_ACCUM_BLUE_SIZE, 0;
270     SDL::GLSetAttribute SDL_GL_ACCUM_ALPHA_SIZE, 0;
271    
272     SDL::GLSetAttribute SDL_GL_DOUBLEBUFFER, 1;
273     SDL::GLSetAttribute SDL_GL_BUFFER_SIZE, 15;
274     SDL::GLSetAttribute SDL_GL_DEPTH_SIZE, 0;
275    
276     SDL::SetVideoMode $WIDTH, $HEIGHT, 0,
277     SDL_HWSURFACE | SDL_ANYFORMAT | SDL_OPENGL | SDL_DOUBLEBUF
278     | ($FULLSCREEN ? SDL_FULLSCREEN : 0)
279     or die "SDL::SetVideoMode failed!\n";
280    
281     SDL::WMSetCaption "Crossfire+ Client", "Crossfire+";
282 root 1.2
283 root 1.58 $SDL_EV = new SDL::Event;
284     $SDL_EV->set_unicode (1);
285    
286 root 1.86 $SDL_ACTIVE = 1;
287    
288 root 1.87 $LAST_REFRESH = time - 0.01;
289 root 1.45
290 root 1.67 CFClient::gl_init;
291 root 1.30
292 root 1.77 $FONTSIZE = int $HEIGHT / 40;
293 root 1.39
294 root 1.52 #############################################################################
295    
296 root 1.99 $DEBUG_STATUS = new CFClient::UI::Label padding => 0, z => 100;
297 root 1.111 $CFClient::UI::ROOT->add ($DEBUG_STATUS);
298 root 1.52
299 root 1.72 $STATUS_LINE = new CFClient::UI::Label
300 root 1.77 padding => 0,
301 root 1.99 y => $HEIGHT * 44 / 45 - $FONTSIZE;
302 root 1.111 $CFClient::UI::ROOT->add ($STATUS_LINE);
303 root 1.51
304 root 1.72 $ALT_ENTER_MESSAGE = new CFClient::UI::Label
305 root 1.77 padding => 0,
306 root 1.99 y => $HEIGHT * 44 / 45,
307     height => $HEIGHT / 45,
308 root 1.68 text => "Use <b>Alt-Enter</b> to toggle fullscreen mode";
309 root 1.111 $CFClient::UI::ROOT->add ($ALT_ENTER_MESSAGE);
310 root 1.30
311 root 1.111 $CFClient::UI::ROOT->add ($MAPWIDGET = new CFClient::UI::MapWidget);
312 root 1.69 $MAPWIDGET->focus_in;
313 elmex 1.102 $MAPWIDGET->connect (activate_console => sub {
314 elmex 1.103 my ($mapwidget, $preset) = @_;
315    
316 elmex 1.102 if ($CONSOLE) {
317     $CONSOLE->{input}->focus_in;
318 elmex 1.103
319     if ($preset && $CONSOLE->{input}->get_text eq '') {
320     $CONSOLE->{input}->set_text ($preset);
321     }
322 elmex 1.102 }
323     });
324 root 1.81
325 root 1.111 $CFClient::UI::ROOT->add ($BUTTONBAR = new CFClient::UI::HBox);
326    
327     $BUTTONBAR->add (new CFClient::UI::Flopper text => "Client Setup", other => client_setup);
328     $BUTTONBAR->add (new CFClient::UI::Flopper text => "Server Setup", other => server_setup);
329     $BUTTONBAR->add (new CFClient::UI::Flopper text => "Message Window", other => message_window);
330    
331     $BUTTONBAR->add (new CFClient::UI::Button text => "Save Config", connect_activate => sub {
332     CFClient::write_cfg "$Crossfire::VARDIR/pclientrc";
333     status "Configuration Saved";
334     });
335 root 1.98
336 root 1.111 $BUTTONBAR->{children}[0]->emit ("activate");
337 root 1.2 }
338    
339 root 1.62 sub destroy_screen {
340 root 1.111 $CFClient::UI::ROOT->{children} = [];
341 root 1.86 undef $SDL_ACTIVE;
342 root 1.62 undef $SDL_EV;
343     SDL::Quit;
344     }
345    
346 root 1.87 my %animate_object;
347     my $animate_timer;
348    
349     my $want_refresh;
350     my $can_refresh;
351    
352     my $fps = 9;
353    
354 root 1.30 sub force_refresh {
355 root 1.87 $fps = $fps * 0.95 + 1 / ($NOW - $LAST_REFRESH) * 0.05;
356     debug sprintf "%3.2f", $fps;
357    
358 root 1.96 $want_refresh = 0;
359 root 1.87 $can_refresh = 0;
360    
361 root 1.111 $CFClient::UI::ROOT->draw;
362 root 1.1
363 root 1.2 SDL::GLSwapBuffers;
364 root 1.87
365     $LAST_REFRESH = $NOW;
366 root 1.1 }
367    
368 root 1.87 my $refresh_watcher = Event->timer (after => 0, hard => 1, interval => 1 / $MAX_FPS, cb => sub {
369     $NOW = time;
370    
371     ($SDL_CB{$SDL_EV->type} || sub { warn "unhandled event ", $SDL_EV->type })->()
372     while $SDL_EV->poll;
373    
374     if (%animate_object) {
375     $_->animate ($LAST_REFRESH - $NOW) for values %animate_object;
376     $want_refresh++;
377     }
378    
379     if ($want_refresh) {
380     force_refresh;
381     } else {
382     $can_refresh = 1;
383     }
384     });
385 root 1.64
386 root 1.30 sub refresh {
387 root 1.87 $want_refresh++;
388 root 1.30 }
389    
390 root 1.45 sub animation_start {
391     my ($widget) = @_;
392 root 1.87 $animate_object{$widget} = $widget;
393 root 1.45 }
394    
395     sub animation_stop {
396     my ($widget) = @_;
397 root 1.87 delete $animate_object{$widget};
398 root 1.45 }
399    
400 root 1.2 @conn::ISA = Crossfire::Protocol::;
401 root 1.1
402 root 1.89 sub conn::user_send {
403 root 1.88 my ($self, $command) = @_;
404    
405 root 1.89 $self->send ($command);
406 root 1.88 status $command;
407     }
408    
409 root 1.94 sub conn::feed_map1a {
410     my ($self, $data) = @_;
411    
412 root 1.95 # $self->Crossfire::Protocol::feed_map1a ($data);
413 root 1.1
414 root 1.95 $MAP->scroll (delete $self->{delayed_scroll_x}, delete $self->{delayed_scroll_y});
415     $MAP->map1a_update ($data);
416 root 1.69 $MAPWIDGET->update;
417 root 1.1 }
418    
419 root 1.95 #sub conn::map_update {
420     # my ($self, $dirty) = @_;
421     #
422     # $MAPWIDGET->update;
423     #}
424 root 1.1
425 root 1.2 sub conn::map_clear {
426 root 1.1 my ($self) = @_;
427    
428 root 1.95 $MAP->clear;
429    
430 root 1.45 # refresh;
431 root 1.1 }
432    
433 root 1.19 sub conn::face_find {
434     my ($self, $face) = @_;
435    
436     $FACECACHE->{"$face->{chksum},$face->{name}"}
437     }
438    
439 root 1.2 sub conn::face_update {
440 root 1.95 my ($self, $facenum, $face) = @_;
441 root 1.19
442     $FACECACHE->{"$face->{chksum},$face->{name}"} = $face->{image};
443 root 1.1
444 root 1.108 my $tex = $face->{texture} =
445     new_from_image CFClient::Texture
446     delete $face->{image}, minify => 1;
447 root 1.95
448 root 1.107 $MAP->set_texture ($facenum, @$tex{qw(name w h s t)}, @{$tex->{minified}});
449 root 1.95 $MAPWIDGET->update;
450 root 1.1 }
451    
452 root 1.33 sub conn::query {
453     my ($self, $flags, $prompt) = @_;
454    
455 root 1.99 #TODO
456 root 1.33 warn "<<<<QUERY:$flags:$prompt>>>\n";#d#
457     }
458    
459 root 1.99 sub conn::drawinfo {
460     my ($self, $color, $text) = @_;
461    
462     my @color = (
463     [1.00, 1.00, 1.00], #[0.00, 0.00, 0.00],
464     [1.00, 1.00, 1.00],
465     [0.00, 0.00, 0.55],
466     [1.00, 0.00, 0.00],
467     [1.00, 0.54, 0.00],
468     [0.11, 0.56, 1.00],
469     [0.93, 0.46, 0.00],
470     [0.18, 0.54, 0.34],
471     [0.56, 0.73, 0.56],
472     [0.80, 0.80, 0.80],
473     [0.55, 0.41, 0.13],
474     [0.99, 0.77, 0.26],
475     [0.74, 0.65, 0.41],
476     );
477    
478     $LOGVIEW->add_paragraph ($color[$color], $text);
479     }
480    
481 root 1.87 %SDL_CB = (
482     SDL_QUIT() => sub {
483     Event::unloop -1;
484     },
485     SDL_VIDEORESIZE() => sub {
486     },
487     SDL_VIDEOEXPOSE() => sub {
488     refresh;
489     },
490     SDL_KEYDOWN() => sub {
491     if ($SDL_EV->key_mod & KMOD_ALT && $SDL_EV->key_sym == SDLK_RETURN) {
492     # alt-enter
493     destroy_screen;
494 root 1.99 $CFG->{fullscreen} = !$CFG->{fullscreen};
495 root 1.87 init_screen;
496     } else {
497     CFClient::UI::feed_sdl_key_down_event ($SDL_EV);
498 elmex 1.23 }
499 root 1.87 },
500     SDL_KEYUP() => sub {
501     CFClient::UI::feed_sdl_key_up_event ($SDL_EV);
502     },
503     SDL_MOUSEMOTION() => sub {
504     CFClient::UI::feed_sdl_motion_event ($SDL_EV);
505     },
506     SDL_MOUSEBUTTONDOWN() => sub {
507     CFClient::UI::feed_sdl_button_down_event ($SDL_EV);
508     },
509     SDL_MOUSEBUTTONUP() => sub {
510     CFClient::UI::feed_sdl_button_up_event ($SDL_EV);
511     },
512     SDL_ACTIVEEVENT() => sub {
513     # printf "active %x %x\n", $SDL_EV->active_gain, $SDL_EV->active_state;#d#
514     },
515     );
516 elmex 1.23
517 root 1.1 #############################################################################
518    
519 root 1.67 CFClient::read_cfg "$Crossfire::VARDIR/pclientrc";
520 elmex 1.10
521 root 1.90 my %DEF_CFG = (
522 root 1.105 sdl_mode => 0,
523 root 1.90 width => 640,
524     height => 480,
525 root 1.105 fullscreen => 0,
526 root 1.90 fast => 0,
527 root 1.97 fow_enable => 1,
528 root 1.90 fow_intensity => 0.45,
529 root 1.92 fow_smooth => 0,
530 root 1.105 log_fontsize => 14,
531 root 1.90 mapsize => 100,
532     host => "crossfire.schmorp.de",
533     port => 13327,
534 elmex 1.101 say_command => 'say',
535 root 1.90 );
536    
537     while (my ($k, $v) = each %DEF_CFG) {
538     $CFG->{$k} = $v unless exists $CFG->{$k};
539     }
540 elmex 1.12
541 root 1.89 sdl_init;
542 root 1.87
543 root 1.93 @SDL_MODES = reverse
544     grep $_->[0] >= 640 && $_->[1] >= 480,
545     map [SDL::RectW ($_), SDL::RectH ($_)],
546     @{ SDL::ListModes 0, SDL_FULLSCREEN | SDL_HWSURFACE | SDL_OPENGL };
547 root 1.87
548 root 1.89 @SDL_MODES or CFClient::fatal "Unable to find a usable video mode\n(hardware accelerated opengl fullscreen)";
549    
550     $CFG->{sdl_mode} = 0 if $CFG->{sdl_mode} > @SDL_MODES;
551    
552 root 1.87 init_screen;
553    
554 root 1.65 {
555 root 1.67 my @fonts = map CFClient::find_rcfile $_, qw(uifont.ttf uifontb.ttf uifonti.ttf uifontbi.ttf);
556 root 1.65
557 root 1.67 CFClient::add_font $_ for @fonts;
558     CFClient::set_font $fonts[0];
559 root 1.65 }
560 root 1.40
561 root 1.24 $FACECACHE = eval { Crossfire::load_ref "$Crossfire::VARDIR/pclient.faces" } || {};
562    
563 root 1.87 Event::loop;
564 root 1.19
565 root 1.68 Crossfire::save_ref $FACECACHE, "$Crossfire::VARDIR/pclient.faces";
566 root 1.82