2013-07-31 04:28:26 -04:00
|
|
|
<?php
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Copyright (c) 2010-2013 Tinyboard Development Group
|
|
|
|
*/
|
2021-02-13 19:24:03 +01:00
|
|
|
require_once 'inc/bootstrap.php';
|
2013-09-06 20:12:04 +10:00
|
|
|
defined('TINYBOARD') or exit;
|
2013-07-31 04:28:26 -04:00
|
|
|
|
|
|
|
$twig = false;
|
|
|
|
|
|
|
|
function load_twig() {
|
|
|
|
global $twig, $config;
|
2023-03-29 18:59:28 -03:00
|
|
|
|
2024-03-31 23:10:07 +02:00
|
|
|
$cache_dir = "{$config['dir']['template']}/cache/";
|
|
|
|
|
2023-03-29 18:59:28 -03:00
|
|
|
$loader = new Twig\Loader\FilesystemLoader($config['dir']['template']);
|
2013-07-31 04:28:26 -04:00
|
|
|
$loader->setPaths($config['dir']['template']);
|
2023-03-29 18:59:28 -03:00
|
|
|
$twig = new Twig\Environment($loader, array(
|
2013-07-31 04:28:26 -04:00
|
|
|
'autoescape' => false,
|
2024-03-31 23:10:07 +02:00
|
|
|
'cache' => is_writable('templates/') || (is_dir($cache_dir) && is_writable($cache_dir)) ?
|
2024-04-26 14:01:32 +02:00
|
|
|
new TinyboardTwigCache($cache_dir) : false,
|
2023-03-29 18:59:28 -03:00
|
|
|
'debug' => $config['debug'],
|
|
|
|
'auto_reload' => $config['twig_auto_reload']
|
2013-07-31 04:28:26 -04:00
|
|
|
));
|
2023-03-29 18:59:28 -03:00
|
|
|
if ($config['debug'])
|
|
|
|
$twig->addExtension(new \Twig\Extension\DebugExtension());
|
|
|
|
$twig->addExtension(new Tinyboard());
|
|
|
|
$twig->addExtension(new PhpMyAdmin\Twig\Extensions\I18nExtension());
|
2013-07-31 04:28:26 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
function Element($templateFile, array $options) {
|
2013-08-29 18:55:25 +10:00
|
|
|
global $config, $debug, $twig, $build_pages;
|
2024-04-07 17:38:00 +02:00
|
|
|
|
2013-07-31 04:28:26 -04:00
|
|
|
if (!$twig)
|
|
|
|
load_twig();
|
2024-04-07 17:38:00 +02:00
|
|
|
|
2013-07-31 04:28:26 -04:00
|
|
|
if (function_exists('create_pm_header') && ((isset($options['mod']) && $options['mod']) || isset($options['__mod'])) && !preg_match('!^mod/!', $templateFile)) {
|
|
|
|
$options['pm'] = create_pm_header();
|
|
|
|
}
|
2024-04-07 17:38:00 +02:00
|
|
|
|
2013-07-31 04:28:26 -04:00
|
|
|
if (isset($options['body']) && $config['debug']) {
|
2013-08-30 09:00:47 +10:00
|
|
|
$_debug = $debug;
|
2024-04-07 17:38:00 +02:00
|
|
|
|
2013-07-31 04:28:26 -04:00
|
|
|
if (isset($debug['start'])) {
|
2013-08-30 09:00:47 +10:00
|
|
|
$_debug['time']['total'] = '~' . round((microtime(true) - $_debug['start']) * 1000, 2) . 'ms';
|
|
|
|
$_debug['time']['init'] = '~' . round(($_debug['start_debug'] - $_debug['start']) * 1000, 2) . 'ms';
|
|
|
|
unset($_debug['start']);
|
|
|
|
unset($_debug['start_debug']);
|
2013-07-31 04:28:26 -04:00
|
|
|
}
|
2013-08-29 18:55:25 +10:00
|
|
|
if ($config['try_smarter'] && isset($build_pages) && !empty($build_pages))
|
2013-08-30 09:00:47 +10:00
|
|
|
$_debug['build_pages'] = $build_pages;
|
|
|
|
$_debug['included'] = get_included_files();
|
|
|
|
$_debug['memory'] = round(memory_get_usage(true) / (1024 * 1024), 2) . ' MiB';
|
|
|
|
$_debug['time']['db_queries'] = '~' . round($_debug['time']['db_queries'] * 1000, 2) . 'ms';
|
|
|
|
$_debug['time']['exec'] = '~' . round($_debug['time']['exec'] * 1000, 2) . 'ms';
|
2013-07-31 04:28:26 -04:00
|
|
|
$options['body'] .=
|
|
|
|
'<h3>Debug</h3><pre style="white-space: pre-wrap;font-size: 10px;">' .
|
2013-08-30 09:00:47 +10:00
|
|
|
str_replace("\n", '<br/>', utf8tohtml(print_r($_debug, true))) .
|
2013-07-31 04:28:26 -04:00
|
|
|
'</pre>';
|
|
|
|
}
|
2024-04-07 17:38:00 +02:00
|
|
|
|
2013-07-31 04:28:26 -04:00
|
|
|
// Read the template file
|
2024-04-03 16:11:54 +02:00
|
|
|
if (@file_get_contents("{$config['dir']['template']}/{$templateFile}")) {
|
2013-07-31 04:28:26 -04:00
|
|
|
$body = $twig->render($templateFile, $options);
|
2024-04-07 17:38:00 +02:00
|
|
|
|
2013-07-31 04:28:26 -04:00
|
|
|
if ($config['minify_html'] && preg_match('/\.html$/', $templateFile)) {
|
|
|
|
$body = trim(preg_replace("/[\t\r\n]/", '', $body));
|
|
|
|
}
|
2024-04-07 17:38:00 +02:00
|
|
|
|
2013-07-31 04:28:26 -04:00
|
|
|
return $body;
|
|
|
|
} else {
|
2024-04-03 16:11:54 +02:00
|
|
|
throw new Exception("Template file '{$templateFile}' does not exist or is empty in '{$config['dir']['template']}'!");
|
2013-07-31 04:28:26 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-04-26 14:01:32 +02:00
|
|
|
class TinyboardTwigCache extends Twig\Cache\FilesystemCache {
|
|
|
|
private string $directory;
|
|
|
|
|
|
|
|
public function __construct(string $directory) {
|
|
|
|
parent::__construct($directory);
|
|
|
|
$this->directory = $directory;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This function was removed in Twig 2.x due to developer views on the Twig library.
|
|
|
|
* Who says we can't keep it for ourselves though?
|
|
|
|
*/
|
|
|
|
public function clear() {
|
|
|
|
$iter = new RecursiveIteratorIterator(
|
|
|
|
new RecursiveDirectoryIterator($this->directory),
|
|
|
|
RecursiveIteratorIterator::LEAVES_ONLY
|
|
|
|
);
|
|
|
|
|
|
|
|
foreach ($iter as $file) {
|
|
|
|
if ($file->isFile()) {
|
|
|
|
@unlink($file->getPathname());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-29 18:59:28 -03:00
|
|
|
class Tinyboard extends Twig\Extension\AbstractExtension
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* Returns a list of filters to add to the existing list.
|
|
|
|
*
|
|
|
|
* @return array An array of filters
|
|
|
|
*/
|
|
|
|
public function getFilters()
|
|
|
|
{
|
|
|
|
return array(
|
|
|
|
new Twig\TwigFilter('filesize', 'format_bytes'),
|
|
|
|
new Twig\TwigFilter('truncate', 'twig_truncate_filter'),
|
|
|
|
new Twig\TwigFilter('truncate_body', 'truncate'),
|
|
|
|
new Twig\TwigFilter('truncate_filename', 'twig_filename_truncate_filter'),
|
|
|
|
new Twig\TwigFilter('extension', 'twig_extension_filter'),
|
|
|
|
new Twig\TwigFilter('sprintf', 'sprintf'),
|
|
|
|
new Twig\TwigFilter('capcode', 'capcode'),
|
|
|
|
new Twig\TwigFilter('remove_modifiers', 'remove_modifiers'),
|
|
|
|
new Twig\TwigFilter('hasPermission', 'twig_hasPermission_filter'),
|
|
|
|
new Twig\TwigFilter('date', 'twig_date_filter'),
|
|
|
|
new Twig\TwigFilter('poster_id', 'poster_id'),
|
|
|
|
new Twig\TwigFilter('count', 'count'),
|
2024-04-30 12:32:40 +02:00
|
|
|
new Twig\TwigFilter('ago', 'Vichan\Functions\Format\ago'),
|
|
|
|
new Twig\TwigFilter('until', 'Vichan\Functions\Format\until'),
|
2023-03-29 18:59:28 -03:00
|
|
|
new Twig\TwigFilter('push', 'twig_push_filter'),
|
|
|
|
new Twig\TwigFilter('bidi_cleanup', 'bidi_cleanup'),
|
|
|
|
new Twig\TwigFilter('addslashes', 'addslashes'),
|
|
|
|
new Twig\TwigFilter('cloak_ip', 'cloak_ip'),
|
|
|
|
new Twig\TwigFilter('cloak_mask', 'cloak_mask'),
|
|
|
|
);
|
|
|
|
}
|
2024-04-07 17:38:00 +02:00
|
|
|
|
2023-03-29 18:59:28 -03:00
|
|
|
/**
|
|
|
|
* Returns a list of functions to add to the existing list.
|
|
|
|
*
|
|
|
|
* @return array An array of filters
|
|
|
|
*/
|
|
|
|
public function getFunctions()
|
|
|
|
{
|
|
|
|
return array(
|
|
|
|
new Twig\TwigFunction('time', 'time'),
|
|
|
|
new Twig\TwigFunction('floor', 'floor'),
|
|
|
|
new Twig\TwigFunction('hiddenInputs', 'hiddenInputs'),
|
|
|
|
new Twig\TwigFunction('hiddenInputsHash', 'hiddenInputsHash'),
|
|
|
|
new Twig\TwigFunction('ratio', 'twig_ratio_function'),
|
|
|
|
new Twig\TwigFunction('secure_link_confirm', 'twig_secure_link_confirm'),
|
|
|
|
new Twig\TwigFunction('secure_link', 'twig_secure_link'),
|
|
|
|
new Twig\TwigFunction('link_for', 'link_for')
|
|
|
|
);
|
|
|
|
}
|
2024-04-07 17:38:00 +02:00
|
|
|
|
2023-03-29 18:59:28 -03:00
|
|
|
/**
|
|
|
|
* Returns the name of the extension.
|
|
|
|
*
|
|
|
|
* @return string The extension name
|
|
|
|
*/
|
|
|
|
public function getName()
|
|
|
|
{
|
|
|
|
return 'tinyboard';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function twig_push_filter($array, $value) {
|
|
|
|
array_push($array, $value);
|
|
|
|
return $array;
|
|
|
|
}
|
|
|
|
|
|
|
|
function twig_date_filter($date, $format) {
|
2024-04-03 17:14:49 +02:00
|
|
|
$date = new DateTime($date, new DateTimeZone('UTC'));
|
|
|
|
return $date->format($format);
|
2023-03-29 18:59:28 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
function twig_hasPermission_filter($mod, $permission, $board = null) {
|
|
|
|
return hasPermission($permission, $board, $mod);
|
|
|
|
}
|
|
|
|
|
|
|
|
function twig_extension_filter($value, $case_insensitive = true) {
|
|
|
|
$ext = mb_substr($value, mb_strrpos($value, '.') + 1);
|
|
|
|
if($case_insensitive)
|
2024-04-07 17:38:00 +02:00
|
|
|
$ext = mb_strtolower($ext);
|
2023-03-29 18:59:28 -03:00
|
|
|
return $ext;
|
|
|
|
}
|
|
|
|
|
|
|
|
function twig_sprintf_filter( $value, $var) {
|
|
|
|
return sprintf($value, $var);
|
|
|
|
}
|
|
|
|
|
|
|
|
function twig_truncate_filter($value, $length = 30, $preserve = false, $separator = '…') {
|
|
|
|
if (mb_strlen($value) > $length) {
|
|
|
|
if ($preserve) {
|
|
|
|
if (false !== ($breakpoint = mb_strpos($value, ' ', $length))) {
|
|
|
|
$length = $breakpoint;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return mb_substr($value, 0, $length) . $separator;
|
|
|
|
}
|
|
|
|
return $value;
|
|
|
|
}
|
|
|
|
|
|
|
|
function twig_filename_truncate_filter($value, $length = 30, $separator = '…') {
|
|
|
|
if (mb_strlen($value) > $length) {
|
|
|
|
$value = strrev($value);
|
|
|
|
$array = array_reverse(explode(".", $value, 2));
|
|
|
|
$array = array_map("strrev", $array);
|
2024-04-07 17:38:00 +02:00
|
|
|
|
2023-03-29 18:59:28 -03:00
|
|
|
$filename = &$array[0];
|
|
|
|
$extension = isset($array[1]) ? $array[1] : false;
|
|
|
|
|
|
|
|
$filename = mb_substr($filename, 0, $length - ($extension ? mb_strlen($extension) + 1 : 0)) . $separator;
|
|
|
|
|
|
|
|
return implode(".", $array);
|
|
|
|
}
|
|
|
|
return $value;
|
|
|
|
}
|
|
|
|
|
|
|
|
function twig_ratio_function($w, $h) {
|
|
|
|
return fraction($w, $h, ':');
|
|
|
|
}
|
|
|
|
function twig_secure_link_confirm($text, $title, $confirm_message, $href) {
|
|
|
|
global $config;
|
|
|
|
|
|
|
|
return '<a onclick="if (event.which==2) return true;if (confirm(\'' . htmlentities(addslashes($confirm_message)) . '\')) document.location=\'?/' . htmlspecialchars(addslashes($href . '/' . make_secure_link_token($href))) . '\';return false;" title="' . htmlentities($title) . '" href="?/' . $href . '">' . $text . '</a>';
|
|
|
|
}
|
|
|
|
function twig_secure_link($href) {
|
|
|
|
return $href . '/' . make_secure_link_token($href);
|
|
|
|
}
|