1 | #!/opt/bin/perl |
1 | #!/opt/bin/perl |
|
|
2 | |
2 | use strict; |
3 | use strict; |
|
|
4 | use utf8; |
3 | |
5 | |
4 | use Glib; |
6 | use Glib; |
5 | use Gtk2 -init; |
7 | use Gtk2 -init; |
6 | |
8 | |
7 | use SDL; |
9 | use SDL; |
… | |
… | |
13 | |
15 | |
14 | use Crossfire; |
16 | use Crossfire; |
15 | use Crossfire::Client; |
17 | use Crossfire::Client; |
16 | use Crossfire::Protocol; |
18 | use Crossfire::Protocol; |
17 | |
19 | |
18 | use Client::Util; |
|
|
19 | use Client::Widget; |
20 | use Crossfire::Client::Widget; |
20 | |
21 | |
21 | our $FACECACHE; |
22 | our $FACECACHE; |
22 | |
23 | |
23 | our $VERSION = '0.1'; |
24 | our $VERSION = '0.1'; |
|
|
25 | |
|
|
26 | our %GL_EXT; |
24 | |
27 | |
25 | our $CFG; |
28 | our $CFG; |
26 | our $CONN; |
29 | our $CONN; |
|
|
30 | |
|
|
31 | our $WIDTH; |
|
|
32 | our $HEIGHT; |
|
|
33 | our $FULLSCREEN; |
|
|
34 | |
|
|
35 | our $FONTSIZE; |
27 | |
36 | |
28 | our $SDL_TIMER; |
37 | our $SDL_TIMER; |
29 | our $SDL_APP; |
38 | our $SDL_APP; |
30 | our $SDL_EV = new SDL::Event; |
39 | our $SDL_EV = new SDL::Event; |
31 | our %SDL_CB; |
40 | our %SDL_CB; |
32 | |
41 | |
33 | our @GL_INIT; # hooks called on every gl init |
42 | our @GL_INIT; # hooks called on every gl init |
34 | |
43 | |
|
|
44 | our $ALT_ENTER_MESSAGE; |
|
|
45 | our $STATUS_LINE; |
|
|
46 | |
|
|
47 | our $TOPLEVEL; |
|
|
48 | |
|
|
49 | our $tw; # Test widget #d# |
|
|
50 | |
|
|
51 | my $last_refresh; |
|
|
52 | my %ANIMATE; |
|
|
53 | my $refresh_handler; |
|
|
54 | |
35 | sub init_screen { |
55 | sub init_screen { |
36 | $SDL_APP = new SDL::App |
56 | $SDL_APP = new SDL::App |
37 | -flags => SDL_ANYFORMAT | SDL_HWSURFACE, |
57 | -flags => SDL_ANYFORMAT | SDL_HWSURFACE, |
38 | -title => "Crossfire+ Client", |
58 | -title => "Crossfire+ Client", |
39 | -width => $CFG->{width}, |
59 | -width => $WIDTH, |
40 | -height => $CFG->{height}, |
60 | -height => $HEIGHT, |
41 | -opengl => 1, |
61 | -opengl => 1, |
42 | -red_size => 8, |
62 | -red_size => 5, |
43 | -green_size => 8, |
63 | -green_size => 5, |
44 | -blue_size => 8, |
64 | -blue_size => 5, |
|
|
65 | -alpha_size => 0, |
45 | -double_buffer => 1, |
66 | -double_buffer => 1, |
46 | -fullscreen => $CFG->{fullscreen}, |
67 | -fullscreen => $FULLSCREEN, |
47 | -resizeable => 0; |
68 | -resizeable => 0; |
48 | |
69 | |
|
|
70 | $last_refresh = SDL::GetTicks; |
|
|
71 | |
|
|
72 | %GL_EXT = map +($_ => 1), split /\s+/, Crossfire::Client::gl_extensions; |
|
|
73 | |
|
|
74 | $GL_EXT{GL_ARB_texture_non_power_of_two} |
|
|
75 | or warn "WARNING: non-power-of-two opengl extension required"; |
|
|
76 | |
|
|
77 | $FONTSIZE = int $HEIGHT / 50; |
|
|
78 | |
|
|
79 | $STATUS_LINE = new Crossfire::Client::Widget::Label |
|
|
80 | 0, $HEIGHT * 59 / 60 - $FONTSIZE, 1, $FONTSIZE, |
|
|
81 | ""; |
|
|
82 | $TOPLEVEL->add ($STATUS_LINE); |
|
|
83 | |
|
|
84 | $ALT_ENTER_MESSAGE = new Crossfire::Client::Widget::Label |
|
|
85 | 0, $HEIGHT * 59 / 60, 1, $HEIGHT / 60, |
|
|
86 | "Use <b>Alt-Enter</b> to toggle fullscreen mode"; |
|
|
87 | $TOPLEVEL->add ($ALT_ENTER_MESSAGE); |
|
|
88 | |
|
|
89 | # Test code #d# |
|
|
90 | unless ($tw) { # haha... |
|
|
91 | $tw = new Crossfire::Client::Widget::Animator; |
|
|
92 | my $lbl1 = new Crossfire::Client::Widget::Label |
|
|
93 | 0, 0, 10, $FONTSIZE, "<i>This</i> is a\n<u>TEST</u>!\nOf a themed\nFrame!"; |
|
|
94 | my $lbl2 = new Crossfire::Client::Widget::Label |
|
|
95 | 0, 0, 10, $FONTSIZE, "LBL2"; |
|
|
96 | |
|
|
97 | my $vb = new Crossfire::Client::Widget::VBox; |
|
|
98 | my $f = new Crossfire::Client::Widget::FancyFrame; |
|
|
99 | my $f2 = new Crossfire::Client::Widget::FancyFrame; |
|
|
100 | $f->add ($lbl1); |
|
|
101 | $f2->add ($lbl2); |
|
|
102 | $vb->add ($f); |
|
|
103 | $vb->add ($f2, 1); |
|
|
104 | |
|
|
105 | $tw->add ($vb); |
|
|
106 | $tw->w (400); |
|
|
107 | $tw->h (300); |
|
|
108 | $tw->move ($WIDTH - 200, 0); |
|
|
109 | $tw->moveto (0, 0); |
|
|
110 | $TOPLEVEL->add ($tw); |
|
|
111 | |
|
|
112 | # $f->move ($WIDTH - 200, 0); |
|
|
113 | # $TOPLEVEL->add ($f); |
|
|
114 | } |
|
|
115 | |
49 | glClearColor 0, 0, 0, 0; |
116 | glClearColor 0, 0, 0, 0; |
50 | |
117 | |
51 | glEnable GL_TEXTURE_2D; |
118 | glEnable GL_TEXTURE_2D; |
52 | glTexEnv GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_MODULATE; |
119 | glEnable GL_COLOR_MATERIAL; |
53 | glShadeModel GL_FLAT; |
120 | glShadeModel GL_FLAT; |
54 | glDisable GL_DEPTH_TEST; |
121 | glDisable GL_DEPTH_TEST; |
55 | glBlendFunc GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA; |
122 | glBlendFunc GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA; |
56 | glEnable GL_BLEND; |
123 | |
|
|
124 | $_->() for @GL_INIT; |
|
|
125 | } |
|
|
126 | |
|
|
127 | sub start_game { |
|
|
128 | $SDL_TIMER = add Glib::Timeout 1000/50, sub { |
|
|
129 | ($SDL_CB{$SDL_EV->type} || sub { warn "unhandled event ", $SDL_EV->type })->() |
|
|
130 | while $SDL_EV->poll; |
|
|
131 | |
|
|
132 | 1 |
|
|
133 | }; |
|
|
134 | |
|
|
135 | $WIDTH = $CFG->{width}; |
|
|
136 | $HEIGHT = $CFG->{height}; |
|
|
137 | $FULLSCREEN = 0; |
|
|
138 | |
|
|
139 | init_screen; |
|
|
140 | |
|
|
141 | my $mapsize = List::Util::min 64, List::Util::max 11, int $HEIGHT * $CFG->{mapsize} * 0.01 / 32; |
|
|
142 | |
|
|
143 | $CONN = new conn |
|
|
144 | host => $CFG->{host}, |
|
|
145 | port => $CFG->{port}, |
|
|
146 | user => $CFG->{user}, |
|
|
147 | pass => $CFG->{password}, |
|
|
148 | mapw => $mapsize, |
|
|
149 | maph => $mapsize, |
|
|
150 | ; |
|
|
151 | |
|
|
152 | Crossfire::Client::lowdelay fileno $CONN->{fh}; |
|
|
153 | } |
|
|
154 | |
|
|
155 | sub stop_game { |
|
|
156 | remove Glib::Source $SDL_TIMER; |
|
|
157 | remove Glib::Source $refresh_handler if $refresh_handler; |
|
|
158 | undef $refresh_handler; |
|
|
159 | |
|
|
160 | undef $SDL_APP; |
|
|
161 | undef $CONN; |
|
|
162 | SDL::Quit; |
|
|
163 | } |
|
|
164 | |
|
|
165 | |
|
|
166 | sub force_refresh { |
|
|
167 | glViewport 0, 0, $WIDTH, $HEIGHT; |
57 | |
168 | |
58 | glMatrixMode GL_PROJECTION; |
169 | glMatrixMode GL_PROJECTION; |
59 | glLoadIdentity; |
170 | glLoadIdentity; |
60 | glOrtho 0, $CFG->{width} / 32, $CFG->{height} / 32, 0, -100 , 100; |
171 | glOrtho 0, $WIDTH, $HEIGHT, 0, -6000 , 6000; |
61 | |
|
|
62 | glMatrixMode GL_MODELVIEW; |
172 | glMatrixMode GL_MODELVIEW; |
63 | |
173 | |
64 | $_->() for @GL_INIT; |
174 | glClear GL_COLOR_BUFFER_BIT; |
65 | } |
|
|
66 | |
175 | |
67 | sub start_game { |
176 | $TOPLEVEL->draw; |
68 | $SDL_TIMER = add Glib::Timeout 1000/20, sub { |
177 | |
69 | while ($SDL_EV->poll) { |
178 | SDL::GLSwapBuffers; |
70 | ($SDL_CB{$SDL_EV->type} || sub { warn "unhandled event ", $SDL_EV->type })->(); |
179 | } |
|
|
180 | |
|
|
181 | sub refresh { |
|
|
182 | $refresh_handler ||= add Glib::Idle sub { |
|
|
183 | return unless $SDL_APP; |
|
|
184 | |
|
|
185 | my $next_refresh = SDL::GetTicks; |
|
|
186 | my $interval = ($next_refresh - $last_refresh) * 0.001; |
|
|
187 | $last_refresh = $next_refresh; |
|
|
188 | |
|
|
189 | force_refresh; |
|
|
190 | $_->animate ($interval) for grep $_, values %ANIMATE; |
|
|
191 | |
|
|
192 | if (%ANIMATE) { |
|
|
193 | 1 |
|
|
194 | } else { |
|
|
195 | undef $refresh_handler; |
|
|
196 | 0 |
71 | } |
197 | } |
72 | |
|
|
73 | 1 |
|
|
74 | }; |
198 | }; |
75 | |
|
|
76 | init_screen; |
|
|
77 | |
|
|
78 | $CONN = new conn |
|
|
79 | host => $CFG->{host}, |
|
|
80 | port => $CFG->{port}; |
|
|
81 | } |
199 | } |
82 | |
200 | |
83 | sub stop_game { |
201 | sub animation_start { |
84 | remove Glib::Source $SDL_TIMER; |
202 | my ($widget) = @_; |
|
|
203 | $ANIMATE{$widget} = $widget; |
|
|
204 | Scalar::Util::weaken $ANIMATE{$widget}; |
85 | |
205 | |
86 | undef $SDL_APP; |
206 | refresh; |
87 | SDL::Quit; |
|
|
88 | } |
207 | } |
89 | |
208 | |
90 | sub refresh { |
209 | sub animation_stop { |
91 | glClear GL_COLOR_BUFFER_BIT; |
210 | my ($widget) = @_; |
92 | |
211 | delete $ANIMATE{$widget}; |
93 | for (values %Client::Widget::ACTIVE_WIDGETS) { |
|
|
94 | $_->draw |
|
|
95 | } |
|
|
96 | |
|
|
97 | SDL::GLSwapBuffers; |
|
|
98 | } |
212 | } |
99 | |
213 | |
100 | %SDL_CB = ( |
214 | %SDL_CB = ( |
101 | SDL_QUIT() => sub { |
215 | SDL_QUIT() => sub { |
102 | warn "sdl quit\n";#d# |
|
|
103 | main_quit Gtk2; |
216 | main_quit Gtk2; |
104 | }, |
217 | }, |
105 | SDL_VIDEORESIZE() => sub { |
218 | SDL_VIDEORESIZE() => sub { |
106 | }, |
219 | }, |
107 | SDL_VIDEOEXPOSE() => sub { |
220 | SDL_VIDEOEXPOSE() => sub { |
108 | refresh; |
221 | refresh; |
109 | }, |
222 | }, |
110 | SDL_KEYDOWN() => sub { |
223 | SDL_KEYDOWN() => sub { |
111 | if ($SDL_EV->key_mod & KMOD_ALT && $SDL_EV->key_sym == SDLK_RETURN) { |
224 | if ($SDL_EV->key_mod & KMOD_ALT && $SDL_EV->key_sym == SDLK_RETURN) { |
112 | # alt-enter |
225 | # alt-enter |
113 | $CFG->{fullscreen} = !$CFG->{fullscreen}; |
226 | $FULLSCREEN = !$FULLSCREEN; |
114 | init_screen; |
227 | init_screen; |
115 | } else { |
228 | } else { |
116 | Client::Widget::feed_sdl_key_down_event ($SDL_EV); |
229 | Crossfire::Client::Widget::feed_sdl_key_down_event ($SDL_EV); |
117 | } |
230 | } |
118 | }, |
231 | }, |
119 | SDL_KEYUP() => sub { |
232 | SDL_KEYUP() => sub { |
120 | Client::Widget::feed_sdl_key_up_event ($SDL_EV); |
233 | Crossfire::Client::Widget::feed_sdl_key_up_event ($SDL_EV); |
121 | }, |
234 | }, |
122 | SDL_MOUSEMOTION() => sub { |
235 | SDL_MOUSEMOTION() => sub { |
123 | warn "sdl motion\n";#d# |
236 | warn "sdl motion\n";#d# |
124 | }, |
237 | }, |
125 | SDL_MOUSEBUTTONDOWN() => sub { |
238 | SDL_MOUSEBUTTONDOWN() => sub { |
126 | Client::Widget::feed_sdl_button_down_event ($SDL_EV); |
239 | Crossfire::Client::Widget::feed_sdl_button_down_event ($SDL_EV); |
127 | }, |
240 | }, |
128 | SDL_MOUSEBUTTONUP() => sub { |
241 | SDL_MOUSEBUTTONUP() => sub { |
129 | Client::Widget::feed_sdl_button_up_event ($SDL_EV); |
242 | Crossfire::Client::Widget::feed_sdl_button_up_event ($SDL_EV); |
130 | }, |
243 | }, |
131 | SDL_ACTIVEEVENT() => sub { |
244 | SDL_ACTIVEEVENT() => sub { |
132 | warn "active\n";#d# |
245 | warn "active\n";#d# |
133 | }, |
246 | }, |
134 | ); |
247 | ); |
… | |
… | |
142 | } |
255 | } |
143 | |
256 | |
144 | sub conn::map_scroll { |
257 | sub conn::map_scroll { |
145 | my ($self, $dx, $dy) = @_; |
258 | my ($self, $dx, $dy) = @_; |
146 | |
259 | |
147 | refresh; |
260 | # refresh; |
148 | } |
261 | } |
149 | |
262 | |
150 | sub conn::map_clear { |
263 | sub conn::map_clear { |
151 | my ($self) = @_; |
264 | my ($self) = @_; |
152 | |
265 | |
153 | refresh; |
266 | # refresh; |
154 | } |
267 | } |
155 | |
268 | |
156 | sub conn::face_find { |
269 | sub conn::face_find { |
157 | my ($self, $face) = @_; |
270 | my ($self, $face) = @_; |
158 | |
271 | |
… | |
… | |
165 | $FACECACHE->{"$face->{chksum},$face->{name}"} = $face->{image}; |
278 | $FACECACHE->{"$face->{chksum},$face->{name}"} = $face->{image}; |
166 | |
279 | |
167 | $face->{texture} = new_from_image Crossfire::Client::Texture delete $face->{image}; |
280 | $face->{texture} = new_from_image Crossfire::Client::Texture delete $face->{image}; |
168 | } |
281 | } |
169 | |
282 | |
|
|
283 | sub conn::query { |
|
|
284 | my ($self, $flags, $prompt) = @_; |
|
|
285 | |
|
|
286 | warn "<<<<QUERY:$flags:$prompt>>>\n";#d# |
|
|
287 | } |
|
|
288 | |
|
|
289 | sub gtk_add_cfg_field { |
|
|
290 | my ($tbl, $cfg, $klbl, $key, $value) = @_; |
|
|
291 | my $i = $cfg->{_i}++; |
|
|
292 | $tbl->attach_defaults (my $lbl = Gtk2::Label->new ($klbl), 0, 1, $i, $i + 1); |
|
|
293 | $tbl->attach_defaults (my $ent = Gtk2::Entry->new, 1, 2, $i, $i + 1); |
|
|
294 | if ($key eq 'password') { |
|
|
295 | $ent->set_invisible_char ("*"); |
|
|
296 | $ent->set (visibility => 0) |
|
|
297 | } |
|
|
298 | $ent->set_text ($value); |
|
|
299 | $ent->signal_connect (changed => sub { |
|
|
300 | my ($ent) = @_; |
|
|
301 | $cfg->{$key} = $ent->get_text; |
|
|
302 | # TODO: Mapsize should be a slider in the game gui |
|
|
303 | if ($key eq 'mapsize' and $cfg->{$key} > 100) { |
|
|
304 | $cfg->{$key} = 100; |
|
|
305 | } elsif ($key eq 'mapsize' and $cfg->{$key} < 50) { |
|
|
306 | $cfg->{$key} = 50; |
|
|
307 | } |
|
|
308 | }); |
|
|
309 | } |
|
|
310 | |
|
|
311 | sub run_config_dialog { |
|
|
312 | my (%events) = @_; |
|
|
313 | |
|
|
314 | my $w = Gtk2::Window->new; |
|
|
315 | |
|
|
316 | my @cfg = ( |
|
|
317 | [qw/Host host/], |
|
|
318 | [qw/Port port/], |
|
|
319 | [qw/Mapsize% mapsize/], |
|
|
320 | [qw/Username user/], |
|
|
321 | [qw/Password password/], |
|
|
322 | ); |
|
|
323 | |
|
|
324 | my $cfg = {}; |
|
|
325 | |
|
|
326 | my $a = SDL::ListModes (0, SDL_FULLSCREEN|SDL_HWSURFACE); |
|
|
327 | my @modes = map { [SDL::RectW ($_), SDL::RectH ($_)] } @$a; |
|
|
328 | |
|
|
329 | $w->add (my $vb = Gtk2::VBox->new); |
|
|
330 | $vb->pack_start (my $t = Gtk2::Table->new (2, scalar @cfg), 0, 0, 0); |
|
|
331 | my $selmode = $::CFG->{width} . 'x' . $::CFG->{height}; |
|
|
332 | $t->attach_defaults (Gtk2::Label->new ("Modes"), 0, 1, 0, 1); |
|
|
333 | $t->attach_defaults (my $cb = Gtk2::ComboBox->new_text, 1, 2, 0, 1); |
|
|
334 | my $i = 0; |
|
|
335 | my $act = 0; |
|
|
336 | for (map { "$_->[0]x$_->[1]" } reverse @modes) { |
|
|
337 | if ($_ eq $selmode) { $act = $i } |
|
|
338 | $cb->append_text ($_); |
|
|
339 | $i++; |
|
|
340 | } |
|
|
341 | $cb->set_active ($act); |
|
|
342 | $cb->signal_connect (changed => sub { |
|
|
343 | my ($cb) = @_; |
|
|
344 | my $txt = $cb->get_active_text; |
|
|
345 | if ($txt =~ m/(\d+)x(\d+)/) { |
|
|
346 | $::CFG->{width} = $1; |
|
|
347 | $::CFG->{height} = $2; |
|
|
348 | } |
|
|
349 | }); |
|
|
350 | |
|
|
351 | $cfg->{_i} = 1; |
|
|
352 | for (@cfg) { |
|
|
353 | gtk_add_cfg_field ($t, $cfg, $_->[0], $_->[1], $::CFG->{$_->[1]}); |
|
|
354 | } |
|
|
355 | |
|
|
356 | $vb->pack_start (my $hb = Gtk2::HBox->new, 0, 0, 0); |
|
|
357 | $hb->pack_start (my $cb = Gtk2::Button->new ("save"), 1, 1, 5); |
|
|
358 | $cb->signal_connect (clicked => sub { |
|
|
359 | for (keys %$cfg) { |
|
|
360 | $::CFG->{$_} = $cfg->{$_} |
|
|
361 | if $_ ne '_i'; |
|
|
362 | } |
|
|
363 | Crossfire::Client::write_cfg "$Crossfire::VARDIR/pclientrc"; |
|
|
364 | }); |
|
|
365 | $hb->pack_start (my $cb = Gtk2::Button->new ("login"), 1, 1, 5); |
|
|
366 | $cb->signal_connect (clicked => sub { |
|
|
367 | for (keys %$cfg) { |
|
|
368 | $::CFG->{$_} = $cfg->{$_} |
|
|
369 | if $_ ne '_i'; |
|
|
370 | } |
|
|
371 | my $cb = $events{login} || sub {}; |
|
|
372 | $cb->($::CFG->{user}, $::CFG->{password}); |
|
|
373 | }); |
|
|
374 | $hb->pack_start (my $cb = Gtk2::Button->new ("logout"), 1, 1, 5); |
|
|
375 | $cb->signal_connect (clicked => sub { |
|
|
376 | my $cb = $events{logout} || sub {}; |
|
|
377 | $cb->(); |
|
|
378 | }); |
|
|
379 | $hb->pack_start (my $cb = Gtk2::Button->new ("quit"), 1, 1, 5); |
|
|
380 | $cb->signal_connect (clicked => sub { $w->destroy }); |
|
|
381 | |
|
|
382 | $w->show_all; |
|
|
383 | |
|
|
384 | $w->signal_connect (destroy => sub { Gtk2->main_quit }); |
|
|
385 | } |
|
|
386 | |
|
|
387 | |
170 | ############################################################################# |
388 | ############################################################################# |
171 | |
389 | |
|
|
390 | SDL::Init SDL_INIT_EVERYTHING; |
|
|
391 | |
|
|
392 | $TOPLEVEL = Crossfire::Client::Widget::Toplevel->new; |
|
|
393 | |
172 | my $mapwidget = Client::MapWidget->new; |
394 | my $mapwidget = Crossfire::Client::Widget::MapWidget->new; |
173 | |
395 | |
174 | $mapwidget->activate; |
396 | $TOPLEVEL->add ($mapwidget); |
175 | $mapwidget->focus_in; |
397 | $mapwidget->focus_in; |
176 | |
398 | |
177 | Client::Util::read_cfg "$Crossfire::VARDIR/pclientrc"; |
399 | Crossfire::Client::read_cfg "$Crossfire::VARDIR/pclientrc"; |
178 | |
|
|
179 | $FACECACHE = eval { Crossfire::load_ref "$Crossfire::VARDIR/pclient.faces" } || {}; |
|
|
180 | |
400 | |
181 | $CFG ||= { |
401 | $CFG ||= { |
182 | width => 640, |
402 | width => 640, |
183 | height => 480, |
403 | height => 480, |
|
|
404 | mapsize => 100, |
184 | fullscreen => 0, |
405 | fullscreen => 0, |
185 | host => "crossfire.schmorp.de", |
406 | host => "crossfire.schmorp.de", |
186 | port => 13327, |
407 | port => 13327, |
187 | }; |
408 | }; |
188 | |
409 | |
189 | Client::Util::run_config_dialog |
410 | Crossfire::Client::set_font Crossfire::Client::find_rcfile "uifont.ttf"; |
|
|
411 | |
|
|
412 | $FACECACHE = eval { Crossfire::load_ref "$Crossfire::VARDIR/pclient.faces" } || {}; |
|
|
413 | |
|
|
414 | run_config_dialog |
190 | login => sub { start_game }, |
415 | login => sub { start_game }, |
191 | logout => sub { stop_game }; |
416 | logout => sub { stop_game }; |
192 | |
417 | |
193 | main Gtk2; |
418 | main Gtk2; |
194 | |
419 | |