ViewVC Help
View File | Revision Log | Show Annotations | Download File
/cvs/deliantra/gde/GCE/Util.pm
Revision: 1.1
Committed: Mon Feb 20 18:21:04 2006 UTC (18 years, 3 months ago) by elmex
Branch: MAIN
Log Message:
implemented layer insertion

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     our @EXPORT = qw(insert_arch_stack_layer replace_arch_stack_layer);
19    
20     sub classify_arch_layer {
21     my ($arch) = @_;
22    
23     if ($arch->{invisible}) { # just a heuristic for 'special' tiles (er. pedestals)
24    
25     return 'below';
26    
27     } elsif ($arch->{monster}) {
28    
29     return 'top';
30    
31     } else { # $arch->{is_floor} and all other arches are 'between' monsters and floor
32    
33     return 'between';
34     }
35     }
36    
37     sub insert_arch_stack_layer {
38     my ($stack, $arch) = @_;
39    
40     unless (@$stack) {
41     return [ $arch ];
42     }
43    
44     my @outstack;
45    
46     my $l = classify_arch_layer ($Crossfire::ARCH{$arch->{_name}});
47    
48     if ($l eq 'between') {
49    
50     # loop until we reached the first 'between' arch above 'below' arches and the floor
51     while (my $a = shift @$stack) {
52    
53     unless ($Crossfire::ARCH{$a->{_name}}->{is_floor}
54     or classify_arch_layer ($Crossfire::ARCH{$a->{_name}}) eq 'below') {
55    
56     unshift @$stack, $a;
57     last;
58     }
59    
60     push @outstack, $a;
61     }
62    
63     # ignore duplicates
64     # FIXME: Broken if non-floor are drawn (too tired to fix)
65     return [ @outstack, @$stack ]
66     if @outstack and $outstack[-1]->{_name} eq $arch->{_name};
67    
68     push @outstack, ($arch, @$stack);
69    
70     } elsif ($l eq 'top') {
71    
72     # ignore duplicates
73     return [ @$stack ]
74     if $stack->[-1]->{_name} eq $arch->{_name};
75    
76     @outstack = (@$stack, $arch);
77    
78     } else {
79    
80     # ignore duplicates
81     return [ @$stack ]
82     if $stack->[0]->{_name} eq $arch->{_name};
83    
84     @outstack = ($arch, @$stack);
85     }
86    
87     return \@outstack;
88     }
89    
90     sub replace_arch_stack_layer {
91     my ($stack, $arch) = @_;
92    
93     my @outstack;
94    
95     my $l = classify_arch_layer ($Crossfire::ARCH{$arch->{_name}});
96    
97     if ($l eq 'between') {
98    
99     while (shift @$stack) {
100     last unless $Crossfire::ARCH{$_->{_name}}->{is_floor};
101     push @outstack, $_;
102     }
103    
104     if (@outstack and $Crossfire::ARCH{$outstack[-1]->{_name}}->{is_floor}) {
105     pop @outstack;
106     }
107    
108     push @outstack, ($arch, @$stack);
109    
110     } elsif ($l eq 'top') {
111    
112     @outstack = (@$stack, $arch);
113    
114     } else {
115    
116     @outstack = ($arch, @$stack);
117     }
118    
119     return \@outstack;
120     }
121    
122     =head1 AUTHOR
123    
124     Marc Lehmann <schmorp@schmorp.de>
125     http://home.schmorp.de/
126    
127     Robin Redeker <elmex@ta-sa.org>
128     http://www.ta-sa.org/
129    
130     =cut
131     1;