2016-05-10 23:48:51 +02:00
|
|
|
<?php
|
|
|
|
|
2018-12-03 00:46:04 +01:00
|
|
|
namespace Shaarli\Render;
|
|
|
|
|
|
|
|
use Exception;
|
|
|
|
use RainTPL;
|
2019-01-12 23:55:38 +01:00
|
|
|
use Shaarli\ApplicationUtils;
|
2019-05-25 15:52:27 +02:00
|
|
|
use Shaarli\Bookmark\BookmarkServiceInterface;
|
2017-03-22 19:16:35 +01:00
|
|
|
use Shaarli\Config\ConfigManager;
|
2020-05-27 13:35:48 +02:00
|
|
|
use Shaarli\Security\SessionManager;
|
2018-07-05 20:29:55 +02:00
|
|
|
use Shaarli\Thumbnailer;
|
2017-03-22 19:16:35 +01:00
|
|
|
|
2016-05-10 23:48:51 +02:00
|
|
|
/**
|
|
|
|
* This class is in charge of building the final page.
|
|
|
|
* (This is basically a wrapper around RainTPL which pre-fills some fields.)
|
|
|
|
* $p = new PageBuilder();
|
|
|
|
* $p->assign('myfield','myvalue');
|
|
|
|
* $p->renderPage('mytemplate');
|
|
|
|
*/
|
|
|
|
class PageBuilder
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* @var RainTPL RainTPL instance.
|
|
|
|
*/
|
|
|
|
private $tpl;
|
|
|
|
|
2016-06-09 20:04:02 +02:00
|
|
|
/**
|
|
|
|
* @var ConfigManager $conf Configuration Manager instance.
|
|
|
|
*/
|
|
|
|
protected $conf;
|
|
|
|
|
2018-06-08 12:50:49 +02:00
|
|
|
/**
|
|
|
|
* @var array $_SESSION
|
|
|
|
*/
|
|
|
|
protected $session;
|
|
|
|
|
2017-05-07 18:19:09 +02:00
|
|
|
/**
|
2019-05-25 15:52:27 +02:00
|
|
|
* @var BookmarkServiceInterface $bookmarkService instance.
|
2017-05-07 18:19:09 +02:00
|
|
|
*/
|
2019-05-25 15:52:27 +02:00
|
|
|
protected $bookmarkService;
|
2018-06-08 12:50:49 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @var null|string XSRF token
|
|
|
|
*/
|
|
|
|
protected $token;
|
|
|
|
|
2018-12-03 00:46:04 +01:00
|
|
|
/**
|
|
|
|
* @var bool $isLoggedIn Whether the user is logged in
|
|
|
|
*/
|
2018-04-18 23:45:05 +02:00
|
|
|
protected $isLoggedIn = false;
|
2017-05-07 18:19:09 +02:00
|
|
|
|
2016-05-10 23:48:51 +02:00
|
|
|
/**
|
|
|
|
* PageBuilder constructor.
|
|
|
|
* $tpl is initialized at false for lazy loading.
|
2016-06-09 20:04:02 +02:00
|
|
|
*
|
2019-05-25 15:52:27 +02:00
|
|
|
* @param ConfigManager $conf Configuration Manager instance (reference).
|
|
|
|
* @param array $session $_SESSION array
|
|
|
|
* @param BookmarkServiceInterface $linkDB instance.
|
|
|
|
* @param string $token Session token
|
|
|
|
* @param bool $isLoggedIn
|
2016-05-10 23:48:51 +02:00
|
|
|
*/
|
2018-06-08 12:50:49 +02:00
|
|
|
public function __construct(&$conf, $session, $linkDB = null, $token = null, $isLoggedIn = false)
|
2016-05-10 23:48:51 +02:00
|
|
|
{
|
|
|
|
$this->tpl = false;
|
2016-06-09 20:04:02 +02:00
|
|
|
$this->conf = $conf;
|
2018-06-08 12:50:49 +02:00
|
|
|
$this->session = $session;
|
2019-05-25 15:52:27 +02:00
|
|
|
$this->bookmarkService = $linkDB;
|
2017-10-22 18:44:46 +02:00
|
|
|
$this->token = $token;
|
2018-04-18 23:45:05 +02:00
|
|
|
$this->isLoggedIn = $isLoggedIn;
|
2016-05-10 23:48:51 +02:00
|
|
|
}
|
|
|
|
|
2020-07-06 08:04:35 +02:00
|
|
|
/**
|
|
|
|
* Reset current state of template rendering.
|
|
|
|
* Mostly useful for error handling. We remove everything, and display the error template.
|
|
|
|
*/
|
|
|
|
public function reset(): void
|
|
|
|
{
|
|
|
|
$this->tpl = false;
|
|
|
|
}
|
|
|
|
|
2016-05-10 23:48:51 +02:00
|
|
|
/**
|
|
|
|
* Initialize all default tpl tags.
|
|
|
|
*/
|
|
|
|
private function initialize()
|
|
|
|
{
|
|
|
|
$this->tpl = new RainTPL();
|
|
|
|
|
|
|
|
try {
|
|
|
|
$version = ApplicationUtils::checkUpdate(
|
2017-10-01 11:09:12 +02:00
|
|
|
SHAARLI_VERSION,
|
2016-06-11 09:08:02 +02:00
|
|
|
$this->conf->get('resource.update_check'),
|
|
|
|
$this->conf->get('updates.check_updates_interval'),
|
|
|
|
$this->conf->get('updates.check_updates'),
|
2018-04-18 23:45:05 +02:00
|
|
|
$this->isLoggedIn,
|
2016-06-11 09:08:02 +02:00
|
|
|
$this->conf->get('updates.check_updates_branch')
|
2016-05-10 23:48:51 +02:00
|
|
|
);
|
|
|
|
$this->tpl->assign('newVersion', escape($version));
|
|
|
|
$this->tpl->assign('versionError', '');
|
|
|
|
} catch (Exception $exc) {
|
2016-06-11 09:08:02 +02:00
|
|
|
logm($this->conf->get('resource.log'), $_SERVER['REMOTE_ADDR'], $exc->getMessage());
|
2016-05-10 23:48:51 +02:00
|
|
|
$this->tpl->assign('newVersion', '');
|
|
|
|
$this->tpl->assign('versionError', escape($exc->getMessage()));
|
|
|
|
}
|
|
|
|
|
2018-04-18 23:45:05 +02:00
|
|
|
$this->tpl->assign('is_logged_in', $this->isLoggedIn);
|
2016-05-10 23:48:51 +02:00
|
|
|
$this->tpl->assign('feedurl', escape(index_url($_SERVER)));
|
|
|
|
$searchcrits = ''; // Search criteria
|
|
|
|
if (!empty($_GET['searchtags'])) {
|
|
|
|
$searchcrits .= '&searchtags=' . urlencode($_GET['searchtags']);
|
|
|
|
}
|
|
|
|
if (!empty($_GET['searchterm'])) {
|
|
|
|
$searchcrits .= '&searchterm=' . urlencode($_GET['searchterm']);
|
|
|
|
}
|
|
|
|
$this->tpl->assign('searchcrits', $searchcrits);
|
|
|
|
$this->tpl->assign('source', index_url($_SERVER));
|
2017-10-01 11:09:12 +02:00
|
|
|
$this->tpl->assign('version', SHAARLI_VERSION);
|
2017-10-01 11:02:48 +02:00
|
|
|
$this->tpl->assign(
|
|
|
|
'version_hash',
|
|
|
|
ApplicationUtils::getVersionHash(SHAARLI_VERSION, $this->conf->get('credentials.salt'))
|
|
|
|
);
|
2018-08-13 10:42:27 +02:00
|
|
|
$this->tpl->assign('index_url', index_url($_SERVER));
|
2018-12-03 00:46:04 +01:00
|
|
|
$visibility = !empty($_SESSION['visibility']) ? $_SESSION['visibility'] : '';
|
2017-12-16 12:36:59 +01:00
|
|
|
$this->tpl->assign('visibility', $visibility);
|
2017-06-01 17:55:26 +02:00
|
|
|
$this->tpl->assign('untaggedonly', !empty($_SESSION['untaggedonly']));
|
2016-07-19 18:03:09 +02:00
|
|
|
$this->tpl->assign('pagetitle', $this->conf->get('general.title', 'Shaarli'));
|
2016-06-09 20:04:02 +02:00
|
|
|
if ($this->conf->exists('general.header_link')) {
|
|
|
|
$this->tpl->assign('titleLink', $this->conf->get('general.header_link'));
|
2016-05-10 23:48:51 +02:00
|
|
|
}
|
2016-07-19 18:03:09 +02:00
|
|
|
$this->tpl->assign('shaarlititle', $this->conf->get('general.title', 'Shaarli'));
|
2016-06-11 09:08:02 +02:00
|
|
|
$this->tpl->assign('openshaarli', $this->conf->get('security.open_shaarli', false));
|
2017-03-11 14:11:06 +01:00
|
|
|
$this->tpl->assign('showatom', $this->conf->get('feed.show_atom', true));
|
|
|
|
$this->tpl->assign('feed_type', $this->conf->get('feed.show_atom', true) !== false ? 'atom' : 'rss');
|
2016-06-11 09:08:02 +02:00
|
|
|
$this->tpl->assign('hide_timestamps', $this->conf->get('privacy.hide_timestamps', false));
|
2017-10-22 18:44:46 +02:00
|
|
|
$this->tpl->assign('token', $this->token);
|
2017-10-01 11:02:48 +02:00
|
|
|
|
2019-02-09 14:29:35 +01:00
|
|
|
$this->tpl->assign('language', $this->conf->get('translation.language'));
|
|
|
|
|
2019-05-25 15:52:27 +02:00
|
|
|
if ($this->bookmarkService !== null) {
|
|
|
|
$this->tpl->assign('tags', $this->bookmarkService->bookmarksCountPerTag());
|
2017-05-07 18:19:09 +02:00
|
|
|
}
|
2016-11-09 18:57:02 +01:00
|
|
|
|
2018-07-05 20:29:55 +02:00
|
|
|
$this->tpl->assign(
|
|
|
|
'thumbnails_enabled',
|
|
|
|
$this->conf->get('thumbnails.mode', Thumbnailer::MODE_NONE) !== Thumbnailer::MODE_NONE
|
|
|
|
);
|
2016-11-09 18:57:02 +01:00
|
|
|
$this->tpl->assign('thumbnails_width', $this->conf->get('thumbnails.width'));
|
|
|
|
$this->tpl->assign('thumbnails_height', $this->conf->get('thumbnails.height'));
|
|
|
|
|
2019-05-25 15:52:27 +02:00
|
|
|
$this->tpl->assign('formatter', $this->conf->get('formatter', 'default'));
|
|
|
|
|
2016-06-15 18:22:19 +02:00
|
|
|
// To be removed with a proper theme configuration.
|
|
|
|
$this->tpl->assign('conf', $this->conf);
|
2016-05-10 23:48:51 +02:00
|
|
|
}
|
|
|
|
|
2020-05-30 14:00:06 +02:00
|
|
|
/**
|
|
|
|
* Affect variable after controller processing.
|
|
|
|
* Used for alert messages.
|
|
|
|
*/
|
2020-05-27 13:35:48 +02:00
|
|
|
protected function finalize(): void
|
|
|
|
{
|
2020-06-13 11:22:14 +02:00
|
|
|
//FIXME - DEV _ REMOVE ME
|
|
|
|
$this->assign('base_path', '/Shaarli');
|
|
|
|
$this->assign('asset_path', '/Shaarli/tpl/default');
|
|
|
|
|
2020-05-27 13:35:48 +02:00
|
|
|
// TODO: use the SessionManager
|
|
|
|
$messageKeys = [
|
|
|
|
SessionManager::KEY_SUCCESS_MESSAGES,
|
|
|
|
SessionManager::KEY_WARNING_MESSAGES,
|
|
|
|
SessionManager::KEY_ERROR_MESSAGES
|
|
|
|
];
|
|
|
|
foreach ($messageKeys as $messageKey) {
|
|
|
|
if (!empty($_SESSION[$messageKey])) {
|
|
|
|
$this->tpl->assign('global_' . $messageKey, $_SESSION[$messageKey]);
|
|
|
|
unset($_SESSION[$messageKey]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-10 23:48:51 +02:00
|
|
|
/**
|
|
|
|
* The following assign() method is basically the same as RainTPL (except lazy loading)
|
|
|
|
*
|
|
|
|
* @param string $placeholder Template placeholder.
|
|
|
|
* @param mixed $value Value to assign.
|
|
|
|
*/
|
|
|
|
public function assign($placeholder, $value)
|
|
|
|
{
|
|
|
|
if ($this->tpl === false) {
|
|
|
|
$this->initialize();
|
|
|
|
}
|
|
|
|
$this->tpl->assign($placeholder, $value);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Assign an array of data to the template builder.
|
|
|
|
*
|
|
|
|
* @param array $data Data to assign.
|
|
|
|
*
|
|
|
|
* @return false if invalid data.
|
|
|
|
*/
|
|
|
|
public function assignAll($data)
|
|
|
|
{
|
|
|
|
if ($this->tpl === false) {
|
|
|
|
$this->initialize();
|
|
|
|
}
|
|
|
|
|
2018-10-13 00:19:03 +02:00
|
|
|
if (empty($data) || !is_array($data)) {
|
2016-05-10 23:48:51 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach ($data as $key => $value) {
|
|
|
|
$this->assign($key, $value);
|
|
|
|
}
|
2016-06-09 20:04:02 +02:00
|
|
|
return true;
|
2016-05-10 23:48:51 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Render a specific page (using a template file).
|
|
|
|
* e.g. $pb->renderPage('picwall');
|
|
|
|
*
|
|
|
|
* @param string $page Template filename (without extension).
|
|
|
|
*/
|
|
|
|
public function renderPage($page)
|
|
|
|
{
|
2016-06-09 20:04:02 +02:00
|
|
|
if ($this->tpl === false) {
|
2016-05-10 23:48:51 +02:00
|
|
|
$this->initialize();
|
|
|
|
}
|
2016-06-09 20:04:02 +02:00
|
|
|
|
2020-05-27 13:35:48 +02:00
|
|
|
$this->finalize();
|
|
|
|
|
2016-05-10 23:48:51 +02:00
|
|
|
$this->tpl->draw($page);
|
|
|
|
}
|
|
|
|
|
2020-01-18 17:50:11 +01:00
|
|
|
/**
|
|
|
|
* Render a specific page as string (using a template file).
|
|
|
|
* e.g. $pb->render('picwall');
|
|
|
|
*
|
|
|
|
* @param string $page Template filename (without extension).
|
|
|
|
*
|
|
|
|
* @return string Processed template content
|
|
|
|
*/
|
|
|
|
public function render(string $page): string
|
|
|
|
{
|
|
|
|
if ($this->tpl === false) {
|
|
|
|
$this->initialize();
|
|
|
|
}
|
|
|
|
|
2020-05-27 13:35:48 +02:00
|
|
|
$this->finalize();
|
|
|
|
|
2020-01-18 17:50:11 +01:00
|
|
|
return $this->tpl->draw($page, true);
|
|
|
|
}
|
|
|
|
|
2016-05-10 23:48:51 +02:00
|
|
|
/**
|
|
|
|
* Render a 404 page (uses the template : tpl/404.tpl)
|
2018-12-03 00:46:04 +01:00
|
|
|
* usage: $PAGE->render404('The link was deleted')
|
2016-05-10 23:48:51 +02:00
|
|
|
*
|
2018-12-03 00:46:04 +01:00
|
|
|
* @param string $message A message to display what is not found
|
2016-05-10 23:48:51 +02:00
|
|
|
*/
|
2017-05-09 18:12:15 +02:00
|
|
|
public function render404($message = '')
|
2016-05-10 23:48:51 +02:00
|
|
|
{
|
2017-05-09 18:12:15 +02:00
|
|
|
if (empty($message)) {
|
|
|
|
$message = t('The page you are trying to reach does not exist or has been deleted.');
|
|
|
|
}
|
2018-12-03 00:46:04 +01:00
|
|
|
header($_SERVER['SERVER_PROTOCOL'] . ' ' . t('404 Not Found'));
|
2016-05-10 23:48:51 +02:00
|
|
|
$this->tpl->assign('error_message', $message);
|
|
|
|
$this->renderPage('404');
|
|
|
|
}
|
|
|
|
}
|