template.class.php 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345
  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. public static $basedir = __DIR__;
  12. private static $tidy;
  13. private static $tidyConfig = [
  14. 'indent'=>true,
  15. 'tab-size'=>4,
  16. 'wrap'=>0,
  17. 'wrap-asp'=>false,
  18. 'wrap-attributes'=>false,
  19. 'wrap-jste'=>false,
  20. 'wrap-php'=>false,
  21. 'wrap-script-literals'=>false,
  22. 'wrap-sections'=>false,
  23. 'char-encoding'=>'utf8',
  24. 'newline'=>'LF',
  25. 'tidy-mark'=>true,
  26. 'merge-divs'=>false,
  27. 'merge-spans'=>false,
  28. 'logical-emphasis'=>false,
  29. 'literal-attributes'=>true
  30. ];
  31. private static $regex = [
  32. 'match'=>'/\{([^#\/?_][^}\n]*?)\}/i',
  33. 'parentmatch'=>'/\{\.\.\/([^#\/?_][^}\n]*?)\}/i',
  34. 'each'=>'/\{#each ([^}]*)\}([\S\s]*)\{\/each \1\}/i',
  35. 'exist'=>'/\{#exist ([^}]*)\}([\S\s]*)\{\/exist \1\}/i',
  36. 'existelse'=>'/\{#exist ([^}]*)\}([\S\s]*)\{#else \1\}([\S\s]*)\{\/exist \1\}/i',
  37. 'ignore'=>'/\{#ignore\}([\S\s]*)\{\/ignore\}/i',
  38. 'ignored'=>'/\{#ignored (\d+?)\}/i',
  39. 'gettext'=>"/{_([^,}]+)(?:, ?([^},]+))*\}/i",
  40. 'gettext_string'=>'/^([\'"])(.+)\1$/i',
  41. 'echo'=>'/\{=([^}]+)\}/i',
  42. 'eval'=>'/\{\?([\W\w\S\s]+)\?\}/i',
  43. 'include'=>'/{#include ([^}]+)}/i'
  44. ];
  45. protected static $parsers;
  46. private $template;
  47. private $name;
  48. private $path;
  49. public function __construct(string $name, string $template, bool $is_file = false){
  50. if(is_null(static::$parsers)){
  51. static::$parsers = [
  52. 'ignore'=>function(&$output, &$ignored){
  53. $output = preg_replace_callback(static::$regex['ignore'], function($matches) use(&$ignored){
  54. $ignored[] = $matches[1];
  55. return '{#ignored '.(count($ignored) - 1).'}';
  56. }, $output);
  57. },
  58. 'include'=>function(&$output, &$ignored = null){
  59. while(preg_match(static::$regex['include'], $output)){
  60. $output = preg_replace_callback(static::$regex['include'], function($matches) use(&$ignored){
  61. $path = static::$basedir.'/'.$matches[1];
  62. if(file_exists($path)){
  63. $output = file_get_contents($path);
  64. if(!is_null($ignored)){
  65. static::$parsers['ignore']($output, $ignored);
  66. }
  67. return $output;
  68. }
  69. return '';
  70. }, $output);
  71. }
  72. },
  73. 'each'=>[
  74. function(&$output, $data){
  75. $output = preg_replace_callback(static::$regex['each'], function($matches) use($data){
  76. $output = '';
  77. if(isset($data[$matches[1]])){
  78. foreach($data[$matches[1]] as $item){
  79. $output = static::parse($matches[2], $item);
  80. }
  81. }
  82. return $output;
  83. }, $output);
  84. },
  85. function(&$output){
  86. $output = preg_replace_callback(static::$regex['each'], function($matches){
  87. $output = "<?php if(isset(\$data[".var_export($matches[1], true)."])): ";
  88. $output .= "foreach(\$data[".var_export($matches[1], true)."] as \$item): ";
  89. $output .= "\$parent[] = \$data; \$data = \$item; ?>";
  90. $output .= static::compile($matches[2]);
  91. $output .= "<?php \$data = array_pop(\$parent);";
  92. $output .= "endforeach;endif; ?>";
  93. return $output;
  94. }, $output);
  95. }
  96. ],
  97. 'existelse'=>[
  98. function(&$output, $data){
  99. $output = preg_replace_callback(static::$regex['existelse'], function($matches) use($data){
  100. if(isset($data[$matches[1]])){
  101. $output = static::parse($matches[2], $data);
  102. }else{
  103. $output = static::parse($matches[3], $data);
  104. }
  105. return $output;
  106. }, $output);
  107. },
  108. function(&$output){
  109. $output = preg_replace_callback(static::$regex['existelse'], function($matches){
  110. $output = "<?php if(isset(\$data[".var_export($matches[1], true)."])): ?>";
  111. $output .= static::compile($matches[2]);
  112. $output .= "<?php else: ?>";
  113. $output .= static::compile($matches[3]);
  114. $output .= "<?php endif; ?>";
  115. return $output;
  116. }, $output);
  117. }
  118. ],
  119. 'exist'=>[
  120. function(&$output, $data){
  121. $output = preg_replace_callback(static::$regex['exist'], function($matches) use($data){
  122. if(isset($data[$matches[1]])){
  123. return static::parse($data[$matches[2]], $data);
  124. }
  125. return '';
  126. }, $output);
  127. },
  128. function(&$output){
  129. $output = preg_replace_callback(static::$regex['exist'], function($matches){
  130. $output = "<?php if(isset(\$data[".var_export($matches[1], true)."])): ?>";
  131. $output .= static::compile($matches[2]);
  132. $output .= "<?php endif; ?>";
  133. return $output;
  134. }, $output);
  135. }
  136. ],
  137. 'gettext'=>[
  138. function(&$output, $data){
  139. $output = preg_replace_callback(static::$regex['gettext'], function($matches) use($data){
  140. $args = array_map(function($item) use($data){
  141. if(preg_match(static::$regex['gettext_string'], $item)){
  142. return preg_replace(static::$regex['gettext_string'], '\2', $item);
  143. }else{
  144. return $data[$item] ?? '';
  145. }
  146. }, array_slice($matches, 1));
  147. return htmlentities(_(sprintf(...$args)));
  148. }, $output);
  149. },
  150. function(&$output){
  151. $output = preg_replace_callback(static::$regex['gettext'], function($matches){
  152. if(count($matches) > 2){
  153. $output = "<?=htmlentities(sprintf(_({$matches[1]})";
  154. foreach(array_slice($matches, 2) as $item){
  155. if(preg_match(static::$regex['gettext_string'], $item)){
  156. $output .= ", $item";
  157. }else{
  158. $output .= ", (\$data['{$item}'] ?? '')";
  159. }
  160. }
  161. }else{
  162. $output = "<?=htmlentities(_({$matches[1]}";
  163. }
  164. return "{$output})); ?>";
  165. }, $output);
  166. }
  167. ],
  168. 'echo'=>[
  169. function(&$output){
  170. $output = preg_replace_callback(static::$regex['echo'], function($matches){
  171. return eval("return {$matches[1]};");
  172. }, $output);
  173. },
  174. function(&$output){
  175. $output = preg_replace_callback(static::$regex['echo'], function($matches){
  176. return "<?= {$matches[1]}; ?>";
  177. }, $output);
  178. }
  179. ],
  180. 'eval'=>[
  181. function(&$output){
  182. $output = preg_replace_callback(static::$regex['eval'], function($matches){
  183. ob_start();
  184. eval($matches[1]);
  185. $output = ob_get_contents();
  186. ob_end_clean();
  187. return $output;
  188. }, $output);
  189. },
  190. function(&$output){
  191. $output = preg_replace_callback(static::$regex['eval'], function($matches){
  192. return "<?php {$matches[1]}; ?>";
  193. }, $output);
  194. }
  195. ],
  196. 'match'=>[
  197. function(&$output, $data){
  198. $output = preg_replace_callback(static::$regex['match'], function($matches) use($data){
  199. return htmlentities($data[$matches[1]] ?? '');
  200. }, $output);
  201. },
  202. function(&$output){
  203. $output = preg_replace_callback(static::$regex['match'], function($matches){
  204. return "<?=htmlentities(\$data[".var_export($matches[1], true)."] ?? ''); ?>";
  205. }, $output);
  206. }
  207. ],
  208. 'parentmatch'=>[
  209. function(&$output, $data){
  210. throw new \Exception("Not supported in non-compiled templates");
  211. },
  212. function(&$output){
  213. $output = preg_replace_callback(static::$regex['parentmatch'], function($matches){
  214. return "<?=htmlentities(\$parent[count(\$parent)-1][".var_export($matches[1], true)."] ?? ''); ?>";
  215. }, $output);
  216. }
  217. ],
  218. 'ignored'=>function(&$output, $ignored){
  219. $output = preg_replace_callback(static::$regex['ignored'], function($matches) use($ignored){
  220. return htmlentities($ignored[(int)$matches[1]] ?? '');
  221. }, $output);
  222. }
  223. ];
  224. }
  225. if(isset(static::$templates[$name])){
  226. throw new Exception("Template {$name} already exists");
  227. }
  228. if($is_file){
  229. $path = realpath($template);
  230. if(!file_exists($path)){
  231. throw new Exception("Template file {$template} doesn't exist");
  232. }
  233. $template = file_get_contents($path);
  234. }
  235. static::$parsers['include']($template);
  236. $this->template = $template;
  237. $this->name = $name;
  238. $this->path = static::$cachedir."/{$this->name}.".md5($this->template).'.php';
  239. static::$templates[$name] = $this;
  240. }
  241. public function run(array $data) : string{
  242. $data = EArray::from($data);
  243. if($this->fire('before', $data) === false){
  244. throw new Exception("Render on template {$this->name} cancelled. Before.");
  245. }
  246. if(!file_exists($this->path)){
  247. file_put_contents($this->path, static::compile($this->template));
  248. }
  249. try{
  250. $output = static::execute($this->path, $data);
  251. }catch(Exception $e){
  252. file_put_contents($this->path, static::compile($this->template));
  253. $output = static::execute($this->path, $data);
  254. }
  255. if(class_exists('tidy')){
  256. if(is_null(static::$tidy)){
  257. static::$tidy = new \tidy();
  258. }
  259. $tidy = static::$tidy;
  260. $tidy->parseString($output, static::$tidyConfig);
  261. if(!$tidy->cleanRepair()){
  262. throw new \Exception($tidy->errorBuffer);
  263. }
  264. $output = "{$tidy}";
  265. }
  266. if($this->fire('after', $output) === false){
  267. throw new Exception("Render on template {$this->name} cancelled. After");
  268. }
  269. return (string)$output;
  270. }
  271. public static function from(string $name, array $data = []) : string{
  272. $template = static::$templates[$name] ?? null;
  273. if(is_null($template)){
  274. throw new Exception("Template {$name} does not exist");
  275. }
  276. return $template->run($data);
  277. }
  278. public static function compile(string $template) : string{
  279. $ignored = [];
  280. $output = $template;
  281. // Handle {#ignore code}
  282. static::$parsers['ignore']($output, $ignored);
  283. // Handle {#include path/to/file}
  284. static::$parsers['include']($output, $ignored);
  285. // Handle {#each name}{/each}
  286. static::$parsers['each'][1]($output);
  287. // Handle {#exist name}{#else}{/exist}
  288. static::$parsers['existelse'][1]($output);
  289. // Handle {#exist name}{/exist}
  290. static::$parsers['exist'][1]($output);
  291. // Handle {gettext}
  292. static::$parsers['gettext'][1]($output);
  293. // Handle {=expression}
  294. static::$parsers['echo'][1]($output);
  295. // Handle {? expression ?}
  296. static::$parsers['eval'][1]($output);
  297. // Handle {../name}
  298. static::$parsers['parentmatch'][1]($output);
  299. // Handle {name}
  300. static::$parsers['match'][1]($output);
  301. // Handle {#ignored i}
  302. static::$parsers['ignored']($output, $ignored);
  303. return $output;
  304. }
  305. public static function parse(string $template, $data) : string{
  306. if(is_null(static::$parsers)){
  307. // Force parsers to be added
  308. new Template(uuid(), $template);
  309. }
  310. $ignored = [];
  311. $output = $template;
  312. // Handle {#ignore code}
  313. static::$parsers['ignore']($output, $ignored);
  314. // Handle {#include path/to/file}
  315. static::$parsers['include']($output, $ignored);
  316. // Handle {#each name}{/each}
  317. static::$parsers['each'][0]($output, $data);
  318. // Handle {#exist name}{#else}{/exist}
  319. static::$parsers['existelse'][0]($output, $data);
  320. // Handle {#exist name}{/exist}
  321. static::$parsers['exist'][0]($output, $data);
  322. // Handle {gettext}
  323. static::$parsers['gettext'][0]($output, $data);
  324. // Handle {=expression}
  325. static::$parsers['echo'][0]($output);
  326. // Handle {? expression ?}
  327. static::$parsers['eval'][0]($output);
  328. // Handle {../name}
  329. static::$parsers['parentmatch'][1]($output, $data);
  330. // Handle {name}
  331. static::$parsers['match'][0]($output, $data);
  332. // Handle {#ignored i}
  333. static::$parsers['ignored']($output, $ignored);
  334. return $output;
  335. }
  336. public static function execute(string $path, $data) : string{
  337. ob_start();
  338. include($path);
  339. $output = ob_get_contents();
  340. ob_end_clean();
  341. return $output;
  342. }
  343. }
  344. ?>