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, 11 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

# User Rev Content
1 elmex 1.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 elmex 1.2 use Crossfire;
19     use Crossfire::MapWidget;
20 elmex 1.8 use File::Spec::Functions;
21 elmex 1.13 use Cwd 'abs_path';
22 elmex 1.2
23 elmex 1.18 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 elmex 1.14
28     sub pseudohtml2txt {
29     my ($html) = @_;
30    
31     $html =~ s/<br\s*?\/?>/\n/gsi;
32     $html =~ s/<b>(.*?)<\/b>/_\1_/gsi;
33 elmex 1.16 $html =~ s/<li>/\n* /gi;
34 elmex 1.15 $html =~ s/<\/?\s*li>//gi;
35     $html =~ s/<\/?\s*ul>//gi;
36     $html =~ s/&gt;/>/g;
37     $html =~ s/&lt;/</g;
38 elmex 1.14 $html
39     }
40 elmex 1.13
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 elmex 1.8
70     sub map2abs {
71     my ($dest, $mape) = @_;
72    
73 elmex 1.17 $mappath = abs_path $mappath;
74 elmex 1.8 my $dir;
75     if (File::Spec->file_name_is_absolute($dest)) {
76 elmex 1.11 $dir = catdir ($::CFG->{MAPDIR}, $dest);
77 elmex 1.8 } 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 elmex 1.5
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 elmex 1.2
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 elmex 1.6 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 elmex 1.2
126     $pb->fill (0x00000000);
127     $TILE->composite ($pb,
128     0, 0,
129     TILESIZE, TILESIZE,
130 elmex 1.6 - ($face->{idx} % 64) * TILESIZE, - TILESIZE * int $face->{idx} / 64,
131 elmex 1.2 1, 1, 'nearest', 255
132     );
133     }
134 elmex 1.1
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 elmex 1.7 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 elmex 1.2 sub arch_is_floor {
159     my ($a) = @_;
160 elmex 1.10 my $ar = Crossfire::arch_attr $a;
161 elmex 1.19 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 elmex 1.18 }
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 elmex 1.2 }
190    
191     sub arch_is_wall {
192     my ($a) = @_;
193 elmex 1.10 my $ar = Crossfire::arch_attr $a;
194     return $ar->{name} eq 'Wall';
195     #return $Crossfire::ARCH{$a->{_name}}->{no_pass};
196 elmex 1.2 }
197    
198 elmex 1.3 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 elmex 1.2 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 elmex 1.1 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 elmex 1.4 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 elmex 1.1 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;