]> git.p6c8.net - devedit.git/blob - modules/Template.pm
file_read() is now able to open a file in binary mode (maybe we will need it)
[devedit.git] / modules / Template.pm
1 package Template;
2
3 #
4 # Template (Version 1.4)
5 #
6 # Klasse zum Parsen von Templates
7 #
8 # Autor: Patrick Canterino <patrick@patshaping.de>
9 # Letzte Aenderung: 5.2.2005
10 #
11
12 use strict;
13
14 use Carp qw(croak);
15
16 # new()
17 #
18 # Konstruktor
19 #
20 # Parameter: -keine-
21 #
22 # Rueckgabe: Template-Objekt
23
24 sub new
25 {
26 my $class = shift;
27 my $self = {file => '', template => ''};
28 return bless($self,$class);
29 }
30
31 # get_template()
32 #
33 # Kompletten Vorlagentext zurueckgeben
34 #
35 # Parameter: -keine-
36 #
37 # Rueckgabe: Kompletter Vorlagentext (String)
38
39 sub get_template
40 {
41 return shift->{'template'};
42 }
43
44 # set_template()
45 #
46 # Kompletten Vorlagentext aendern
47 #
48 # Parameter: Vorlagentext
49 #
50 # Rueckgabe: -nichts- (Template-Objekt wird modifiziert)
51
52 sub set_template($)
53 {
54 # Geht nur so...
55
56 my ($self,$template) = @_;
57 $self->{'template'} = $template;
58 }
59
60 # add_text()
61 #
62 # Vorlagentext ans Template-Objekt anhaengen
63 #
64 # Parameter: Vorlagentext
65 #
66 # Rueckgabe: -nichts- (Template-Objekt wird modifiziert)
67
68 sub add_text($)
69 {
70 my ($self,$text) = @_;
71 $self->set_template($self->get_template.$text);
72 }
73
74 # read_file()
75 #
76 # Einlesen einer Vorlagendatei und {INCLUDE}-Anweisungen ggf. verarbeiten
77 # (Text wird an bereits vorhandenen Text angehaengt)
78 #
79 # Parameter: 1. Datei zum Einlesen
80 # 2. Status-Code (Boolean):
81 # true => {INCLUDE}-Anweisungen nicht verarbeiten
82 # false => {INCLUDE}-Anweisungen verarbeiten (Standard)
83 #
84 # Rueckgabe: -nichts- (Template-Objekt wird modifiziert)
85
86 sub read_file($;$)
87 {
88 my ($self,$file,$not_include) = @_;
89 local *FILE;
90
91 $self->{'file'} = $file;
92
93 open(FILE,'<'.$file) or croak "Open $file: $!";
94 read(FILE, my $content, -s $file);
95 close(FILE) or croak "Closing $file: $!";
96
97 $self->add_text($content);
98 $self->parse_includes unless($not_include);
99 }
100
101 # fillin()
102 #
103 # Variablen durch Text ersetzen
104 #
105 # Parameter: 1. Variable zum Ersetzen
106 # 2. Text, durch den die Variable ersetzt werden soll
107 #
108 # Rueckgabe: -nichts- (Template-Objekt wird modifiziert)
109
110 sub fillin($$)
111 {
112 my ($self,$var,$text) = @_;
113
114 $var = quotemeta($var);
115 $text = '' unless defined $text; # Um Fehler zu vermeiden
116
117 my $template = $self->get_template;
118 $template =~ s/\{$var\}/$text/g;
119
120 $self->set_template($template);
121 }
122
123 # fillin_array()
124 #
125 # Variable durch Array ersetzen
126 #
127 # Parameter: 1. Variable zum Ersetzen
128 # 2. Array-Referenz, durch die die Variable ersetzt werden soll
129 # 3. Zeichenkette, mit der das Array verbunden werden soll
130 # (Standard: '')
131 #
132 # Rueckgabe: -nichts- (Template-Objekt wird modifiziert)
133
134 sub fillin_array($$;$)
135 {
136 my ($self,$var,$array,$glue) = @_;
137 $glue = '' unless defined $glue;
138
139 $self->fillin($var,join($glue,@$array));
140 }
141
142 # to_file()
143 #
144 # Template in Datei schreiben
145 #
146 # Parameter: Datei-Handle
147 #
148 # Rueckgabe: Status-Code (Boolean)
149
150 sub to_file($)
151 {
152 my ($self,$handle) = @_;
153 return print $handle $self->get_template;
154 }
155
156 # parse_if_block()
157 #
158 # IF-Bloecke verarbeiten
159 #
160 # Parameter: 1. Name des IF-Blocks (das, was nach dem IF steht)
161 # 2. Status-Code (true => Inhalt anzeigen
162 # false => Inhalt nicht anzeigen
163 #
164 # Rueckgabe: -nichts- (Template-Objekt wird modifiziert)
165
166 sub parse_if_block($$)
167 {
168 my ($self,$name,$state) = @_;
169 my $template = $self->get_template;
170
171 my $count = 0;
172
173 while(index($template,'{IF '.$name.'}') >= 0)
174 {
175 # Das alles hier ist nicht wirklich elegant geloest...
176 # ... aber solange es funktioniert... ;-)
177
178 $count++;
179
180 my $start = index($template,'{IF '.$name.'}');
181 my $tpl_tmp = substr($template,$start);
182 my @splitted = split(/\{ENDIF\}/,$tpl_tmp);
183 push(@splitted,'') if(substr($template,-7) eq '{ENDIF}');
184
185 my $block = ''; # Kompletter bedingter Block
186 my $ifs = 0; # IF-Zaehler (wird fuer jedes IF erhoeht und fuer jedes ENDIF erniedrigt)
187
188 # {IF}
189
190 for(my $x=0;$x<@splitted;$x++)
191 {
192 croak 'Nesting error found while parsing IF block "'.$name.'" nr. '.$count.' in template file "'.$self->{'file'}.'"' if($x == $#splitted);
193
194 $ifs += substr_count($splitted[$x],'{IF '); # Zum Zaehler jedes Vorkommen von IF hinzuzaehlen
195 $ifs--; # Zaehler um 1 erniedrigen
196 $block .= $splitted[$x].'{ENDIF}'; # Daten zum Block hinzufuegen
197
198 if($ifs == 0)
199 {
200 # Zaehler wieder 0, also haben wir das Ende des IF-Blocks gefunden :-))
201
202 last;
203 }
204 }
205
206 my $if_block = substr($block,length($name)+5,-7); # Alles zwischen {IF} und {ENDIF}
207
208 # {ELSE}
209
210 my $else_block = ''; # Alles ab {ELSE}
211 $ifs = 0; # IF-Zaehler
212
213 @splitted = split(/\{ELSE\}/,$if_block);
214
215 for(my $x=0;$x<@splitted;$x++)
216 {
217 $ifs += substr_count($splitted[$x],'{IF '); # Zum Zaehler jedes Vorkommen von IF hinzuzaehlen
218 $ifs -= substr_count($splitted[$x],'{ENDIF}'); # Vom Zaehler jedes Vorkommen von ENDIF abziehen
219
220 if($ifs == 0)
221 {
222 # Zaehler 0, also haben wir das Ende des IF-Abschnitts gefunden
223
224 # Aus dem Rest den ELSE-Block zusammenbauen
225
226 for(my $y=$x+1;$y<@splitted;$y++)
227 {
228 $else_block .= '{ELSE}'.$splitted[$y];
229 }
230
231 if($else_block)
232 {
233 $if_block = substr($if_block,0,length($if_block)-length($else_block));
234 $else_block = (length($else_block) > 6) ? substr($else_block,6) : ''; # Ansonsten gibt es Fehler
235 }
236
237 last;
238 }
239 }
240
241 my $replacement = ($state) ? $if_block : $else_block;
242
243 my $qmblock = quotemeta($block);
244
245 $template =~ s/$qmblock/$replacement/;
246 }
247
248 $self->set_template($template);
249 }
250
251 # parse_condtag()
252 #
253 # Bedingungstags in einem Vorlagentext verarbeiten
254 #
255 # Parameter: 1. Tagname
256 # 2. Status-Code (true => Tag-Inhalt anzeigen
257 # false => Tag-Inhalt nicht anzeigen
258 #
259 # Rueckgabe: -nichts- (Template-Objekt wird modifiziert)
260
261 sub parse_condtag($$)
262 {
263 my ($self,$condtag,$state) = @_;
264
265 my $template = $self->get_template;
266
267 while(index($template,'<'.$condtag.'>') >= 0)
268 {
269 my $start = index($template,'<'.$condtag.'>'); # Beginn des Blocks
270 my $end = index($template,'</'.$condtag.'>')+length($condtag)+3; # Ende des Blocks
271
272 my $extract = substr($template,$start,$end-$start); # Kompletten Bedingungsblock extrahieren...
273
274 my $replacement = ($state) ? substr($extract,length($condtag)+2,0-length($condtag)-3) : '';
275
276 $extract = quotemeta($extract);
277
278 $template =~ s/$extract/$replacement/g; # Block durch neue Daten ersetzen
279 }
280 $self->set_template($template);
281 }
282
283 # parse_includes()
284 #
285 # {INCLUDE}-Anweisungen verarbeiten
286 #
287 # Parameter: -nichts-
288 #
289 # Rueckgabe: -nichts- (Template-Objekt wird modifiziert)
290
291 sub parse_includes
292 {
293 my $self = shift;
294 my $template = $self->get_template;
295
296 while($template =~ /(\{INCLUDE (\S+?)\})/)
297 {
298 my ($directive,$file) = ($1,$2);
299 my $qm_directive = quotemeta($directive);
300 my $replacement = '';
301
302 if(-f $file)
303 {
304 local *FILE;
305
306 open(FILE,'<'.$file) or croak "Open $file: $!";
307 read(FILE, $replacement, -s $file);
308 close(FILE) or croak "Closing $file: $!";
309 }
310
311 $template =~ s/$qm_directive/$replacement/g;
312 }
313
314 $self->set_template($template);
315 }
316
317 # ==================
318 # Private Funktion
319 # ==================
320
321 # substr_count()
322 #
323 # Zaehlt, wie oft ein String in einem String vorkommt
324 # (Emulation der PHP-Funktion substr_count())
325 #
326 # Parameter: 1. Zu durchsuchender String
327 # 2. Zu suchender String
328 #
329 # Rueckgabe: Anzahl der Vorkommnisse (Integer)
330
331 sub substr_count($$)
332 {
333 my ($haystack,$needle) = @_;
334 my $qmneedle = quotemeta($needle);
335
336 my $count = 0;
337
338 $count++ while($haystack =~ /$qmneedle/g);
339
340 return $count;
341 }
342
343 # ==================
344 # Alias-Funktionen
345 # ==================
346
347 sub addtext($)
348 {
349 shift->add_text(shift);
350 }
351
352 sub as_string
353 {
354 return shift->get_template;
355 }
356
357 sub condtag($$)
358 {
359 shift->parse_condtag(@_);
360 }
361
362 sub readin($)
363 {
364 shift->read_file(shift);
365 }
366
367 # it's true, baby ;-)
368
369 1;
370
371 #
372 ### Ende ###

patrick-canterino.de