template.class.php 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207
  1. <?php
  2. namespace Juju\Data;
  3. require_once(realpath(dirname(__DIR__).'/events.trait.php'));
  4. require_once('earray.class.php');
  5. use Juju\{Events, Data\EArray};
  6. use \Exception;
  7. class Template {
  8. use Events;
  9. private static $templates = [];
  10. public static $cachedir;
  11. private static $regex = [
  12. 'match'=>'/\{([^#\/][^}\n]+?)\}/i',
  13. 'each'=>'/\{#each ([^}]*)\}([\S\s]*)\{\/each\}/i',
  14. 'exist'=>'/\{#exist ([^}]*)\}([\S\s]*)\{\/exist\}/i',
  15. 'existelse'=>'/\{#exist ([^}]*)\}([\S\s]*)\{#else\}([\S\s]*)\{\/exist\}/i',
  16. 'ignore'=>'/\{#ignore\}([\S\s]*)\{\/ignore\}/i',
  17. 'ignored'=>'/\{#ignored (\d+?)\}/i',
  18. 'gettext'=>"/{_([^,}]+)(?:, ?([^},]+))*\}/i",
  19. 'gettext_string'=>'/^([\'"])(.+)\1$/i',
  20. 'echo'=>'/\{=([^}]+)\}/i',
  21. 'eval'=>'/\{\?([\W\w\S\s]+)\?\}/i'
  22. ];
  23. private $template;
  24. private $name;
  25. private $path;
  26. public function __construct(string $name, string $template, bool $is_file = false){
  27. if(isset(static::$templates[$name])){
  28. throw new Exception("Template {$name} already exists");
  29. }
  30. if($is_file){
  31. $path = realpath($template);
  32. if(!file_exists($path)){
  33. throw new Exception("Template file {$template} doesn't exist");
  34. }
  35. $template = file_get_contents($path);
  36. }
  37. $this->template = $template;
  38. $this->name = $name;
  39. $this->path = static::$cachedir."/{$this->name}.".md5($this->template).'.php';
  40. static::$templates[$name] = $this;
  41. }
  42. public function run(array $data) : string{
  43. $data = EArray::from($data);
  44. if($this->fire('before', $data) === false){
  45. throw new Exception("Render on template {$this->name} cancelled. Before.");
  46. }
  47. if(!file_exists($this->path)){
  48. file_put_contents($this->path, static::compile($this->template));
  49. }
  50. try{
  51. $output = static::execute($this->path, $data);
  52. }catch(Exception $e){
  53. file_put_contents($this->path, static::compile($this->template));
  54. $output = static::execute($this->path, $data);
  55. }
  56. if($this->fire('after', $output) === false){
  57. throw new Exception("Render on template {$this->name} cancelled. After");
  58. }
  59. return (string)$output;
  60. }
  61. public static function from(string $name, array $data = []) : string{
  62. $template = static::$templates[$name] ?? null;
  63. if(is_null($template)){
  64. throw new Exception("Template {$name} does not exist");
  65. }
  66. return $template->run($data);
  67. }
  68. public static function compile(string $template) : string{
  69. $ignored = [];
  70. // Handle {#ignore code}
  71. $output = preg_replace_callback(static::$regex['ignore'], function($matches) use(&$ignored){
  72. $ignored[] = $matches[1];
  73. return '{#ignored '.(count($ignored) - 1).'}';
  74. }, $template);
  75. // Handle {#each name}{/each}
  76. $output = preg_replace_callback(static::$regex['each'], function($matches){
  77. $output = "<?php if(isset(\$data[".var_export($matches[1], true)."])): ";
  78. $output .= "foreach(\$data[".var_export($matches[1], true)."] as \$item): ";
  79. $output .= "\$parent[] = \$data; \$data = \$item; ?>";
  80. $output .= static::compile($matches[2]);
  81. $output .= "<?php \$data = array_pop(\$parent);";
  82. $output .= "endforeach;endif; ?>";
  83. return $output;
  84. }, $output);
  85. // Handle {#exist name}{#else}{/exist}
  86. $output = preg_replace_callback(static::$regex['existelse'], function($matches){
  87. $output = "<?php if(isset(\$data[".var_export($matches[1], true)."])): ?>";
  88. $output .= static::compile($matches[2]);
  89. $output .= "<php else: ?>";
  90. $output .= static::compile($matches[3]);
  91. $output .= "<?php endif; ?>";
  92. return $output;
  93. }, $output);
  94. // Handle {#exist name}{/exist}
  95. $output = preg_replace_callback(static::$regex['exist'], function($matches){
  96. $output = "<?php if(isset(\$data[".var_export($matches[1], true)."])): ?>";
  97. $output .= static::compile($matches[2]);
  98. $output .= "<?php endif; ?>";
  99. }, $output);
  100. // Handle {gettext}
  101. $output = preg_replace_callback(static::$regex['gettext'], function($matches){
  102. if(count($matches) > 2){
  103. $output = "<?= sprintf(_({$matches[1]})";
  104. foreach(array_slice($matches, 2) as $item){
  105. if(preg_match(static::$regex['gettext_string'], $item)){
  106. $output .= ", $item";
  107. }else{
  108. $output .= ", (\$data['{$item}'] ?? '')";
  109. }
  110. }
  111. }else{
  112. $output = "<?= _({$matches[1]}";
  113. }
  114. return "{$output}); ?>";
  115. }, $output);
  116. // Handle {=expression}
  117. $output = preg_replace_callback(static::$regex['echo'], function($matches){
  118. return "<?= {$matches[1]}; ?>";
  119. }, $output);
  120. // Handle {? expression ?}
  121. $output = preg_replace_callback(static::$regex['eval'], function($matches){
  122. return "<?php {$matches[1]}; ?>";
  123. }, $output);
  124. // Handle {name}
  125. $output = preg_replace_callback(static::$regex['match'], function($matches){
  126. return "<?=(\$data[".var_export($matches[1], true)."] ?? ''); ?>";
  127. }, $output);
  128. // Handle {#ignored i}
  129. return preg_replace_callback(static::$regex['ignored'], function($matches) use(&$ignored){
  130. return htmlentities($ignored[(int)$matches[1]] ?? '');
  131. }, $output);
  132. }
  133. public static function parse(string $template, $data) : string{
  134. $ignored = [];
  135. // Handle {#ignore code}
  136. $output = preg_replace_callback(static::$regex['ignore'], function($matches) use(&$ignored){
  137. $ignored[] = $matches[1];
  138. return '{#ignored '.(count($ignored) - 1).'}';
  139. }, $template);
  140. // Handle {#each name}{/each}
  141. $output = preg_replace_callback(static::$regex['each'], function($matches) use($data){
  142. $output = '';
  143. if(isset($data[$matches[1]])){
  144. foreach($data[$matches[1]] as $item){
  145. $output = static::parse($matches[2], $item);
  146. }
  147. }
  148. return $output;
  149. }, $output);
  150. // Handle {#exist name}{#else}{/exist}
  151. $output = preg_replace_callback(static::$regex['existelse'], function($matches) use($data){
  152. if(isset($data[$matches[1]])){
  153. $output = static::parse($matches[2], $data);
  154. }else{
  155. $output = static::parse($matches[3], $data);
  156. }
  157. return $output;
  158. }, $output);
  159. // Handle {#exist name}{/exist}
  160. $output = preg_replace_callback(static::$regex['exist'], function($matches) use($data){
  161. if(isset($data[$matches[1]])){
  162. return static::parse($data[$matches[2]], $data);
  163. }
  164. return '';
  165. }, $output);
  166. // Handle {gettext}
  167. $output = preg_replace_callback(static::$regex['gettext'], function($matches) use($data){
  168. $args = array_map(function($item) use($data){
  169. if(preg_match(static::$regex['gettext_string'], $item)){
  170. return preg_replace(static::$regex['gettext_string'], '\2', $item);
  171. }else{
  172. return $data[$item] ?? '';
  173. }
  174. }, array_slice($matches, 1));
  175. return _(sprintf(...$args));
  176. }, $output);
  177. // Handle {=expression}
  178. $output = preg_replace_callback(static::$regex['echo'], function($matches) use($data){
  179. return eval("return {$matches[1]};");
  180. }, $output);
  181. // Handle {? expression ?}
  182. $output = preg_replace_callback(static::$regex['eval'], function($matches) use($data){
  183. ob_start();
  184. eval($matches[1]);
  185. $output = ob_get_contents();
  186. ob_end_clean();
  187. return $output;
  188. }, $output);
  189. // Handle {name}
  190. $output = preg_replace_callback(static::$regex['match'], function($matches) use($data){
  191. return $data[$matches[1]] ?? '';
  192. }, $output);
  193. // Handle {#ignored i}
  194. return preg_replace_callback(static::$regex['ignored'], function($matches) use(&$ignored){
  195. return $ignored[(int)$matches[1]] ?? '';
  196. }, $output);
  197. }
  198. public static function execute(string $path, $data) : string{
  199. ob_start();
  200. include($path);
  201. $output = ob_get_contents();
  202. ob_end_clean();
  203. return $output;
  204. }
  205. }
  206. ?>