You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							462 lines
						
					
					
						
							14 KiB
						
					
					
				
			
		
		
		
			
			
			
				
					
				
				
					
				
			
		
		
	
	
							462 lines
						
					
					
						
							14 KiB
						
					
					
				
								<?php
							 | 
						|
								
							 | 
						|
								class XTemplate {
							 | 
						|
								
							 | 
						|
								/*
							 | 
						|
									xtemplate class 0.2.4-3
							 | 
						|
									html generation with templates - fast & easy
							 | 
						|
									copyright (c) 2000 barnabás debreceni [cranx@users.sourceforge.net]
							 | 
						|
									code optimization by Ivar Smolin <okul@linux.ee> 14-march-2001
							 | 
						|
									latest stable & CVS version always available @ http://sourceforge.net/projects/xtpl
							 | 
						|
								
							 | 
						|
									tested with php 3.0.11 and 4.0.4pl1
							 | 
						|
								
							 | 
						|
									This program is free software; you can redistribute it and/or
							 | 
						|
									modify it under the terms of the GNU Lesser General Public License
							 | 
						|
									version 2.1 as published by the Free Software Foundation.
							 | 
						|
								
							 | 
						|
									This library is distributed in the hope that it will be useful,
							 | 
						|
									but WITHOUT ANY WARRANTY; without even the implied warranty of
							 | 
						|
									MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
							 | 
						|
									GNU Lesser General Public License for more details at 
							 | 
						|
									http://www.gnu.org/copyleft/lgpl.html
							 | 
						|
								  
							 | 
						|
									You should have received a copy of the GNU General Public License
							 | 
						|
									along with this program; if not, write to the Free Software
							 | 
						|
									Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
							 | 
						|
								
							 | 
						|
									$Id: xtpl.php,v 1.2 2004/10/06 09:02:02 jack Exp $
							 | 
						|
								*/
							 | 
						|
								
							 | 
						|
								/***[ variables ]***********************************************************/
							 | 
						|
								
							 | 
						|
								var $filecontents="";								/* raw contents of template file */
							 | 
						|
								var $blocks=array();								/* unparsed blocks */
							 | 
						|
								var $parsed_blocks=array();					/* parsed blocks */
							 | 
						|
								var $block_parse_order=array();			/* block parsing order for recursive parsing (sometimes reverse:) */
							 | 
						|
								var $sub_blocks=array();						/* store sub-block names for fast resetting */
							 | 
						|
								var $VARS=array();									/* variables array */
							 | 
						|
								var $alternate_include_directory = "";
							 | 
						|
								
							 | 
						|
								var $file_delim="/\{FILE\s*\"([^\"]+)\"\s*\}/m";  /* regexp for file includes */
							 | 
						|
								var $block_start_delim="<!-- ";			/* block start delimiter */
							 | 
						|
								var $block_end_delim="-->";					/* block end delimiter */
							 | 
						|
								var $block_start_word="BEGIN:";			/* block start word */
							 | 
						|
								var $block_end_word="END:";					/* block end word */
							 | 
						|
								
							 | 
						|
								/* this makes the delimiters look like: <!-- BEGIN: block_name --> if you use my syntax. */
							 | 
						|
								
							 | 
						|
								var $NULL_STRING=array(""=>"");				/* null string for unassigned vars */
							 | 
						|
								var $NULL_BLOCK=array(""=>"");	/* null string for unassigned blocks */
							 | 
						|
								var $mainblock="";
							 | 
						|
								var $ERROR="";
							 | 
						|
								var $AUTORESET=1;										/* auto-reset sub blocks */
							 | 
						|
								
							 | 
						|
								/***[ constructor ]*********************************************************/
							 | 
						|
								
							 | 
						|
								function XTemplate ($file, $alt_include = "", $mainblock="main") {
							 | 
						|
									$this->alternate_include_directory = $alt_include;
							 | 
						|
									$this->mainblock=$mainblock;
							 | 
						|
									$this->filecontents=$this->r_getfile($file);	/* read in template file */
							 | 
						|
									$this->blocks=$this->maketree($this->filecontents,$mainblock);	/* preprocess some stuff */
							 | 
						|
									$this->scan_globals();
							 | 
						|
								}
							 | 
						|
								
							 | 
						|
								
							 | 
						|
								/***************************************************************************/
							 | 
						|
								/***[ public stuff ]********************************************************/
							 | 
						|
								/***************************************************************************/
							 | 
						|
								
							 | 
						|
								
							 | 
						|
								/***[ assign ]**************************************************************/
							 | 
						|
								/*
							 | 
						|
									assign a variable 
							 | 
						|
								*/
							 | 
						|
								
							 | 
						|
								function assign ($name,$val="") {
							 | 
						|
									if (gettype($name)=="array")
							 | 
						|
										while (list($k,$v)=each($name))
							 | 
						|
											$this->VARS[$k]=$v;
							 | 
						|
									else
							 | 
						|
										$this->VARS[$name]=$val;
							 | 
						|
								}
							 | 
						|
								
							 | 
						|
								/***[ parse ]***************************************************************/
							 | 
						|
								/*
							 | 
						|
									parse a block
							 | 
						|
								*/
							 | 
						|
								
							 | 
						|
								function parse ($bname) {
							 | 
						|
									$copy=$this->blocks[$bname];
							 | 
						|
									if (!isset($this->blocks[$bname]))
							 | 
						|
										$this->set_error ("parse: blockname [$bname] does not exist");
							 | 
						|
									preg_match_all("/\{([A-Za-z0-9\._]+?)}/",$this->blocks[$bname],$var_array);
							 | 
						|
									$var_array=$var_array[1];
							 | 
						|
									while (list($k,$v)=each($var_array)) {
							 | 
						|
										$sub=explode(".",$v);
							 | 
						|
										if ($sub[0]=="_BLOCK_") {
							 | 
						|
											unset($sub[0]);
							 | 
						|
											$bname2=implode(".",$sub);
							 | 
						|
											
							 | 
						|
											if(array_key_exists($bname2, $this->parsed_blocks))
							 | 
						|
											{
							 | 
						|
												$var=$this->parsed_blocks[$bname2];
							 | 
						|
											}
							 | 
						|
											else
							 | 
						|
											{
							 | 
						|
												$var = null;
							 | 
						|
											}
							 | 
						|
												
							 | 
						|
											$nul=(!isset($this->NULL_BLOCK[$bname2])) ? $this->NULL_BLOCK[""] : $this->NULL_BLOCK[$bname2];
							 | 
						|
											$var=(empty($var))?$nul:trim($var);
							 | 
						|
											// Commented out due to regular expression issue with '$' in replacement string.
							 | 
						|
											//$copy=preg_replace("/\{".$v."\}/","$var",$copy);
							 | 
						|
											// This should be faster and work better for '$'
							 | 
						|
											$copy=str_replace("{".$v."}",$var,$copy);
							 | 
						|
										} else {
							 | 
						|
											$var=$this->VARS;
							 | 
						|
								
							 | 
						|
											while(list($k1,$v1)=each($sub))
							 | 
						|
											{
							 | 
						|
												if(is_array($var) && array_key_exists($v1, $var))
							 | 
						|
												{
							 | 
						|
													$var=$var[$v1];
							 | 
						|
												}
							 | 
						|
												else
							 | 
						|
												{
							 | 
						|
													$var = null;
							 | 
						|
												}
							 | 
						|
											}
							 | 
						|
												
							 | 
						|
											$nul=(!isset($this->NULL_STRING[$v])) ? ($this->NULL_STRING[""]) : ($this->NULL_STRING[$v]);
							 | 
						|
											$var=(!isset($var))?$nul:$var;
							 | 
						|
											// Commented out due to regular expression issue with '$' in replacement string.
							 | 
						|
											//$copy=preg_replace("/\{$v\}/","$var",$copy);
							 | 
						|
											// This should be faster and work better for '$'
							 | 
						|
											$copy=str_replace("{".$v."}",$var,$copy);
							 | 
						|
										}
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									if(array_key_exists($bname, $this->parsed_blocks))
							 | 
						|
									{
							 | 
						|
										$this->parsed_blocks[$bname].=$copy;
							 | 
						|
									}
							 | 
						|
									else
							 | 
						|
									{
							 | 
						|
										$this->parsed_blocks[$bname]=$copy;
							 | 
						|
									}
							 | 
						|
									
							 | 
						|
									// reset sub-blocks 
							 | 
						|
									if ($this->AUTORESET && (!empty($this->sub_blocks[$bname]))) {
							 | 
						|
										reset($this->sub_blocks[$bname]);
							 | 
						|
										while (list($k,$v)=each($this->sub_blocks[$bname]))
							 | 
						|
											$this->reset($v);
							 | 
						|
									}
							 | 
						|
								}
							 | 
						|
								
							 | 
						|
								/***[ exists ]**************************************************************/
							 | 
						|
								/*
							 | 
						|
									returns true if a block exists otherwise returns false.
							 | 
						|
								*/
							 | 
						|
								function exists($bname){
							 | 
						|
									return (!empty($this->parsed_blocks[$bname])) || (!empty($this->blocks[$bname]));
							 | 
						|
								}
							 | 
						|
								/***[ rparse ]**************************************************************/
							 | 
						|
								/*
							 | 
						|
									returns the parsed text for a block, including all sub-blocks.
							 | 
						|
								*/
							 | 
						|
								
							 | 
						|
								function rparse($bname) {
							 | 
						|
									if (!empty($this->sub_blocks[$bname])) {
							 | 
						|
										reset($this->sub_blocks[$bname]);
							 | 
						|
										while (list($k,$v)=each($this->sub_blocks[$bname]))
							 | 
						|
											if (!empty($v)) 
							 | 
						|
												$this->rparse($v,$indent."\t");
							 | 
						|
									}
							 | 
						|
									$this->parse($bname);
							 | 
						|
								}
							 | 
						|
								
							 | 
						|
								/***[ insert_loop ]*********************************************************/
							 | 
						|
								/*
							 | 
						|
									inserts a loop ( call assign & parse )
							 | 
						|
								*/
							 | 
						|
								
							 | 
						|
								function insert_loop($bname,$var,$value="") {
							 | 
						|
									$this->assign($var,$value);		
							 | 
						|
									$this->parse($bname);
							 | 
						|
								}
							 | 
						|
								
							 | 
						|
								/***[ text ]****************************************************************/
							 | 
						|
								/*
							 | 
						|
									returns the parsed text for a block
							 | 
						|
								*/
							 | 
						|
								
							 | 
						|
								function text($bname) {
							 | 
						|
									return $this->parsed_blocks[isset($bname) ? $bname :$this->mainblock];
							 | 
						|
								}
							 | 
						|
								
							 | 
						|
								/***[ out ]*****************************************************************/
							 | 
						|
								/*
							 | 
						|
									prints the parsed text
							 | 
						|
								*/
							 | 
						|
								
							 | 
						|
								function out ($bname) {
							 | 
						|
									echo $this->text($bname);
							 | 
						|
								}
							 | 
						|
								
							 | 
						|
								/***[ reset ]***************************************************************/
							 | 
						|
								/*
							 | 
						|
									resets the parsed text
							 | 
						|
								*/
							 | 
						|
								
							 | 
						|
								function reset ($bname) {
							 | 
						|
									$this->parsed_blocks[$bname]="";
							 | 
						|
								}
							 | 
						|
								
							 | 
						|
								/***[ parsed ]**************************************************************/
							 | 
						|
								/*
							 | 
						|
									returns true if block was parsed, false if not
							 | 
						|
								*/
							 | 
						|
								
							 | 
						|
								function parsed ($bname) {
							 | 
						|
									return (!empty($this->parsed_blocks[$bname]));
							 | 
						|
								}
							 | 
						|
								
							 | 
						|
								/***[ SetNullString ]*******************************************************/
							 | 
						|
								/*
							 | 
						|
									sets the string to replace in case the var was not assigned
							 | 
						|
								*/
							 | 
						|
								
							 | 
						|
								function SetNullString($str,$varname="") {
							 | 
						|
									$this->NULL_STRING[$varname]=$str;
							 | 
						|
								}
							 | 
						|
								
							 | 
						|
								/***[ SetNullBlock ]********************************************************/
							 | 
						|
								/*
							 | 
						|
									sets the string to replace in case the block was not parsed
							 | 
						|
								*/
							 | 
						|
								
							 | 
						|
								function SetNullBlock($str,$bname="") {
							 | 
						|
									$this->NULL_BLOCK[$bname]=$str;
							 | 
						|
								}
							 | 
						|
								
							 | 
						|
								/***[ set_autoreset ]*******************************************************/
							 | 
						|
								/*
							 | 
						|
									sets AUTORESET to 1. (default is 1)
							 | 
						|
									if set to 1, parse() automatically resets the parsed blocks' sub blocks
							 | 
						|
									(for multiple level blocks)
							 | 
						|
								*/
							 | 
						|
								
							 | 
						|
								function set_autoreset() {
							 | 
						|
									$this->AUTORESET=1;
							 | 
						|
								}
							 | 
						|
								
							 | 
						|
								/***[ clear_autoreset ]*****************************************************/
							 | 
						|
								/*
							 | 
						|
									sets AUTORESET to 0. (default is 1)
							 | 
						|
									if set to 1, parse() automatically resets the parsed blocks' sub blocks
							 | 
						|
									(for multiple level blocks)
							 | 
						|
								*/
							 | 
						|
								
							 | 
						|
								function clear_autoreset() {
							 | 
						|
									$this->AUTORESET=0;
							 | 
						|
								}
							 | 
						|
								
							 | 
						|
								/***[ scan_globals ]********************************************************/
							 | 
						|
								/*
							 | 
						|
									scans global variables
							 | 
						|
								*/
							 | 
						|
								
							 | 
						|
								function scan_globals() {
							 | 
						|
									reset($GLOBALS);
							 | 
						|
									while (list($k,$v)=each($GLOBALS))
							 | 
						|
										$GLOB[$k]=$v;
							 | 
						|
									$this->assign("PHP",$GLOB);	/* access global variables as {PHP.HTTP_HOST} in your template! */
							 | 
						|
								}
							 | 
						|
								
							 | 
						|
								/******
							 | 
						|
								
							 | 
						|
										WARNING
							 | 
						|
										PUBLIC FUNCTIONS BELOW THIS LINE DIDN'T GET TESTED
							 | 
						|
								
							 | 
						|
								******/
							 | 
						|
								
							 | 
						|
								
							 | 
						|
								/***************************************************************************/
							 | 
						|
								/***[ private stuff ]*******************************************************/
							 | 
						|
								/***************************************************************************/
							 | 
						|
								
							 | 
						|
								/***[ maketree ]************************************************************/
							 | 
						|
								/*
							 | 
						|
									generates the array containing to-be-parsed stuff:
							 | 
						|
								  $blocks["main"],$blocks["main.table"],$blocks["main.table.row"], etc.
							 | 
						|
									also builds the reverse parse order.
							 | 
						|
								*/
							 | 
						|
								
							 | 
						|
								
							 | 
						|
								function maketree($con,$block) {
							 | 
						|
									$con2=explode($this->block_start_delim,$con);
							 | 
						|
									$level=0;
							 | 
						|
									$block_names=array();
							 | 
						|
									$blocks=array();
							 | 
						|
									reset($con2);
							 | 
						|
									while(list($k,$v)=each($con2)) {
							 | 
						|
										$patt="($this->block_start_word|$this->block_end_word)\s*(\w+)\s*$this->block_end_delim(.*)";
							 | 
						|
										if (preg_match_all("/$patt/ims",$v,$res, PREG_SET_ORDER)) {
							 | 
						|
											// $res[0][1] = BEGIN or END
							 | 
						|
											// $res[0][2] = block name
							 | 
						|
											// $res[0][3] = kinda content
							 | 
						|
											if ($res[0][1]==$this->block_start_word) {
							 | 
						|
												$parent_name=implode(".",$block_names);
							 | 
						|
												$block_names[++$level]=$res[0][2];							/* add one level - array("main","table","row")*/
							 | 
						|
												$cur_block_name=implode(".",$block_names);	/* make block name (main.table.row) */
							 | 
						|
												$this->block_parse_order[]=$cur_block_name;	/* build block parsing order (reverse) */
							 | 
						|
								
							 | 
						|
												if(array_key_exists($cur_block_name, $blocks))
							 | 
						|
												{
							 | 
						|
													$blocks[$cur_block_name].=$res[0][3];				/* add contents */
							 | 
						|
												}
							 | 
						|
												else 
							 | 
						|
												{
							 | 
						|
													$blocks[$cur_block_name]=$res[0][3];				/* add contents */
							 | 
						|
												}
							 | 
						|
												
							 | 
						|
												/* add {_BLOCK_.blockname} string to parent block */
							 | 
						|
												if(array_key_exists($parent_name, $blocks))
							 | 
						|
												{
							 | 
						|
													$blocks[$parent_name].="{_BLOCK_.$cur_block_name}";				
							 | 
						|
												}
							 | 
						|
												else 
							 | 
						|
												{
							 | 
						|
													$blocks[$parent_name]="{_BLOCK_.$cur_block_name}";				
							 | 
						|
												}
							 | 
						|
												
							 | 
						|
												$this->sub_blocks[$parent_name][]=$cur_block_name;		/* store sub block names for autoresetting and recursive parsing */
							 | 
						|
												$this->sub_blocks[$cur_block_name][]="";		/* store sub block names for autoresetting */
							 | 
						|
											} else if ($res[0][1]==$this->block_end_word) {
							 | 
						|
												unset($block_names[$level--]);
							 | 
						|
												$parent_name=implode(".",$block_names);
							 | 
						|
												$blocks[$parent_name].=$res[0][3];	/* add rest of block to parent block */
							 | 
						|
								  			}
							 | 
						|
										} else { /* no block delimiters found */
							 | 
						|
											$index = implode(".",$block_names);	
							 | 
						|
											if(array_key_exists($index, $blocks))
							 | 
						|
											{
							 | 
						|
												$blocks[].=$this->block_start_delim.$v;
							 | 
						|
											}
							 | 
						|
											else 
							 | 
						|
											{
							 | 
						|
												$blocks[]=$this->block_start_delim.$v;
							 | 
						|
											}
							 | 
						|
										}
							 | 
						|
									}
							 | 
						|
									return $blocks;	
							 | 
						|
								}
							 | 
						|
								
							 | 
						|
								
							 | 
						|
								
							 | 
						|
								/***[ error stuff ]*********************************************************/
							 | 
						|
								/*
							 | 
						|
									sets and gets error
							 | 
						|
								*/
							 | 
						|
								
							 | 
						|
								function get_error()	{
							 | 
						|
									return ($this->ERROR=="")?0:$this->ERROR;
							 | 
						|
								}
							 | 
						|
								
							 | 
						|
								
							 | 
						|
								function set_error($str)	{
							 | 
						|
									$this->ERROR=$str;
							 | 
						|
								}
							 | 
						|
								
							 | 
						|
								/***[ getfile ]*************************************************************/
							 | 
						|
								/*
							 | 
						|
									returns the contents of a file
							 | 
						|
								*/
							 | 
						|
								
							 | 
						|
								function getfile($file) {
							 | 
						|
									if (!isset($file)) {
							 | 
						|
										$this->set_error("!isset file name!");
							 | 
						|
										return "";
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									// Pick which folder we should include from
							 | 
						|
									// Prefer the local directory, then try the theme directory.
							 | 
						|
									if (!is_file($file)) 
							 | 
						|
										$file = $this->alternate_include_directory.$file;
							 | 
						|
									
							 | 
						|
									if(is_file($file))
							 | 
						|
									{
							 | 
						|
										if (!($fh=fopen($file,"r"))) {
							 | 
						|
											$this->set_error("Cannot open file: $file");
							 | 
						|
											return "";
							 | 
						|
										}
							 | 
						|
								
							 | 
						|
										$file_text=fread($fh,filesize($file));
							 | 
						|
										fclose($fh);
							 | 
						|
									} else { 
							 | 
						|
										$this->set_error("[$file] does not exist");
							 | 
						|
										$file_text="<b>__XTemplate fatal error: file [$file] does not exist__</b>";
							 | 
						|
									}
							 | 
						|
										
							 | 
						|
									return $file_text;
							 | 
						|
								}
							 | 
						|
								
							 | 
						|
								/***[ r_getfile ]***********************************************************/
							 | 
						|
								/*
							 | 
						|
									recursively gets the content of a file with {FILE "filename.tpl"} directives
							 | 
						|
								*/
							 | 
						|
								
							 | 
						|
								
							 | 
						|
								function r_getfile($file) {
							 | 
						|
									$text=$this->getfile($file);
							 | 
						|
									while (preg_match($this->file_delim,$text,$res)) {
							 | 
						|
										$text2=$this->getfile($res[1]);
							 | 
						|
										$text=preg_replace("'".preg_quote($res[0])."'",$text2,$text);
							 | 
						|
									}
							 | 
						|
									return $text;
							 | 
						|
								}
							 | 
						|
								
							 | 
						|
								} /* end of XTemplate class. */
							 | 
						|
								
							 | 
						|
								/*
							 | 
						|
								  	$Log: xtpl.php,v $
							 | 
						|
								  	Revision 1.2  2004/10/06 09:02:02  jack
							 | 
						|
								  	* Modifications made for 3.0 Beta Features
							 | 
						|
								  	
							 | 
						|
								  	Revision 1.8  2004/08/26 00:43:37  sugarmsi
							 | 
						|
								  	added an exists method to check if a block exists in a template
							 | 
						|
								  	
							 | 
						|
								  	Revision 1.7  2004/08/08 09:28:36  sugarjacob
							 | 
						|
								  	Fix: XTemplate changed to use <?php script declarations
							 | 
						|
								  	
							 | 
						|
								  	Revision 1.6  2004/07/31 22:13:22  sugarjacob
							 | 
						|
								  	Removing default setting of template language arrays.
							 | 
						|
								  	
							 | 
						|
								  	Revision 1.5  2004/07/31 21:37:36  sugarjacob
							 | 
						|
								  	Adding code to automatically assign the language strings to every template created.
							 | 
						|
								  	
							 | 
						|
								  	Revision 1.4  2004/07/16 08:21:57  sugarjacob
							 | 
						|
								  	Changing the XTemplate replacement mechanism to allow for '$' in the text being substituted.
							 | 
						|
								  	
							 | 
						|
								  	Revision 1.3  2004/06/11 23:39:47  sugarjacob
							 | 
						|
								  	Fixing issue with a variable not being an array in some cases.
							 | 
						|
								  	
							 | 
						|
								  	Revision 1.2  2004/06/11 23:34:17  sugarjacob
							 | 
						|
								  	Removing errors or notices about invalid indexs.
							 | 
						|
								  	
							 | 
						|
								  	Revision 1.1  2004/05/27 05:30:47  sugarjacob
							 | 
						|
								  	Moving project to SourceForge.
							 | 
						|
								  	
							 | 
						|
								  	Revision 1.1  2004/05/19 01:48:20  sugarcrm
							 | 
						|
								  	Adding files with binary option as appropriate.
							 | 
						|
								  	
							 | 
						|
								  	Revision 1.4  2001/03/26 23:25:02  cranx
							 | 
						|
								  	added keyword expansion to be more clear
							 | 
						|
								  	
							 | 
						|
								  	Revision 1.3  2001/03/26 23:14:56  cranx
							 | 
						|
								  	*** empty log message ***
							 | 
						|
								  	
							 | 
						|
								*/
							 | 
						|
								
							 | 
						|
								?>
							 | 
						|
								
							 |