Files
blog/app/Template/Twig.php
2025-06-20 07:55:37 +00:00

157 lines
4.5 KiB
PHP

<?php
namespace Blog\Template;
use Exception;
/**
* Einfache Template-Engine zur Verarbeitung von Twig-ähnlichen Templates.
*/
class Twig {
/**
* @var array Enthält die erkannten Template-Blöcke.
*/
private array $blocks = [];
/**
* @var string Pfad zum Template-Verzeichnis.
*/
private string $viewsPath;
/**
* @var array Globale Variablen, die in allen Templates verfügbar sind.
*/
private array $globals = [];
/**
* Konstruktor.
*
* @param string $viewsPath Pfad zum Template-Verzeichnis.
*/
public function __construct($viewsPath) {
$this->viewsPath = rtrim($viewsPath, '/') . '/';
}
/**
* Setzt globale Variablen für alle Templates.
*
* @param array $globals Assoziatives Array mit globalen Variablen.
*/
public function setGlobals(array $globals) {
$this->globals = $globals;
}
/**
* Rendert ein Template mit den übergebenen Daten.
*
* @param string $file Name der Template-Datei (ohne Pfad).
* @param array $data Assoziatives Array mit Variablen für das Template.
* @return string Gerenderter HTML-Code.
* @throws Exception Wenn das Template nicht gefunden oder ein Fehler auftritt.
*/
public function render($file, $data = []): string {
$data = array_merge($this->globals, $data);
try {
$code = $this->includeFiles($file);
$code = $this->compileCode($code);
extract($data, EXTR_SKIP);
ob_start();
eval('?>' . $code);
return ob_get_clean();
} catch(Exception $e) {
throw new Exception("Error rendering template: " . $e->getMessage());
}
}
/**
* Wandelt den Template-Code in ausführbaren PHP-Code um.
*
* @param string $code Der Template-Code.
* @return string Kompilierter PHP-Code.
*/
private function compileCode($code): string {
$code = $this->compileBlock($code);
$code = $this->compileYield($code);
$code = $this->compileEchos($code);
$code = $this->compilePHP($code);
return $code;
}
/**
* Fügt eingebundene oder erweiterte Templates ein.
*
* @param string $file Name der Template-Datei.
* @return string Template-Code mit eingebundenen Dateien.
* @throws Exception Wenn die Datei nicht gefunden wird.
*/
private function includeFiles($file): string {
$filePath = $this->viewsPath . preg_replace("/\.twig$/", "", $file) . ".twig";
if(!file_exists($filePath)) {
throw new Exception("View file not found: {$filePath}");
}
$code = file_get_contents($filePath);
preg_match_all('/{% ?(extends|include) ?\'?(.*?)\'? ?%}/i', $code, $matches, PREG_SET_ORDER);
foreach($matches as $match) {
$includedCode = $this->includeFiles($match[2]);
$code = str_replace($match[0], $includedCode, $code);
}
return preg_replace('/{% ?(extends|include) ?\'?(.*?)\'? ?%}/i', '', $code);
}
/**
* Wandelt {% ... %} in PHP-Code um.
*
* @param string $code Der Template-Code.
* @return string PHP-Code.
*/
private function compilePHP($code): string {
return preg_replace('~\{%\s*(.+?)\s*%}~is', '<?php $1 ?>', $code);
}
/**
* Wandelt {{ ... }} in PHP-Echo-Ausgaben um.
*
* @param string $code Der Template-Code.
* @return string PHP-Code mit Echo-Ausgaben.
*/
private function compileEchos($code): string {
$code = preg_replace('~\{\{\{\s*(.+?)\s*\}\}\}~is', '<?=htmlspecialchars($1, ENT_QUOTES, "UTF-8")?>', $code);
return preg_replace('~\{\{\s*(.+?)\s*\}\}~is', '<?=$1?>', $code);
}
/**
* Sammelt und entfernt Block-Definitionen aus dem Template.
*
* @param string $code Der Template-Code.
* @return string Template-Code ohne Block-Definitionen.
*/
private function compileBlock($code): string {
preg_match_all('/{% ?block ?(.*?) ?%}(.*?){% ?endblock ?%}/is', $code, $matches, PREG_SET_ORDER);
foreach($matches as $match) {
if(!isset($this->blocks[$match[1]])) {
$this->blocks[$match[1]] = '';
}
$this->blocks[$match[1]] = str_replace('@parent', $this->blocks[$match[1]], $match[2]);
$code = str_replace($match[0], '', $code);
}
return $code;
}
/**
* Ersetzt yield-Platzhalter durch Block-Inhalte.
*
* @param string $code Der Template-Code.
* @return string Template-Code mit ersetzten Yields.
*/
private function compileYield($code): string {
foreach($this->blocks as $block => $value) {
$code = preg_replace('/{% ?yield ?' . $block . ' ?%}/', $value, $code);
}
return preg_replace('/{% ?yield ?(.*?) ?%}/i', '', $code);
}
}