ViewVC Help
View File | Revision Log | Show Annotations | Download File
/cvs/deliantra/gde/GCE/Util.pm
Revision: 1.19
Committed: Mon Aug 14 03:55:50 2006 UTC (17 years, 9 months ago) by elmex
Branch: MAIN
Changes since 1.18: +5 -1 lines
Log Message:
Shop FLoor is now recognized as floor too :)

File Contents

# Content
1 package GCE::Util;
2 =head1 NAME
3
4 GCE::Util - some utility functions
5
6 =over 4
7
8 =cut
9
10 use base 'Exporter';
11
12 use Crossfire;
13
14 use Carp ();
15 use Storable;
16 use List::Util qw(min max);
17
18 use Crossfire;
19 use Crossfire::MapWidget;
20 use File::Spec::Functions;
21 use Cwd 'abs_path';
22
23 our @EXPORT = qw(insert_arch_stack_layer replace_arch_stack_layer new_arch_pb
24 fill_pb_from_arch arch_is_floor stack_find_floor stack_find_wall
25 stack_find arch_is_wall arch_is_monster add_table_widget quick_msg
26 def arch_is_exit map2abs exit_paths pseudohtml2txt arch_is_connector);
27
28 sub pseudohtml2txt {
29 my ($html) = @_;
30
31 $html =~ s/<br\s*?\/?>/\n/gsi;
32 $html =~ s/<b>(.*?)<\/b>/_\1_/gsi;
33 $html =~ s/<li>/\n* /gi;
34 $html =~ s/<\/?\s*li>//gi;
35 $html =~ s/<\/?\s*ul>//gi;
36 $html =~ s/&gt;/>/g;
37 $html =~ s/&lt;/</g;
38 $html
39 }
40
41 sub exit_paths {
42 my ($mappath, $map1path, $map2path) = @_;
43 $mappath = abs_path $mappath;
44 $map1path = abs_path $map1path;
45 $map2path = abs_path $map2path;
46
47 if ( (substr $map1path, 0, length $mappath) eq $mappath
48 and (substr $map2path, 0, length $mappath) eq $mappath) {
49 substr $map1path, 0, length $mappath, '';
50 substr $map2path, 0, length $mappath, '';
51
52 my ($v1, $d1, $f1) = File::Spec->splitpath ($map1path);
53 my ($v2, $d2, $f2) = File::Spec->splitpath ($map2path);
54
55 my @di1 = File::Spec->splitdir ($d1);
56 my @di2 = File::Spec->splitdir ($d2);
57
58 if ((defined $di1[1]) and (defined $di2[1]) and $di1[1] eq $di2[1]) {
59 my $m1 = File::Spec->abs2rel ($map1path, File::Spec->catdir (@di2));
60 my $m2 = File::Spec->abs2rel ($map2path, File::Spec->catdir (@di1));
61 return ($m1, $m2);
62 } else {
63 return ($map1path, $map2path);
64 }
65 } else {
66 return ('', '');
67 }
68 }
69
70 sub map2abs {
71 my ($dest, $mape) = @_;
72
73 $mappath = abs_path $mappath;
74 my $dir;
75 if (File::Spec->file_name_is_absolute($dest)) {
76 $dir = catdir ($::CFG->{MAPDIR}, $dest);
77 } else {
78 my ($v, $p, $f) = File::Spec->splitpath ($mape->{path});
79 $dir = File::Spec->rel2abs ($dest, File::Spec->catpath ($v, $p));
80 }
81 return $dir;
82 }
83
84 sub def($$) {
85 return defined ($_[0]) ? $_[0] : $_[1];
86 }
87
88 sub quick_msg {
89 my $wid = shift;
90 my $msg;
91 my $win = $::MAINWIN;
92 if (ref $wid) {
93 $win = $wid;
94 $msg = shift;
95 } else {
96 $msg = $wid;
97 }
98 my $dia = Gtk2::Dialog->new ('Message', $win, 'destroy-with-parent', 'gtk-ok' => 'none');
99
100 my $lbl = Gtk2::Label->new ($msg);
101 $dia->vbox->add ($lbl);
102 $dia->signal_connect (response => sub { $_[0]->destroy });
103
104 unless (defined $_[0]) {
105 Glib::Timeout->add (1000, sub { $dia->destroy; 0 });
106 }
107
108 $dia->show_all;
109 }
110
111 sub new_arch_pb {
112 # this is awful, is this really the best way?
113 my $pb = new Gtk2::Gdk::Pixbuf 'rgb', 1, 8, TILESIZE, TILESIZE;
114 return $pb;
115 }
116
117 sub fill_pb_from_arch {
118 my ($pb, $a) = @_;
119
120 my $o = $Crossfire::ARCH{$a->{_name}};
121 my $face = $Crossfire::FACE{$a->{face} || $o->{face} || "blank.111"}
122 or warn "no gfx found for arch '$a->{_name}' at ($x|$y)\n";
123
124 $face or return;
125
126 $pb->fill (0x00000000);
127 $TILE->composite ($pb,
128 0, 0,
129 TILESIZE, TILESIZE,
130 - ($face->{idx} % 64) * TILESIZE, - TILESIZE * int $face->{idx} / 64,
131 1, 1, 'nearest', 255
132 );
133 }
134
135 sub classify_arch_layer {
136 my ($arch) = @_;
137
138 if ($arch->{invisible}) { # just a heuristic for 'special' tiles (er. pedestals)
139
140 return 'below';
141
142 } elsif ($arch->{monster}) {
143
144 return 'top';
145
146 } else { # $arch->{is_floor} and all other arches are 'between' monsters and floor
147
148 return 'between';
149 }
150 }
151
152 sub arch_is_exit {
153 my ($a) = @_;
154 my $type = $Crossfire::ARCH{$a->{_name}}->{type};
155 return $type eq '66' || $type eq '41';
156 }
157
158 sub arch_is_floor {
159 my ($a) = @_;
160 my $ar = Crossfire::arch_attr $a;
161 warn "NAME: $ar->{name}\n";
162 return (
163 (substr $ar->{name}, 0, 5) eq 'Floor'
164 or (substr $ar->{name}, 0, 10) eq 'Shop Floor'
165 )
166 }
167
168 sub arch_is_connector {
169 my ($a) = @_;
170 my $ar = Crossfire::arch_attr $a;
171 my $has_connect_field = 0;
172
173 TOP: for (@{$ar->{section}}) {
174 my $name = shift @$_;
175 my @r = @$_;
176 if ($name eq 'general') {
177 for (@r) {
178 my ($k, $s) = ($_->[0], $_->[1]);
179 if ($k eq 'connected' && $s->{name} eq 'connection') {
180 $has_connect_field = 1;
181 last TOP;
182 }
183 }
184 last TOP;
185 }
186 }
187
188 return $has_connect_field;
189 }
190
191 sub arch_is_wall {
192 my ($a) = @_;
193 my $ar = Crossfire::arch_attr $a;
194 return $ar->{name} eq 'Wall';
195 #return $Crossfire::ARCH{$a->{_name}}->{no_pass};
196 }
197
198 sub arch_is_monster {
199 my ($a) = @_;
200 my $arch = $Crossfire::ARCH{$a->{_name}};
201 return $arch->{alive} and ($arch->{monster} or $arch->{generator});
202 }
203
204 sub stack_find {
205 my ($stack, $dir, $pred) = @_;
206
207
208 if ($dir eq 'from_top') {
209 my $i = scalar (@$stack) - 1;
210 if ($i < 0) { $i = 0 }
211
212 for (reverse @$stack) {
213 $pred->($_)
214 and return $i;
215
216 $i--;
217 }
218
219 } else {
220 my $i = 0;
221
222 for (@$stack) {
223 $pred->($_)
224 and return $i;
225
226 $i++;
227 }
228 }
229
230 return 0;
231
232 }
233
234 sub stack_find_floor {
235 my ($stack, $dir) = @_;
236 return stack_find ($stack, $dir, \&arch_is_floor);
237 }
238
239 sub stack_find_wall {
240 my ($stack, $dir) = @_;
241 return stack_find ($stack, $dir, \&arch_is_wall);
242 }
243
244 sub insert_arch_stack_layer {
245 my ($stack, $arch) = @_;
246
247 unless (@$stack) {
248 return [ $arch ];
249 }
250
251 my @outstack;
252
253 my $l = classify_arch_layer ($Crossfire::ARCH{$arch->{_name}});
254
255 if ($l eq 'between') {
256
257 # loop until we reached the first 'between' arch above 'below' arches and the floor
258 while (my $a = shift @$stack) {
259
260 unless ($Crossfire::ARCH{$a->{_name}}->{is_floor}
261 or classify_arch_layer ($Crossfire::ARCH{$a->{_name}}) eq 'below') {
262
263 unshift @$stack, $a;
264 last;
265 }
266
267 push @outstack, $a;
268 }
269
270 # ignore duplicates
271 # FIXME: Broken if non-floor are drawn (too tired to fix)
272 return [ @outstack, @$stack ]
273 if @outstack and $outstack[-1]->{_name} eq $arch->{_name};
274
275 push @outstack, ($arch, @$stack);
276
277 } elsif ($l eq 'top') {
278
279 # ignore duplicates
280 return [ @$stack ]
281 if $stack->[-1]->{_name} eq $arch->{_name};
282
283 @outstack = (@$stack, $arch);
284
285 } else {
286
287 # ignore duplicates
288 return [ @$stack ]
289 if $stack->[0]->{_name} eq $arch->{_name};
290
291 @outstack = ($arch, @$stack);
292 }
293
294 return \@outstack;
295 }
296
297 sub add_table_widget {
298 my ($table, $row, $data, $type, $cb) = @_;
299 my $edwid;
300
301 if ($type eq 'string') {
302 $table->attach_defaults (my $lbl = Gtk2::Label->new ($data->[0]), 0, 1, $row, $row + 1);
303 $edwid = Gtk2::Entry->new;
304 $edwid->set_text ($data->[1]);
305 $edwid->signal_connect (changed => sub {
306 $data->[1] = $_[0]->get_text;
307 $cb->($data->[1]) if $cb;
308 });
309 $table->attach_defaults ($edwid, 1, 2, $row, $row + 1);
310
311 } elsif ($type eq 'button') {
312 $table->attach_defaults (my $b = Gtk2::Button->new_with_label ($data), 0, 2, $row, $row + 1);
313 $b->signal_connect (clicked => ($cb || sub {}));
314
315 } elsif ($type eq 'label') {
316 $table->attach_defaults (my $lbl = Gtk2::Label->new ($data->[0]), 0, 1, $row, $row + 1);
317 $edwid = Gtk2::Label->new ($data->[1]);
318 $table->attach_defaults ($edwid, 1, 2, $row, $row + 1);
319
320 } else {
321 $edwid = Gtk2::Label->new ("FOO");
322 }
323 }
324
325 sub replace_arch_stack_layer {
326 my ($stack, $arch) = @_;
327
328 my @outstack;
329
330 my $l = classify_arch_layer ($Crossfire::ARCH{$arch->{_name}});
331
332 if ($l eq 'between') {
333
334 while (shift @$stack) {
335 last unless $Crossfire::ARCH{$_->{_name}}->{is_floor};
336 push @outstack, $_;
337 }
338
339 if (@outstack and $Crossfire::ARCH{$outstack[-1]->{_name}}->{is_floor}) {
340 pop @outstack;
341 }
342
343 push @outstack, ($arch, @$stack);
344
345 } elsif ($l eq 'top') {
346
347 @outstack = (@$stack, $arch);
348
349 } else {
350
351 @outstack = ($arch, @$stack);
352 }
353
354 return \@outstack;
355 }
356
357 =head1 AUTHOR
358
359 Marc Lehmann <schmorp@schmorp.de>
360 http://home.schmorp.de/
361
362 Robin Redeker <elmex@ta-sa.org>
363 http://www.ta-sa.org/
364
365 =cut
366 1;