/[cvs]/nfo/php/libs/net.php.smarty/Smarty_Compiler.class.php
ViewVC logotype

Annotation of /nfo/php/libs/net.php.smarty/Smarty_Compiler.class.php

Parent Directory Parent Directory | Revision Log Revision Log


Revision 1.2 - (hide annotations)
Thu Dec 19 16:40:20 2002 UTC (22 years ago) by joko
Branch: MAIN
Changes since 1.1: +19 -9 lines
+ updated to version 2.3.1

1 cvsjoko 1.1 <?php
2    
3     /*
4     * Project: Smarty: the PHP compiling template engine
5     * File: Smarty_Compiler.class.php
6     * Author: Monte Ohrt <monte@ispi.net>
7     * Andrei Zmievski <andrei@php.net>
8     *
9 joko 1.2 * Version: 2.3.1
10 cvsjoko 1.1 * Copyright: 2001,2002 ispi of Lincoln, Inc.
11     *
12     * This library is free software; you can redistribute it and/or
13     * modify it under the terms of the GNU Lesser General Public
14     * License as published by the Free Software Foundation; either
15     * version 2.1 of the License, or (at your option) any later version.
16     *
17     * This library is distributed in the hope that it will be useful,
18     * but WITHOUT ANY WARRANTY; without even the implied warranty of
19     * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
20     * Lesser General Public License for more details.
21     *
22     * You should have received a copy of the GNU Lesser General Public
23     * License along with this library; if not, write to the Free Software
24     * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
25     *
26     * You may contact the authors of Smarty by e-mail at:
27     * monte@ispi.net
28     * andrei@php.net
29     *
30     * Or, write to:
31     * Monte Ohrt
32     * Director of Technology, ispi
33     * 237 S. 70th suite 220
34     * Lincoln, NE 68510
35     *
36     * The latest version of Smarty can be obtained from:
37     * http://www.phpinsider.com/
38     *
39     */
40    
41     class Smarty_Compiler extends Smarty {
42    
43     // internal vars
44     var $_sectionelse_stack = array(); // keeps track of whether section had 'else' part
45     var $_foreachelse_stack = array(); // keeps track of whether foreach had 'else' part
46     var $_literal_blocks = array(); // keeps literal template blocks
47     var $_php_blocks = array(); // keeps php code blocks
48     var $_current_file = null; // the current template being compiled
49     var $_current_line_no = 1; // line number for error messages
50     var $_capture_stack = array(); // keeps track of nested capture buffers
51     var $_plugin_info = array(); // keeps track of plugins to load
52     var $_init_smarty_vars = false;
53    
54    
55     /*======================================================================*\
56     Function: _compile_file()
57     Input: compile a template file
58     \*======================================================================*/
59     function _compile_file($tpl_file, $template_source, &$template_compiled)
60     {
61     if ($this->security) {
62     // do not allow php syntax to be executed unless specified
63     if ($this->php_handling == SMARTY_PHP_ALLOW &&
64     !$this->security_settings['PHP_HANDLING']) {
65     $this->php_handling = SMARTY_PHP_PASSTHRU;
66     }
67     }
68    
69     $this->_load_filters();
70    
71     $this->_current_file = $tpl_file;
72     $this->_current_line_no = 1;
73     $ldq = preg_quote($this->left_delimiter, '!');
74     $rdq = preg_quote($this->right_delimiter, '!');
75    
76     // run template source through prefilter functions
77     if (count($this->_plugins['prefilter']) > 0) {
78     foreach ($this->_plugins['prefilter'] as $filter_name => $prefilter) {
79     if ($prefilter === false) continue;
80     if ($prefilter[3] || function_exists($prefilter[0])) {
81     $template_source = $prefilter[0]($template_source, $this);
82     $this->_plugins['prefilter'][$filter_name][3] = true;
83     } else {
84     $this->_trigger_plugin_error("Smarty plugin error: prefilter '$filter_name' is not implemented");
85     }
86     }
87     }
88    
89     /* Annihilate the comments. */
90     $template_source = preg_replace("!({$ldq})\*(.*?)\*({$rdq})!se",
91     "'\\1*'.str_repeat(\"\n\", substr_count('\\2', \"\n\")) .'*\\3'",
92     $template_source);
93    
94     /* Pull out the literal blocks. */
95     preg_match_all("!{$ldq}literal{$rdq}(.*?){$ldq}/literal{$rdq}!s", $template_source, $match);
96     $this->_literal_blocks = $match[1];
97     $template_source = preg_replace("!{$ldq}literal{$rdq}(.*?){$ldq}/literal{$rdq}!s",
98     $this->quote_replace($this->left_delimiter.'literal'.$this->right_delimiter), $template_source);
99    
100     /* Pull out the php code blocks. */
101     preg_match_all("!{$ldq}php{$rdq}(.*?){$ldq}/php{$rdq}!s", $template_source, $match);
102     $this->_php_blocks = $match[1];
103     $template_source = preg_replace("!{$ldq}php{$rdq}(.*?){$ldq}/php{$rdq}!s",
104     $this->quote_replace($this->left_delimiter.'php'.$this->right_delimiter), $template_source);
105    
106     /* Gather all template tags. */
107     preg_match_all("!{$ldq}\s*(.*?)\s*{$rdq}!s", $template_source, $match);
108     $template_tags = $match[1];
109     /* Split content by template tags to obtain non-template content. */
110     $text_blocks = preg_split("!{$ldq}.*?{$rdq}!s", $template_source);
111    
112     /* loop through text blocks */
113     for ($curr_tb = 0, $for_max = count($text_blocks); $curr_tb < $for_max; $curr_tb++) {
114     /* match anything within <? ?> */
115     if (preg_match_all('!(<\?[^?]*?\?>|<script\s+language\s*=\s*[\"\']?php[\"\']?\s*>)!is', $text_blocks[$curr_tb], $sp_match)) {
116     /* found at least one match, loop through each one */
117     for ($curr_sp = 0, $for_max2 = count($sp_match[0]); $curr_sp < $for_max2; $curr_sp++) {
118     if (preg_match('!^(<\?(php\s|\s|=\s)|<script\s*language\s*=\s*[\"\']?php[\"\']?\s*>)!is', $sp_match[0][$curr_sp])) {
119     /* php tag */
120     if ($this->php_handling == SMARTY_PHP_PASSTHRU) {
121     /* echo php contents */
122     $text_blocks[$curr_tb] = str_replace($sp_match[0][$curr_sp], '<?php echo \''.str_replace("'", "\'", $sp_match[0][$curr_sp]).'\'; ?>'."\n", $text_blocks[$curr_tb]);
123     } else if ($this->php_handling == SMARTY_PHP_QUOTE) {
124     /* quote php tags */
125     $text_blocks[$curr_tb] = str_replace($sp_match[0][$curr_sp], htmlspecialchars($sp_match[0][$curr_sp]), $text_blocks[$curr_tb]);
126     } else if ($this->php_handling == SMARTY_PHP_REMOVE) {
127     /* remove php tags */
128     if (substr($sp_match[0][$curr_sp], 0, 2) == '<?')
129     $text_blocks[$curr_tb] = str_replace($sp_match[0][$curr_sp], '', $text_blocks[$curr_tb]);
130     else
131     /* attempt to remove everything between <script ...> and </script> */
132     $text_blocks[$curr_tb] = preg_replace('!'.preg_quote($sp_match[0][$curr_sp], '!').'.*?</script\s*>!is', '', $text_blocks[$curr_tb]);
133     }
134     } else
135     /* echo the non-php tags */
136     $text_blocks[$curr_tb] = str_replace($sp_match[0][$curr_sp], '<?php echo \''.str_replace("'", "\'", $sp_match[0][$curr_sp]).'\'; ?>'."\n", $text_blocks[$curr_tb]);
137     }
138     }
139     }
140    
141     /* Compile the template tags into PHP code. */
142     $compiled_tags = array();
143     for ($i = 0, $for_max = count($template_tags); $i < $for_max; $i++) {
144     $this->_current_line_no += substr_count($text_blocks[$i], "\n");
145     $compiled_tags[] = $this->_compile_tag($template_tags[$i]);
146     $this->_current_line_no += substr_count($template_tags[$i], "\n");
147     }
148    
149     $template_compiled = '';
150    
151     /* Interleave the compiled contents and text blocks to get the final result. */
152     for ($i = 0, $for_max = count($compiled_tags); $i < $for_max; $i++) {
153     $template_compiled .= $text_blocks[$i].$compiled_tags[$i];
154     }
155     $template_compiled .= $text_blocks[$i];
156    
157     /* Reformat data between 'strip' and '/strip' tags, removing spaces, tabs and newlines. */
158     if (preg_match_all("!{$ldq}strip{$rdq}.*?{$ldq}/strip{$rdq}!s", $template_compiled, $match)) {
159     $strip_tags = $match[0];
160     $strip_tags_modified = preg_replace("!{$ldq}/?strip{$rdq}|[\t ]+$|^[\t ]+!m", '', $strip_tags);
161     $strip_tags_modified = preg_replace('![\r\n]+!m', '', $strip_tags_modified);
162     for ($i = 0, $for_max = count($strip_tags); $i < $for_max; $i++)
163     $template_compiled = preg_replace("!{$ldq}strip{$rdq}.*?{$ldq}/strip{$rdq}!s",
164     $this->quote_replace($strip_tags_modified[$i]),
165     $template_compiled, 1);
166     }
167    
168     // remove \n from the end of the file, if any
169     if ($template_compiled{strlen($template_compiled) - 1} == "\n" ) {
170     $template_compiled = substr($template_compiled, 0, -1);
171     }
172    
173     // run compiled template through postfilter functions
174     if (count($this->_plugins['postfilter']) > 0) {
175     foreach ($this->_plugins['postfilter'] as $filter_name => $postfilter) {
176     if ($postfilter === false) continue;
177     if ($postfilter[3] || function_exists($postfilter[0])) {
178     $template_compiled = $postfilter[0]($template_compiled, $this);
179     $this->_plugins['postfilter'][$filter_name][3] = true;
180     } else {
181     $this->_trigger_plugin_error("Smarty plugin error: postfilter '$filter_name' is not implemented");
182     }
183     }
184     }
185    
186     // put header at the top of the compiled template
187     $template_header = "<?php /* Smarty version ".$this->_version.", created on ".strftime("%Y-%m-%d %H:%M:%S")."\n";
188     $template_header .= " compiled from ".$tpl_file." */ ?>\n";
189    
190     /* Emit code to load needed plugins. */
191     if (count($this->_plugin_info)) {
192     $plugins_code = '<?php $this->_load_plugins(array(';
193     foreach ($this->_plugin_info as $plugin_type => $plugins) {
194     foreach ($plugins as $plugin_name => $plugin_info) {
195     $plugins_code .= "\narray('$plugin_type', '$plugin_name', '$plugin_info[0]', $plugin_info[1], ";
196     $plugins_code .= $plugin_info[2] ? 'true),' : 'false),';
197     }
198     }
199     $plugins_code .= ")); ?>";
200     $template_header .= $plugins_code;
201     $this->_plugin_info = array();
202     }
203    
204     if ($this->_init_smarty_vars) {
205     $template_header .= "<?php \$this->_assign_smarty_interface(); ?>\n";
206     $this->_init_smarty_vars = false;
207     }
208    
209     $template_compiled = $template_header . $template_compiled;
210    
211     return true;
212     }
213    
214    
215     /*======================================================================*\
216     Function: _compile_tag
217     Purpose: Compile a template tag
218     \*======================================================================*/
219     function _compile_tag($template_tag)
220     {
221     /* Matched comment. */
222     if ($template_tag{0} == '*' && $template_tag{strlen($template_tag) - 1} == '*')
223     return '';
224    
225     $qstr_regexp = '"[^"\\\\]*(?:\\\\.[^"\\\\]*)*"|\'[^\'\\\\]*(?:\\\\.[^\'\\\\]*)*\'';
226    
227     /* Split tag into two parts: command and the arguments. */
228     preg_match('/^(
229     (?: ' . $qstr_regexp . ' | (?>[^"\'\s]+))+
230     )
231     (?:\s+(.*))?
232     /xs', $template_tag, $match);
233     $tag_command = $match[1];
234     $tag_args = isset($match[2]) ? $match[2] : '';
235    
236     /* If the tag name matches a variable or section property definition,
237     we simply process it. */
238     if (preg_match('!^\$\w+(?>(\[(\d+|\$\w+|\w+(\.\w+)?)\])|((\.|->)\$?\w+))*(?>\|@?\w+(:(?>' . $qstr_regexp . '|[^|]+))*)*$!', $tag_command) || // if a variable
239     preg_match('!^#(\w+)#(?>\|@?\w+(:(?>' . $qstr_regexp . '|[^|]+))*)*$!', $tag_command) || // or a configuration variable
240     preg_match('!^%\w+\.\w+%(?>\|@?\w+(:(?>' . $qstr_regexp . '|[^|]+))*)*$!', $tag_command)) { // or a section property
241     settype($tag_command, 'array');
242     $this->_parse_vars_props($tag_command);
243     return "<?php echo $tag_command[0]; ?>\n";
244     }
245    
246     switch ($tag_command) {
247     case 'include':
248     return $this->_compile_include_tag($tag_args);
249    
250     case 'include_php':
251     return $this->_compile_include_php_tag($tag_args);
252    
253     case 'if':
254     return $this->_compile_if_tag($tag_args);
255    
256     case 'else':
257     return '<?php else: ?>';
258    
259     case 'elseif':
260     return $this->_compile_if_tag($tag_args, true);
261    
262     case '/if':
263     return '<?php endif; ?>';
264    
265     case 'capture':
266     return $this->_compile_capture_tag(true, $tag_args);
267    
268     case '/capture':
269     return $this->_compile_capture_tag(false);
270    
271     case 'ldelim':
272     return $this->left_delimiter;
273    
274     case 'rdelim':
275     return $this->right_delimiter;
276    
277     case 'section':
278     array_push($this->_sectionelse_stack, false);
279     return $this->_compile_section_start($tag_args);
280    
281     case 'sectionelse':
282     $this->_sectionelse_stack[count($this->_sectionelse_stack)-1] = true;
283     return "<?php endfor; else: ?>";
284    
285     case '/section':
286     if (array_pop($this->_sectionelse_stack))
287     return "<?php endif; ?>";
288     else
289     return "<?php endfor; endif; ?>";
290    
291     case 'foreach':
292     array_push($this->_foreachelse_stack, false);
293     return $this->_compile_foreach_start($tag_args);
294     break;
295    
296     case 'foreachelse':
297     $this->_foreachelse_stack[count($this->_foreachelse_stack)-1] = true;
298     return "<?php endforeach; else: ?>";
299    
300     case '/foreach':
301     if (array_pop($this->_foreachelse_stack))
302     return "<?php endif; ?>";
303     else
304     return "<?php endforeach; endif; ?>";
305    
306     case 'config_load':
307     return $this->_compile_config_load_tag($tag_args);
308    
309     case 'strip':
310     case '/strip':
311     return $this->left_delimiter.$tag_command.$this->right_delimiter;
312    
313     case 'literal':
314     list (,$literal_block) = each($this->_literal_blocks);
315     $this->_current_line_no += substr_count($literal_block, "\n");
316     return "<?php echo '".str_replace("'", "\'", str_replace("\\", "\\\\", $literal_block))."'; ?>\n";
317    
318     case 'php':
319     if ($this->security && !$this->security_settings['PHP_TAGS']) {
320     $this->_syntax_error("(secure mode) php tags not permitted", E_USER_WARNING);
321     return;
322     }
323     list (,$php_block) = each($this->_php_blocks);
324     $this->_current_line_no += substr_count($php_block, "\n");
325     return '<?php '.$php_block.' ?>';
326    
327     case 'insert':
328     return $this->_compile_insert_tag($tag_args);
329    
330     default:
331     if ($this->_compile_compiler_tag($tag_command, $tag_args, $output)) {
332     return $output;
333     } else if ($this->_compile_block_tag($tag_command, $tag_args, $output)) {
334     return $output;
335     } else {
336     return $this->_compile_custom_tag($tag_command, $tag_args);
337     }
338     }
339     }
340    
341    
342     /*======================================================================*\
343     Function: _compile_compiler_tag
344     Purpose: compile the custom compiler tag
345     \*======================================================================*/
346     function _compile_compiler_tag($tag_command, $tag_args, &$output)
347     {
348     $found = false;
349     $have_function = true;
350    
351     /*
352     * First we check if the compiler function has already been registered
353     * or loaded from a plugin file.
354     */
355     if (isset($this->_plugins['compiler'][$tag_command])) {
356     $found = true;
357     $plugin_func = $this->_plugins['compiler'][$tag_command][0];
358     if (!function_exists($plugin_func)) {
359     $message = "compiler function '$tag_command' is not implemented";
360     $have_function = false;
361     }
362     }
363     /*
364     * Otherwise we need to load plugin file and look for the function
365     * inside it.
366     */
367     else if ($plugin_file = $this->_get_plugin_filepath('compiler', $tag_command)) {
368     $found = true;
369    
370     include_once $plugin_file;
371    
372     $plugin_func = 'smarty_compiler_' . $tag_command;
373     if (!function_exists($plugin_func)) {
374     $message = "plugin function $plugin_func() not found in $plugin_file\n";
375     $have_function = false;
376     } else {
377     $this->_plugins['compiler'][$tag_command] = array($plugin_func, null, null);
378     }
379     }
380    
381     /*
382     * True return value means that we either found a plugin or a
383     * dynamically registered function. False means that we didn't and the
384     * compiler should now emit code to load custom function plugin for this
385     * tag.
386     */
387     if ($found) {
388     if ($have_function) {
389     $output = '<?php ' . $plugin_func($tag_args, $this) . ' ?>';
390     } else {
391     $this->_syntax_error($message, E_USER_WARNING);
392     }
393     return true;
394     } else {
395     return false;
396     }
397     }
398    
399    
400     /*======================================================================*\
401     Function: _compile_block_tag
402     Purpose: compile block function tag
403     \*======================================================================*/
404     function _compile_block_tag($tag_command, $tag_args, &$output)
405     {
406     if ($tag_command{0} == '/') {
407     $start_tag = false;
408     $tag_command = substr($tag_command, 1);
409     } else
410     $start_tag = true;
411    
412     $found = false;
413     $have_function = true;
414    
415     /*
416     * First we check if the block function has already been registered
417     * or loaded from a plugin file.
418     */
419     if (isset($this->_plugins['block'][$tag_command])) {
420     $found = true;
421     $plugin_func = $this->_plugins['block'][$tag_command][0];
422     if (!function_exists($plugin_func)) {
423     $message = "block function '$tag_command' is not implemented";
424     $have_function = false;
425     }
426     }
427     /*
428     * Otherwise we need to load plugin file and look for the function
429     * inside it.
430     */
431     else if ($plugin_file = $this->_get_plugin_filepath('block', $tag_command)) {
432     $found = true;
433    
434     include_once $plugin_file;
435    
436     $plugin_func = 'smarty_block_' . $tag_command;
437     if (!function_exists($plugin_func)) {
438     $message = "plugin function $plugin_func() not found in $plugin_file\n";
439     $have_function = false;
440     } else {
441     $this->_plugins['block'][$tag_command] = array($plugin_func, null, null);
442     }
443     }
444    
445     if (!$found) {
446     return false;
447     } else if (!$have_function) {
448     $this->_syntax_error($message, E_USER_WARNING);
449     return true;
450     }
451    
452     /*
453     * Even though we've located the plugin function, compilation
454     * happens only once, so the plugin will still need to be loaded
455     * at runtime for future requests.
456     */
457     $this->_add_plugin('block', $tag_command);
458    
459     if ($start_tag) {
460     $arg_list = array();
461     $attrs = $this->_parse_attrs($tag_args);
462     foreach ($attrs as $arg_name => $arg_value) {
463     if (is_bool($arg_value))
464     $arg_value = $arg_value ? 'true' : 'false';
465     $arg_list[] = "'$arg_name' => $arg_value";
466     }
467    
468     $output = "<?php \$this->_tag_stack[] = array('$tag_command', array(".implode(',', (array)$arg_list).")); \$this->_plugins['block']['$tag_command'][0](array(".implode(',', (array)$arg_list)."), null, \$this); ob_start(); ?>";
469     } else {
470     $output = "<?php \$this->_block_content = ob_get_contents(); ob_end_clean(); \$this->_plugins['block']['$tag_command'][0](\$this->_tag_stack[count(\$this->_tag_stack)-1][1], \$this->_block_content, \$this); array_pop(\$this->_tag_stack); ?>";
471     }
472    
473     return true;
474     }
475    
476    
477     /*======================================================================*\
478     Function: _compile_custom_tag
479     Purpose: compile custom function tag
480     \*======================================================================*/
481     function _compile_custom_tag($tag_command, $tag_args)
482     {
483     $this->_add_plugin('function', $tag_command);
484    
485     $arg_list = array();
486     $attrs = $this->_parse_attrs($tag_args);
487     foreach ($attrs as $arg_name => $arg_value) {
488     if (is_bool($arg_value))
489     $arg_value = $arg_value ? 'true' : 'false';
490     $arg_list[] = "'$arg_name' => $arg_value";
491     }
492    
493     return "<?php \$this->_plugins['function']['$tag_command'][0](array(".implode(',', (array)$arg_list)."), \$this); if(\$this->_extract) { extract(\$this->_tpl_vars); \$this->_extract=false; } ?>";
494     }
495    
496    
497     /*======================================================================*\
498     Function: _compile_insert_tag
499     Purpose: Compile {insert ...} tag
500     \*======================================================================*/
501     function _compile_insert_tag($tag_args)
502     {
503     $attrs = $this->_parse_attrs($tag_args);
504     $name = $this->_dequote($attrs['name']);
505    
506     if (empty($name)) {
507     $this->_syntax_error("missing insert name");
508     }
509    
510     if (!empty($attrs['script'])) {
511     $delayed_loading = true;
512     }
513    
514     foreach ($attrs as $arg_name => $arg_value) {
515     if (is_bool($arg_value))
516     $arg_value = $arg_value ? 'true' : 'false';
517     $arg_list[] = "'$arg_name' => $arg_value";
518     }
519    
520     $this->_add_plugin('insert', $name, $delayed_loading);
521    
522     return "<?php echo \$this->_run_insert_handler(array(".implode(', ', (array)$arg_list).")); ?>\n";
523     }
524    
525    
526     /*======================================================================*\
527     Function: _compile_config_load_tag
528     Purpose: Compile {config_load ...} tag
529     \*======================================================================*/
530     function _compile_config_load_tag($tag_args)
531     {
532     $attrs = $this->_parse_attrs($tag_args);
533    
534     if (empty($attrs['file'])) {
535     $this->_syntax_error("missing 'file' attribute in config_load tag");
536     }
537    
538     if (empty($attrs['section'])) {
539     $attrs['section'] = 'null';
540     }
541    
542     $scope = @$this->_dequote($attrs['scope']);
543     if (!empty($scope)) {
544     if ($scope != 'local' &&
545     $scope != 'parent' &&
546     $scope != 'global') {
547     $this->_syntax_error("invalid 'scope' attribute value");
548     }
549     } else {
550     if (!empty($attrs['global']) && $attrs['global'])
551     $scope = 'parent';
552     else
553     $scope = 'local';
554     }
555    
556     $output = '<?php $this->_config_load(' . $attrs['file'] . ', ' . $attrs['section'] . ", '$scope'); ?>";
557    
558     return $output;
559     }
560    
561    
562     /*======================================================================*\
563     Function: _compile_include_tag
564     Purpose: Compile {include ...} tag
565     \*======================================================================*/
566     function _compile_include_tag($tag_args)
567     {
568     $attrs = $this->_parse_attrs($tag_args);
569     $arg_list = array();
570    
571     if (empty($attrs['file'])) {
572     $this->_syntax_error("missing 'file' attribute in include tag");
573     }
574    
575     foreach ($attrs as $arg_name => $arg_value) {
576     if ($arg_name == 'file') {
577     $include_file = $arg_value;
578     continue;
579     } else if ($arg_name == 'assign') {
580     $assign_var = $arg_value;
581     continue;
582     }
583     if (is_bool($arg_value))
584     $arg_value = $arg_value ? 'true' : 'false';
585     $arg_list[] = "'$arg_name' => $arg_value";
586     }
587    
588     $output = '<?php ';
589    
590     if (isset($assign_var)) {
591     $output .= "ob_start();\n";
592     }
593    
594     $output .=
595     "\$_smarty_tpl_vars = \$this->_tpl_vars;\n" .
596     "\$this->_smarty_include(".$include_file.", array(".implode(',', (array)$arg_list)."));\n" .
597     "\$this->_tpl_vars = \$_smarty_tpl_vars;\n" .
598     "unset(\$_smarty_tpl_vars);\n";
599    
600     if (isset($assign_var)) {
601     $output .= "\$this->assign(" . $assign_var . ", ob_get_contents()); ob_end_clean();\n";
602     }
603    
604     $output .= ' ?>';
605    
606     return $output;
607    
608     }
609    
610     /*======================================================================*\
611     Function: _compile_include_php_tag
612     Purpose: Compile {include ...} tag
613     \*======================================================================*/
614     function _compile_include_php_tag($tag_args)
615     {
616     $attrs = $this->_parse_attrs($tag_args);
617    
618     if (empty($attrs['file'])) {
619     $this->_syntax_error("missing 'file' attribute in include_php tag");
620     }
621    
622     $assign_var = $this->_dequote($attrs['assign']);
623    
624     $once_var = ( $attrs['once'] === false ) ? 'false' : 'true';
625    
626     return "<?php \$this->_smarty_include_php($attrs[file], '$assign_var', $once_var); ?>";
627     }
628    
629    
630     /*======================================================================*\
631     Function: _compile_section_start
632     Purpose: Compile {section ...} tag
633     \*======================================================================*/
634     function _compile_section_start($tag_args)
635     {
636     $attrs = $this->_parse_attrs($tag_args);
637     $arg_list = array();
638    
639     $output = "<?php ";
640     $section_name = $attrs['name'];
641     if (empty($section_name)) {
642     $this->_syntax_error("missing section name");
643     }
644    
645     $output .= "if (isset(\$this->_sections[$section_name])) unset(\$this->_sections[$section_name]);\n";
646     $section_props = "\$this->_sections[$section_name]";
647    
648     foreach ($attrs as $attr_name => $attr_value) {
649     switch ($attr_name) {
650     case 'loop':
651     $output .= "{$section_props}['loop'] = is_array($attr_value) ? count($attr_value) : max(0, (int)$attr_value);\n";
652     break;
653    
654     case 'show':
655     if (is_bool($attr_value))
656     $show_attr_value = $attr_value ? 'true' : 'false';
657     else
658     $show_attr_value = "(bool)$attr_value";
659     $output .= "{$section_props}['show'] = $show_attr_value;\n";
660     break;
661    
662     case 'name':
663     $output .= "{$section_props}['$attr_name'] = $attr_value;\n";
664     break;
665    
666     case 'max':
667     case 'start':
668     $output .= "{$section_props}['$attr_name'] = (int)$attr_value;\n";
669     break;
670    
671     case 'step':
672     $output .= "{$section_props}['$attr_name'] = ((int)$attr_value) == 0 ? 1 : (int)$attr_value;\n";
673     break;
674    
675     default:
676     $this->_syntax_error("unknown section attribute - '$attr_name'");
677     break;
678     }
679     }
680    
681     if (!isset($attrs['show']))
682     $output .= "{$section_props}['show'] = true;\n";
683    
684     if (!isset($attrs['loop']))
685     $output .= "{$section_props}['loop'] = 1;\n";
686    
687     if (!isset($attrs['max']))
688     $output .= "{$section_props}['max'] = {$section_props}['loop'];\n";
689     else
690     $output .= "if ({$section_props}['max'] < 0)\n" .
691     " {$section_props}['max'] = {$section_props}['loop'];\n";
692    
693     if (!isset($attrs['step']))
694     $output .= "{$section_props}['step'] = 1;\n";
695    
696     if (!isset($attrs['start']))
697     $output .= "{$section_props}['start'] = {$section_props}['step'] > 0 ? 0 : {$section_props}['loop']-1;\n";
698     else {
699     $output .= "if ({$section_props}['start'] < 0)\n" .
700     " {$section_props}['start'] = max({$section_props}['step'] > 0 ? 0 : -1, {$section_props}['loop'] + {$section_props}['start']);\n" .
701     "else\n" .
702     " {$section_props}['start'] = min({$section_props}['start'], {$section_props}['step'] > 0 ? {$section_props}['loop'] : {$section_props}['loop']-1);\n";
703     }
704    
705     $output .= "if ({$section_props}['show']) {\n";
706     if (!isset($attrs['start']) && !isset($attrs['step']) && !isset($attrs['max'])) {
707     $output .= " {$section_props}['total'] = {$section_props}['loop'];\n";
708     } else {
709     $output .= " {$section_props}['total'] = min(ceil(({$section_props}['step'] > 0 ? {$section_props}['loop'] - {$section_props}['start'] : {$section_props}['start']+1)/abs({$section_props}['step'])), {$section_props}['max']);\n";
710     }
711     $output .= " if ({$section_props}['total'] == 0)\n" .
712     " {$section_props}['show'] = false;\n" .
713     "} else\n" .
714     " {$section_props}['total'] = 0;\n";
715    
716     $output .= "if ({$section_props}['show']):\n";
717     $output .= "
718     for ({$section_props}['index'] = {$section_props}['start'], {$section_props}['iteration'] = 1;
719     {$section_props}['iteration'] <= {$section_props}['total'];
720     {$section_props}['index'] += {$section_props}['step'], {$section_props}['iteration']++):\n";
721     $output .= "{$section_props}['rownum'] = {$section_props}['iteration'];\n";
722     $output .= "{$section_props}['index_prev'] = {$section_props}['index'] - {$section_props}['step'];\n";
723     $output .= "{$section_props}['index_next'] = {$section_props}['index'] + {$section_props}['step'];\n";
724     $output .= "{$section_props}['first'] = ({$section_props}['iteration'] == 1);\n";
725     $output .= "{$section_props}['last'] = ({$section_props}['iteration'] == {$section_props}['total']);\n";
726    
727     $output .= "?>";
728    
729     return $output;
730     }
731    
732    
733     /*======================================================================*\
734     Function: _compile_foreach_start
735     Purpose: Compile {foreach ...} tag
736     \*======================================================================*/
737     function _compile_foreach_start($tag_args)
738     {
739     $attrs = $this->_parse_attrs($tag_args);
740     $arg_list = array();
741    
742     if (empty($attrs['from'])) {
743     $this->_syntax_error("missing 'from' attribute");
744     }
745    
746     if (empty($attrs['item'])) {
747     $this->_syntax_error("missing 'item' attribute");
748     }
749    
750     $from = $attrs['from'];
751     $item = $this->_dequote($attrs['item']);
752     if (isset($attrs['name']))
753     $name = $attrs['name'];
754    
755     $output = '<?php ';
756     if (isset($name)) {
757     $output .= "if (isset(\$this->_foreach[$name])) unset(\$this->_foreach[$name]);\n";
758     $foreach_props = "\$this->_foreach[$name]";
759     }
760    
761     $key_part = '';
762    
763     foreach ($attrs as $attr_name => $attr_value) {
764     switch ($attr_name) {
765     case 'key':
766     $key = $this->_dequote($attrs['key']);
767     $key_part = "\$this->_tpl_vars['$key'] => ";
768     break;
769    
770     case 'name':
771     $output .= "{$foreach_props}['$attr_name'] = $attr_value;\n";
772     break;
773     }
774     }
775    
776     if (isset($name)) {
777     $output .= "{$foreach_props}['total'] = count((array)$from);\n";
778     $output .= "{$foreach_props}['show'] = {$foreach_props}['total'] > 0;\n";
779     $output .= "if ({$foreach_props}['show']):\n";
780     $output .= "{$foreach_props}['iteration'] = 0;\n";
781     $output .= " foreach ((array)$from as $key_part\$this->_tpl_vars['$item']):\n";
782     $output .= " {$foreach_props}['iteration']++;\n";
783     $output .= " {$foreach_props}['first'] = ({$foreach_props}['iteration'] == 1);\n";
784     $output .= " {$foreach_props}['last'] = ({$foreach_props}['iteration'] == {$foreach_props}['total']);\n";
785     } else {
786     $output .= "if (count((array)$from)):\n";
787     $output .= " foreach ((array)$from as $key_part\$this->_tpl_vars['$item']):\n";
788     }
789     $output .= '?>';
790    
791     return $output;
792     }
793    
794    
795     /*======================================================================*\
796     Function: _compile_capture_tag
797     Purpose: Compile {capture} .. {/capture} tags
798     \*======================================================================*/
799     function _compile_capture_tag($start, $tag_args = '')
800     {
801     $attrs = $this->_parse_attrs($tag_args);
802    
803     if ($start) {
804     if (isset($attrs['name']))
805     $buffer = $attrs['name'];
806     else
807     $buffer = "'default'";
808    
809     $output = "<?php ob_start(); ?>";
810     $this->_capture_stack[] = $buffer;
811     } else {
812     $buffer = array_pop($this->_capture_stack);
813     $output = "<?php \$this->_smarty_vars['capture'][$buffer] = ob_get_contents(); ob_end_clean(); ?>";
814     }
815    
816     return $output;
817     }
818    
819    
820     /*======================================================================*\
821     Function: _compile_if_tag
822     Purpose: Compile {if ...} tag
823     \*======================================================================*/
824     function _compile_if_tag($tag_args, $elseif = false)
825     {
826     /* Tokenize args for 'if' tag. */
827     preg_match_all('/(?:
828     "[^"\\\\]*(?:\\\\.[^"\\\\]*)*" | # match all double quoted strings allowing escaped double quotes
829     \'[^\'\\\\]*(?:\\\\.[^\'\\\\]*)*\' | # match all single quoted strings allowing escaped single quotes
830     [(),] | # match parentheses and commas
831     [^\s(),]+ # match any other token that is not any of the above
832     )/x', $tag_args, $match);
833     $tokens = $match[0];
834    
835     $this->_parse_vars_props($tokens);
836    
837     $is_arg_stack = array();
838    
839     for ($i = 0, $for_max = count($tokens); $i < $for_max; $i++) {
840    
841     $token = &$tokens[$i];
842     switch ($token) {
843     case 'eq':
844     $token = '==';
845     break;
846    
847     case 'ne':
848     case 'neq':
849     $token = '!=';
850     break;
851    
852     case 'lt':
853     $token = '<';
854     break;
855    
856     case 'le':
857     case 'lte':
858     $token = '<=';
859     break;
860    
861     case 'gt':
862     $token = '>';
863     break;
864    
865     case 'ge':
866     case 'gte':
867     $token = '>=';
868     break;
869    
870     case 'and':
871     $token = '&&';
872     break;
873    
874     case 'or':
875     $token = '||';
876     break;
877    
878     case 'not':
879     $token = '!';
880     break;
881    
882     case 'mod':
883     $token = '%';
884     break;
885    
886     case '(':
887     array_push($is_arg_stack, $i);
888     break;
889    
890     case 'is':
891     /* If last token was a ')', we operate on the parenthesized
892     expression. The start of the expression is on the stack.
893     Otherwise, we operate on the last encountered token. */
894     if ($tokens[$i-1] == ')')
895     $is_arg_start = array_pop($is_arg_stack);
896     else
897     $is_arg_start = $i-1;
898     /* Construct the argument for 'is' expression, so it knows
899     what to operate on. */
900     $is_arg = implode(' ', array_slice($tokens, $is_arg_start, $i - $is_arg_start));
901    
902     /* Pass all tokens from next one until the end to the
903     'is' expression parsing function. The function will
904     return modified tokens, where the first one is the result
905     of the 'is' expression and the rest are the tokens it
906     didn't touch. */
907     $new_tokens = $this->_parse_is_expr($is_arg, array_slice($tokens, $i+1));
908    
909     /* Replace the old tokens with the new ones. */
910     array_splice($tokens, $is_arg_start, count($tokens), $new_tokens);
911    
912     /* Adjust argument start so that it won't change from the
913     current position for the next iteration. */
914     $i = $is_arg_start;
915     break;
916    
917     default:
918     if($this->security &&
919     $i+1 < count($tokens) &&
920     $tokens[$i+1] == '(' &&
921     preg_match('!^[a-zA-Z_]\w+$!', $tokens[$i]) &&
922     !in_array($tokens[$i], $this->security_settings['IF_FUNCS'])) {
923     $this->_syntax_error("(secure mode) '".$tokens[$i]."' not allowed in if statement");
924     }
925     break;
926     }
927     }
928    
929     if ($elseif)
930     return '<?php elseif ('.implode(' ', $tokens).'): ?>';
931     else
932     return '<?php if ('.implode(' ', $tokens).'): ?>';
933     }
934    
935    
936     /*======================================================================*\
937     Function: _parse_is_expr
938     Purpose: Parse is expression
939     \*======================================================================*/
940     function _parse_is_expr($is_arg, $tokens)
941     {
942     $expr_end = 0;
943     $negate_expr = false;
944    
945     if (($first_token = array_shift($tokens)) == 'not') {
946     $negate_expr = true;
947     $expr_type = array_shift($tokens);
948     } else
949     $expr_type = $first_token;
950    
951     switch ($expr_type) {
952     case 'even':
953     if (@$tokens[$expr_end] == 'by') {
954     $expr_end++;
955     $expr_arg = $tokens[$expr_end++];
956     $expr = "!(($is_arg / $expr_arg) % $expr_arg)";
957     } else
958     $expr = "!($is_arg % 2)";
959     break;
960    
961     case 'odd':
962     if (@$tokens[$expr_end] == 'by') {
963     $expr_end++;
964     $expr_arg = $tokens[$expr_end++];
965     $expr = "(($is_arg / $expr_arg) % $expr_arg)";
966     } else
967     $expr = "($is_arg % 2)";
968     break;
969    
970     case 'div':
971     if (@$tokens[$expr_end] == 'by') {
972     $expr_end++;
973     $expr_arg = $tokens[$expr_end++];
974     $expr = "!($is_arg % $expr_arg)";
975     } else {
976     $this->_syntax_error("expecting 'by' after 'div'");
977     }
978     break;
979    
980     default:
981     $this->_syntax_error("unknown 'is' expression - '$expr_type'");
982     break;
983     }
984    
985     if ($negate_expr) {
986     $expr = "!($expr)";
987     }
988    
989     array_splice($tokens, 0, $expr_end, $expr);
990    
991     return $tokens;
992     }
993    
994    
995     /*======================================================================*\
996     Function: _parse_attrs
997     Purpose: Parse attribute string
998     \*======================================================================*/
999     function _parse_attrs($tag_args, $quote = true)
1000     {
1001     /* Tokenize tag attributes. */
1002     preg_match_all('/(?:"[^"\\\\]*(?:\\\\.[^"\\\\]*)*" |
1003     \'[^\'\\\\]*(?:\\\\.[^\'\\\\]*)*\' | (?>[^"\'=\s]+)
1004     )+ |
1005     [=]
1006     /x', $tag_args, $match);
1007     $tokens = $match[0];
1008     $var_delims = array('$', '#', '%');
1009    
1010     $attrs = array();
1011     /* Parse state:
1012     0 - expecting attribute name
1013     1 - expecting '='
1014     2 - expecting attribute value (not '=') */
1015     $state = 0;
1016    
1017     foreach ($tokens as $token) {
1018     switch ($state) {
1019     case 0:
1020     /* If the token is a valid identifier, we set attribute name
1021     and go to state 1. */
1022     if (preg_match('!^\w+$!', $token)) {
1023     $attr_name = $token;
1024     $state = 1;
1025     } else
1026     $this->_syntax_error("invalid attribute name - '$token'");
1027     break;
1028    
1029     case 1:
1030     /* If the token is '=', then we go to state 2. */
1031     if ($token == '=') {
1032     $state = 2;
1033     } else
1034     $this->_syntax_error("expecting '=' after attribute name");
1035     break;
1036    
1037     case 2:
1038     /* If token is not '=', we set the attribute value and go to
1039     state 0. */
1040     if ($token != '=') {
1041     /* We booleanize the token if it's a non-quoted possible
1042     boolean value. */
1043     if (preg_match('!^(on|yes|true)$!', $token))
1044     $token = true;
1045     else if (preg_match('!^(off|no|false)$!', $token))
1046     $token = false;
1047     /* If the token is not variable (doesn't start with
1048     '$', '#', or '%') and not enclosed in single or
1049     double quotes we single-quote it. */
1050     else if ($quote && !in_array($token{0}, $var_delims) &&
1051     !(($token{0} == '"' || $token{0} == "'") &&
1052     $token{strlen($token)-1} == $token{0}))
1053     $token = '"'.$token.'"';
1054    
1055     $attrs[$attr_name] = $token;
1056     $state = 0;
1057     } else
1058     $this->_syntax_error("'=' cannot be an attribute value");
1059     break;
1060     }
1061     }
1062    
1063     $this->_parse_vars_props($attrs);
1064    
1065     return $attrs;
1066     }
1067    
1068    
1069     /*======================================================================*\
1070     Function: _parse_vars_props
1071     Purpose: compile variables and section properties tokens into
1072     PHP code
1073     \*======================================================================*/
1074     function _parse_vars_props(&$tokens)
1075     {
1076     $qstr_regexp = '"[^"\\\\]*(?:\\\\.[^"\\\\]*)*"|\'[^\'\\\\]*(?:\\\\.[^\'\\\\]*)*\'';
1077    
1078     $var_exprs = preg_grep('!^\$\w+(?>(\[(\d+|\$\w+|\w+(\.\w+)?)\])|((\.|->)\$?\w+))*(?>\|@?\w+(:(?>' . $qstr_regexp . '|[^|]+))*)*$!', $tokens);
1079     $conf_var_exprs = preg_grep('!^#(\w+)#(?>\|@?\w+(:(?>' . $qstr_regexp . '|[^|]+))*)*$!', $tokens);
1080     $sect_prop_exprs = preg_grep('!^%\w+\.\w+%(?>\|@?\w+(:(?>' . $qstr_regexp . '|[^|]+))*)*$!', $tokens);
1081    
1082     if (count($var_exprs)) {
1083     foreach ($var_exprs as $expr_index => $var_expr) {
1084     $tokens[$expr_index] = $this->_parse_var($var_expr);
1085     }
1086     }
1087    
1088     if (count($conf_var_exprs)) {
1089     foreach ($conf_var_exprs as $expr_index => $var_expr) {
1090     $tokens[$expr_index] = $this->_parse_conf_var($var_expr);
1091     }
1092     }
1093    
1094     if (count($sect_prop_exprs)) {
1095     foreach ($sect_prop_exprs as $expr_index => $section_prop_expr) {
1096     $tokens[$expr_index] = $this->_parse_section_prop($section_prop_expr);
1097     }
1098     }
1099     }
1100    
1101    
1102     /*======================================================================*\
1103     Function: _parse_var
1104     Purpose: parse variable expression into PHP code
1105     \*======================================================================*/
1106     function _parse_var($var_expr)
1107     {
1108     $parts = explode('|', substr($var_expr, 1), 2);
1109     $var_ref = $parts[0];
1110     $modifiers = isset($parts[1]) ? $parts[1] : '';
1111    
1112     if(!empty($this->default_modifiers) && !preg_match('!(^|\|)smarty:nodefaults($|\|)!',$modifiers)) {
1113     $_default_mod_string = implode('|',(array)$this->default_modifiers);
1114     $modifiers = empty($modifiers) ? $_default_mod_string : $_default_mod_string . '|' . $modifiers;
1115     }
1116    
1117     preg_match_all('!\[(?:\$\w+|\w+(\.\w+)?)\]|(->|\.)\$?\w+|^\w+!', $var_ref, $match);
1118     $indexes = $match[0];
1119     $var_name = array_shift($indexes);
1120    
1121     /* Handle $smarty.* variable references as a special case. */
1122     if ($var_name == 'smarty') {
1123     /*
1124     * If the reference could be compiled, use the compiled output;
1125     * otherwise, fall back on the $smarty variable generated at
1126     * run-time.
1127     */
1128     if (($smarty_ref = $this->_compile_smarty_ref($indexes)) !== null) {
1129     $output = $smarty_ref;
1130     } else {
1131     $var_name = substr(array_shift($indexes), 1);
1132     $output = "\$this->_smarty_vars['$var_name']";
1133     }
1134     } else {
1135     $output = "\$this->_tpl_vars['$var_name']";
1136     }
1137    
1138     foreach ($indexes as $index) {
1139     if ($index{0} == '[') {
1140     $index = substr($index, 1, -1);
1141     if (is_numeric($index)) {
1142     $output .= "[$index]";
1143     } elseif ($index{0} == '$') {
1144     $output .= "[\$this->_tpl_vars['" . substr($index, 1) . "']]";
1145     } else {
1146     $parts = explode('.', $index);
1147     $section = $parts[0];
1148     $section_prop = isset($parts[1]) ? $parts[1] : 'index';
1149     $output .= "[\$this->_sections['$section']['$section_prop']]";
1150     }
1151     } else if ($index{0} == '.') {
1152     if ($index{1} == '$')
1153     $output .= "[\$this->_tpl_vars['" . substr($index, 2) . "']]";
1154     else
1155     $output .= "['" . substr($index, 1) . "']";
1156     } else {
1157     $output .= $index;
1158     }
1159     }
1160    
1161     $this->_parse_modifiers($output, $modifiers);
1162    
1163     return $output;
1164     }
1165    
1166    
1167     /*======================================================================*\
1168     Function: _parse_conf_var
1169     Purpose: parse configuration variable expression into PHP code
1170     \*======================================================================*/
1171     function _parse_conf_var($conf_var_expr)
1172     {
1173     $parts = explode('|', $conf_var_expr, 2);
1174     $var_ref = $parts[0];
1175     $modifiers = isset($parts[1]) ? $parts[1] : '';
1176    
1177     $var_name = substr($var_ref, 1, -1);
1178    
1179     $output = "\$this->_config[0]['vars']['$var_name']";
1180    
1181     $this->_parse_modifiers($output, $modifiers);
1182    
1183     return $output;
1184     }
1185    
1186    
1187     /*======================================================================*\
1188     Function: _parse_section_prop
1189     Purpose: parse section property expression into PHP code
1190     \*======================================================================*/
1191     function _parse_section_prop($section_prop_expr)
1192     {
1193     $parts = explode('|', $section_prop_expr, 2);
1194     $var_ref = $parts[0];
1195     $modifiers = isset($parts[1]) ? $parts[1] : '';
1196    
1197     preg_match('!%(\w+)\.(\w+)%!', $var_ref, $match);
1198     $section_name = $match[1];
1199     $prop_name = $match[2];
1200    
1201     $output = "\$this->_sections['$section_name']['$prop_name']";
1202    
1203     $this->_parse_modifiers($output, $modifiers);
1204    
1205     return $output;
1206     }
1207    
1208    
1209     /*======================================================================*\
1210     Function: _parse_modifiers
1211     Purpose: parse modifier chain into PHP code
1212     \*======================================================================*/
1213     function _parse_modifiers(&$output, $modifier_string)
1214     {
1215     $qstr_regexp = '"[^"\\\\]*(?:\\\\.[^"\\\\]*)*"|\'[^\'\\\\]*(?:\\\\.[^\'\\\\]*)*\'';
1216     preg_match_all('!\|(@?\w+)((?>:(?:'. $qstr_regexp . '|[^|]+))*)!', '|' . $modifier_string, $match);
1217     list(, $modifiers, $modifier_arg_strings) = $match;
1218    
1219     for ($i = 0, $for_max = count($modifiers); $i < $for_max; $i++) {
1220     $modifier_name = $modifiers[$i];
1221    
1222     if($modifier_name == 'smarty') {
1223     // skip smarty modifier
1224     continue;
1225     }
1226    
1227     preg_match_all('!:(' . $qstr_regexp . '|[^:]+)!', $modifier_arg_strings[$i], $match);
1228     $modifier_args = $match[1];
1229    
1230     if ($modifier_name{0} == '@') {
1231     $map_array = 'false';
1232     $modifier_name = substr($modifier_name, 1);
1233     } else {
1234     $map_array = 'true';
1235     }
1236    
1237     $this->_add_plugin('modifier', $modifier_name);
1238    
1239     $this->_parse_vars_props($modifier_args);
1240    
1241     if (count($modifier_args) > 0)
1242     $modifier_args = ', '.implode(', ', $modifier_args);
1243     else
1244     $modifier_args = '';
1245    
1246     $output = "\$this->_run_mod_handler('$modifier_name', $map_array, $output$modifier_args)";
1247     }
1248     }
1249    
1250    
1251     /*======================================================================*\
1252     Function: _add_plugin
1253     Purpose:
1254     \*======================================================================*/
1255     function _add_plugin($type, $name, $delayed_loading = null)
1256     {
1257     if (!isset($this->_plugin_info[$type])) {
1258     $this->_plugin_info[$type] = array();
1259     }
1260     if (!isset($this->_plugin_info[$type][$name])) {
1261     $this->_plugin_info[$type][$name] = array($this->_current_file,
1262     $this->_current_line_no,
1263     $delayed_loading);
1264     }
1265     }
1266    
1267    
1268     /*======================================================================*\
1269     Function: _compile_smarty_ref
1270     Purpose: Compiles references of type $smarty.foo
1271     \*======================================================================*/
1272     function _compile_smarty_ref(&$indexes)
1273     {
1274     /* Extract the reference name. */
1275     $ref = substr($indexes[0], 1);
1276    
1277     switch ($ref) {
1278     case 'now':
1279     $compiled_ref = 'time()';
1280     if (count($indexes) > 1) {
1281     $this->_syntax_error('$smarty' . implode('', $indexes) .' is an invalid reference');
1282     }
1283     break;
1284    
1285     case 'foreach':
1286     case 'section':
1287     if ($indexes[1]{0} != '.') {
1288     $this->_syntax_error('$smarty' . implode('', array_slice($indexes, 0, 2)) . ' is an invalid reference');
1289     }
1290     $name = substr($indexes[1], 1);
1291     array_shift($indexes);
1292     if ($ref == 'foreach')
1293     $compiled_ref = "\$this->_foreach['$name']";
1294     else
1295     $compiled_ref = "\$this->_sections['$name']";
1296     break;
1297    
1298     case 'get':
1299     array_shift($indexes);
1300 joko 1.2 $compiled_ref = "\$GLOBALS['HTTP_GET_VARS']";
1301     if ($name = substr($indexes[0], 1))
1302     $compiled_ref .= "['$name']";
1303 cvsjoko 1.1 break;
1304    
1305     case 'post':
1306     array_shift($indexes);
1307     $name = substr($indexes[0], 1);
1308 joko 1.2 $compiled_ref = "\$GLOBALS['HTTP_POST_VARS']";
1309     if ($name = substr($indexes[0], 1))
1310     $compiled_ref .= "['$name']";
1311 cvsjoko 1.1 break;
1312    
1313     case 'cookies':
1314     array_shift($indexes);
1315     $name = substr($indexes[0], 1);
1316 joko 1.2 $compiled_ref = "\$GLOBALS['HTTP_COOKIE_VARS']";
1317     if ($name = substr($indexes[0], 1))
1318     $compiled_ref .= "['$name']";
1319 cvsjoko 1.1 break;
1320    
1321     case 'env':
1322     array_shift($indexes);
1323 joko 1.2 $compiled_ref = "\$GLOBALS['HTTP_ENV_VARS']";
1324     if ($name = substr($indexes[0], 1))
1325     $compiled_ref .= "['$name']";
1326 cvsjoko 1.1 break;
1327    
1328     case 'server':
1329     array_shift($indexes);
1330     $name = substr($indexes[0], 1);
1331 joko 1.2 $compiled_ref = "\$GLOBALS['HTTP_SERVER_VARS']";
1332     if ($name = substr($indexes[0], 1))
1333     $compiled_ref .= "['$name']";
1334 cvsjoko 1.1 break;
1335    
1336     case 'session':
1337     array_shift($indexes);
1338     $name = substr($indexes[0], 1);
1339 joko 1.2 $compiled_ref = "\$GLOBALS['HTTP_SESSION_VARS']";
1340     if ($name = substr($indexes[0], 1))
1341     $compiled_ref .= "['$name']";
1342 cvsjoko 1.1 break;
1343    
1344     /*
1345     * These cases are handled either at run-time or elsewhere in the
1346     * compiler.
1347     */
1348     case 'request':
1349     $this->_init_smarty_vars = true;
1350     return null;
1351    
1352     case 'capture':
1353     return null;
1354    
1355     case 'template':
1356     $compiled_ref = "'$this->_current_file'";
1357     if (count($indexes) > 1) {
1358     $this->_syntax_error('$smarty' . implode('', $indexes) .' is an invalid reference');
1359     }
1360     break;
1361    
1362     case 'version':
1363     $compiled_ref = "'$this->_version'";
1364     break;
1365    
1366     default:
1367     $this->_syntax_error('$smarty.' . $ref . ' is an unknown reference');
1368     break;
1369     }
1370    
1371     array_shift($indexes);
1372     return $compiled_ref;
1373     }
1374    
1375    
1376     /*======================================================================*\
1377     Function: _load_filters
1378     Purpose: load pre- and post-filters
1379     \*======================================================================*/
1380     function _load_filters()
1381     {
1382     if (count($this->_plugins['prefilter']) > 0) {
1383     foreach ($this->_plugins['prefilter'] as $filter_name => $prefilter) {
1384     if ($prefilter === false) {
1385     unset($this->_plugins['prefilter'][$filter_name]);
1386     $this->_load_plugins(array(array('prefilter', $filter_name, null, null, false)));
1387     }
1388     }
1389     }
1390     if (count($this->_plugins['postfilter']) > 0) {
1391     foreach ($this->_plugins['postfilter'] as $filter_name => $postfilter) {
1392     if ($postfilter === false) {
1393     unset($this->_plugins['postfilter'][$filter_name]);
1394     $this->_load_plugins(array(array('postfilter', $filter_name, null, null, false)));
1395     }
1396     }
1397     }
1398     }
1399    
1400    
1401     /*======================================================================*\
1402     Function: _syntax_error
1403     Purpose: display Smarty syntax error
1404     \*======================================================================*/
1405     function _syntax_error($error_msg, $error_type = E_USER_ERROR)
1406     {
1407     trigger_error("Smarty: [in " . $this->_current_file . " line " .
1408     $this->_current_line_no . "]: syntax error: $error_msg", $error_type);
1409     }
1410     }
1411    
1412     /* vim: set et: */
1413    
1414     ?>

MailToCvsAdmin">MailToCvsAdmin
ViewVC Help
Powered by ViewVC 1.1.26 RSS 2.0 feed