koblog/bl-kernel/pagex.class.php

625 lines
14 KiB
PHP
Raw Permalink Normal View History

<?php defined('BLUDIT') or die('Bludit CMS.');
2024-06-23 21:46:43 +02:00
class Page
{
protected $vars;
function __construct($key)
{
2018-08-03 18:59:23 +02:00
global $pages;
$this->vars['key'] = $key;
// If key is FALSE, the page is create with default values, like an empty page
// Useful for Page Not Found
2024-06-23 21:46:43 +02:00
if ($key === false) {
2018-08-03 18:59:23 +02:00
$row = $pages->getDefaultFields();
} else {
2018-08-03 18:59:23 +02:00
if (Text::isEmpty($key) || !$pages->exists($key)) {
2024-06-23 21:46:43 +02:00
$errorMessage = 'Page not found in database by key [' . $key . ']';
Log::set(__METHOD__ . LOG_SEP . $errorMessage);
throw new Exception($errorMessage);
}
2018-08-03 18:59:23 +02:00
$row = $pages->getPageDB($key);
}
2024-06-23 21:46:43 +02:00
foreach ($row as $field => $value) {
if ($field == 'date') {
$this->setField('dateRaw', $value);
} else {
$this->setField($field, $value);
}
}
}
public function getValue($field)
{
if (isset($this->vars[$field])) {
return $this->vars[$field];
}
return false;
}
public function setField($field, $value)
{
$this->vars[$field] = $value;
return true;
}
// Returns the raw content
// This content is not markdown parser
// (boolean) $sanitize, TRUE returns the content sanitized
2024-06-23 21:46:43 +02:00
public function contentRaw($sanitize = false)
{
$key = $this->key();
2024-06-23 21:46:43 +02:00
$filePath = PATH_PAGES . $key . DS . FILENAME;
$contentRaw = file_get_contents($filePath);
if ($sanitize) {
return Sanitize::html($contentRaw);
}
return $contentRaw;
}
2019-01-04 14:19:51 +01:00
// Returns the full content
// This content is markdown parser
2019-01-04 14:19:51 +01:00
// (boolean) $sanitize, TRUE returns the content sanitized
2024-06-23 21:46:43 +02:00
public function content($sanitize = false)
{
// If already set the content, return it
2018-09-05 22:55:14 +02:00
$content = $this->getValue('content');
if (!empty($content)) {
return $content;
}
2019-01-04 14:19:51 +01:00
// Get the raw content
2018-10-23 23:50:56 +02:00
$content = $this->contentRaw();
// Parse Markdown
if (MARKDOWN_PARSER) {
$parsedown = new Parsedown();
$content = $parsedown->text($content);
}
// Parse img src relative to absolute (with domain)
if (IMAGE_RELATIVE_TO_ABSOLUTE) {
2024-06-23 21:46:43 +02:00
$domain = IMAGE_RESTRICT ? DOMAIN_UPLOADS_PAGES . $this->uuid() . '/' : DOMAIN_UPLOADS;
$content = Text::imgRel2Abs($content, $domain);
}
if ($sanitize) {
return Sanitize::html($content);
}
return $content;
}
// Returns the first part of the content if the content is splited, otherwise is returned the full content
2019-01-04 14:19:51 +01:00
// This content is markdown parser
// (boolean) $sanitize, TRUE returns the content sanitized
2024-06-23 21:46:43 +02:00
public function contentBreak($sanitize = false)
{
2019-01-04 14:19:51 +01:00
$content = $this->content($sanitize);
$explode = explode(PAGE_BREAK, $content);
return $explode[0];
}
// Returns the date according to locale settings and the format defined in the system
2024-06-23 21:46:43 +02:00
public function date($format = false)
{
$dateRaw = $this->dateRaw();
2024-06-23 21:46:43 +02:00
if ($format === false) {
global $site;
$format = $site->dateFormat();
}
return Date::format($dateRaw, DB_DATE_FORMAT, $format);
}
// Returns the date according to locale settings and format as database stored
public function dateRaw()
{
// This field is set in the constructor
return $this->getValue('dateRaw');
}
// Returns the date according to locale settings and format settings
2024-06-23 21:46:43 +02:00
public function dateModified($format = false)
{
2019-11-18 20:18:29 +01:00
$dateRaw = $this->getValue('dateModified');
2024-06-23 21:46:43 +02:00
if ($format === false) {
2019-11-18 20:18:29 +01:00
global $site;
$format = $site->dateFormat();
}
return Date::format($dateRaw, DB_DATE_FORMAT, $format);
}
// Returns the username who created the page
public function username()
{
return $this->getValue('username');
}
// TODO: Check if necessary this function
public function getDB()
{
return $this->vars;
}
// Returns the permalink
// (boolean) $absolute, TRUE returns the page link with the DOMAIN, FALSE without the DOMAIN
2024-06-23 21:46:43 +02:00
public function permalink($absolute = true)
{
// Get the key of the page
$key = $this->key();
2024-06-23 21:46:43 +02:00
if ($absolute) {
return DOMAIN_PAGES . $key;
}
2024-06-23 21:46:43 +02:00
return HTML_PATH_ROOT . PAGE_URI_FILTER . $key;
}
// Returns the previous page key
public function previousKey()
{
2018-08-03 18:59:23 +02:00
global $pages;
return $pages->previousPageKey($this->key());
}
// Returns the next page key
public function nextKey()
{
2018-08-03 18:59:23 +02:00
global $pages;
return $pages->nextPageKey($this->key());
}
// Returns the category name
public function category()
{
return $this->categoryMap('name');
}
2018-08-06 21:46:58 +02:00
// Returns the category template
public function categoryTemplate()
{
return $this->categoryMap('template');
}
2018-08-06 21:46:58 +02:00
// Returns the category description
public function categoryDescription()
{
return $this->categoryMap('description');
}
// Returns the category key
public function categoryKey()
{
return $this->getValue('category');
}
// Returns the category permalink
public function categoryPermalink()
{
2024-06-23 21:46:43 +02:00
return DOMAIN_CATEGORIES . $this->categoryKey();
}
// Returns the field from the array
// categoryMap = array( 'name'=>'', 'list'=>array() )
public function categoryMap($field)
{
2018-08-02 22:33:53 +02:00
global $categories;
$categoryKey = $this->categoryKey();
2018-08-02 22:33:53 +02:00
$map = $categories->getMap($categoryKey);
2024-06-23 21:46:43 +02:00
if ($field == 'key') {
return $this->categoryKey();
2024-06-23 21:46:43 +02:00
} elseif (isset($map[$field])) {
2018-08-06 21:46:58 +02:00
return $map[$field];
}
return false;
}
2018-07-28 18:33:37 +02:00
// Returns the user object or passing the method returns the object User method
2024-06-23 21:46:43 +02:00
public function user($method = false)
{
$username = $this->username();
2018-07-28 18:33:37 +02:00
try {
$user = new User($username);
if ($method) {
return $user->{$method}();
}
return $user;
} catch (Exception $e) {
return false;
}
}
public function template()
{
return $this->getValue('template');
}
// Returns the description field
public function description()
{
return $this->getValue('description');
}
// Returns the tags separated by comma
2018-08-01 14:46:12 +02:00
// (boolean) $returnsArray, TRUE to get the tags as an array, FALSE to get the tags separated by comma
2018-07-28 18:33:37 +02:00
// The tags in array format returns array( tagKey => tagName )
2024-06-23 21:46:43 +02:00
public function tags($returnsArray = false)
{
$tags = $this->getValue('tags');
2018-07-28 18:33:37 +02:00
if ($returnsArray) {
if (empty($tags)) {
return array();
}
return $tags;
}
2018-07-28 18:33:37 +02:00
if (empty($tags)) {
return '';
}
2018-08-01 14:46:12 +02:00
// Return string with tags separated by comma.
2019-02-26 00:47:34 +01:00
return implode(',', $tags);
}
2020-07-23 12:14:43 +02:00
2024-06-23 21:46:43 +02:00
public function json($returnsArray = false)
{
$tmp['key'] = $this->key();
$tmp['title'] = $this->title();
$tmp['content'] = $this->content(); // Markdown parsed
$tmp['contentRaw'] = $this->contentRaw(true); // No Markdown parsed
$tmp['description'] = $this->description();
2019-02-19 08:38:17 +01:00
$tmp['type'] = $this->type();
2019-02-25 16:32:29 +01:00
$tmp['slug'] = $this->slug();
2019-06-16 22:31:48 +02:00
$tmp['date'] = $this->date();
$tmp['dateRaw'] = $this->dateRaw();
2019-02-26 00:47:34 +01:00
$tmp['tags'] = $this->tags(false);
2019-06-16 22:31:48 +02:00
$tmp['username'] = $this->username();
2019-11-05 21:51:46 +01:00
$tmp['category'] = $this->category();
2020-08-23 18:59:18 +02:00
$tmp['uuid'] = $this->uuid();
$tmp['dateUTC'] = Date::convertToUTC($this->dateRaw(), DB_DATE_FORMAT, DB_DATE_FORMAT);
$tmp['permalink'] = $this->permalink(true);
$tmp['coverImage'] = $this->coverImage(true);
$tmp['coverImageFilename'] = $this->coverImage(false);
if ($returnsArray) {
return $tmp;
}
return json_encode($tmp);
}
2018-10-07 15:54:28 +02:00
// Returns the endpoint of the coverimage, FALSE if the page doesn't have a cover image
// (boolean) $absolute, TRUE returns the complete URL, FALSE returns the filename
// If the user defined an external cover image the function returns it
2024-06-23 21:46:43 +02:00
public function coverImage($absolute = true)
{
2018-10-07 15:54:28 +02:00
$filename = $this->getValue('coverImage');
if (empty($filename)) {
return false;
}
// Check is external cover image
2018-10-07 15:54:28 +02:00
if (filter_var($filename, FILTER_VALIDATE_URL)) {
return $filename;
}
if ($absolute) {
2018-10-07 15:54:28 +02:00
if (IMAGE_RESTRICT) {
2024-06-23 21:46:43 +02:00
return DOMAIN_UPLOADS_PAGES . $this->uuid() . '/' . $filename;
2018-10-07 15:54:28 +02:00
}
2024-06-23 21:46:43 +02:00
return DOMAIN_UPLOADS . $filename;
}
2018-10-07 15:54:28 +02:00
return $filename;
}
2018-10-07 15:54:28 +02:00
// Returns the endpoint of the thumbnail cover image, FALSE if the page doesn't have a cover image
public function thumbCoverImage()
{
$filename = $this->coverImage(false);
2024-06-23 21:46:43 +02:00
if ($filename == false) {
return false;
}
// Check is external cover image
if (filter_var($filename, FILTER_VALIDATE_URL)) {
return $filename;
}
2018-10-07 15:54:28 +02:00
if (IMAGE_RESTRICT) {
2024-06-23 21:46:43 +02:00
return DOMAIN_UPLOADS_PAGES . $this->uuid() . '/thumbnails/' . $filename;
2018-10-07 15:54:28 +02:00
}
2024-06-23 21:46:43 +02:00
return DOMAIN_UPLOADS_THUMBNAILS . $filename;
}
// Returns TRUE if the content has the text splited
public function readMore()
{
$content = $this->contentRaw();
return Text::stringContains($content, PAGE_BREAK);
}
public function uuid()
{
return $this->getValue('uuid');
}
// Returns the field key
public function key()
{
return $this->getValue('key');
}
// (boolean) Returns TRUE if the page is published, FALSE otherwise
public function published()
{
2024-06-23 21:46:43 +02:00
return ($this->getValue('type') === 'published');
}
// (boolean) Returns TRUE if the page is scheduled, FALSE otherwise
public function scheduled()
{
2024-06-23 21:46:43 +02:00
return ($this->getValue('type') === 'scheduled');
}
// (boolean) Returns TRUE if the page is draft, FALSE otherwise
public function draft()
{
2024-06-23 21:46:43 +02:00
return ($this->getValue('type') == 'draft');
}
2019-05-26 23:08:50 +02:00
// (boolean) Returns TRUE if the page is autosave, FALSE otherwise
public function autosave()
{
2024-06-23 21:46:43 +02:00
return ($this->getValue('type') == 'autosave');
2019-05-26 23:08:50 +02:00
}
// (boolean) Returns TRUE if the page is sticky, FALSE otherwise
public function sticky()
{
2024-06-23 21:46:43 +02:00
return ($this->getValue('type') == 'sticky');
}
// (boolean) Returns TRUE if the page is static, FALSE otherwise
public function isStatic()
{
2024-06-23 21:46:43 +02:00
return ($this->getValue('type') == 'static');
}
// (string) Returns type of the page
public function type()
{
return $this->getValue('type');
}
// Returns the title field
public function title()
{
return $this->getValue('title');
}
// Returns TRUE if the page has enabled the comments, FALSE otherwise
public function allowComments()
{
return $this->getValue('allowComments');
}
// Returns the page position
public function position()
{
return $this->getValue('position');
}
// Returns the page noindex
public function noindex()
{
return $this->getValue('noindex');
}
// Returns the page nofollow
public function nofollow()
{
return $this->getValue('nofollow');
}
// Returns the page noarchive
public function noarchive()
{
return $this->getValue('noarchive');
}
// Returns the page slug
public function slug()
{
$explode = explode('/', $this->key());
2019-02-08 08:53:26 +01:00
return end($explode);
}
// Returns the parent key, if the page doesn't have a parent returns FALSE
public function parent()
{
return $this->parentKey();
}
// Returns the parent key, if the page doesn't have a parent returns FALSE
public function parentKey()
{
$explode = explode('/', $this->key());
if (isset($explode[1])) {
return $explode[0];
}
return false;
}
// Returns TRUE if the page is a parent, has or not children
public function isParent()
{
2024-06-23 21:46:43 +02:00
return $this->parentKey() === false;
}
// Returns the parent method output, if the page doesn't have a parent returns FALSE
public function parentMethod($method)
{
$parentKey = $this->parentKey();
if ($parentKey) {
try {
2018-08-02 17:06:53 +02:00
$page = new Page($parentKey);
return $page->{$method}();
} catch (Exception $e) {
// Continoue
}
}
return false;
}
// Returns TRUE if the page is a child, FALSE otherwise
public function isChild()
{
2024-06-23 21:46:43 +02:00
return $this->parentKey() !== false;
}
// Returns TRUE if the page has children
public function hasChildren()
{
$childrenKeys = $this->childrenKeys();
return !empty($childrenKeys);
}
// Returns an array with all children's keys
public function childrenKeys()
{
2018-08-03 18:59:23 +02:00
global $pages;
$key = $this->key();
2018-08-03 18:59:23 +02:00
return $pages->getChildren($key);
}
// Returns an array with all children as Page-Object
public function children()
{
2018-08-03 18:59:23 +02:00
global $pages;
$list = array();
2018-08-03 18:59:23 +02:00
$childrenKeys = $pages->getChildren($this->key());
foreach ($childrenKeys as $childKey) {
try {
2018-08-02 17:06:53 +02:00
$child = new Page($childKey);
array_push($list, $child);
} catch (Exception $e) {
// Continue
}
}
return $list;
}
// Returns the amount of minutes takes to read the page
2024-06-23 21:46:43 +02:00
public function readingTime()
{
global $L;
$words = $this->content(true);
$words = strip_tags($words);
$words = str_word_count($words);
$average = $words / 200;
$minutes = round($average);
2024-06-23 21:46:43 +02:00
if ($minutes > 1) {
return $minutes . ' ' . $L->get('minutes');
}
2024-06-23 21:46:43 +02:00
return '~1 ' . $L->get('minute');
}
// Returns relative time (e.g. "1 minute ago")
// Based on http://stackoverflow.com/a/18602474
// Modified for Bludit
// $complete = false : short version
// $complete = true : full version
2024-06-23 21:46:43 +02:00
public function relativeTime($complete = false)
{
$current = new DateTime;
$past = new DateTime($this->getValue('dateRaw'));
$elapsed = $current->diff($past);
// Calculate weeks separately
$weeks = floor($elapsed->d / 7);
$elapsed->d -= $weeks * 7;
$string = array(
'y' => 'year',
'm' => 'month',
'w' => $weeks,
'd' => 'day',
'h' => 'hour',
'i' => 'minute',
's' => 'second',
);
foreach ($string as $key => &$value) {
if ($key == 'w') {
if ($weeks > 0) {
$value = $weeks . ' week' . ($weeks > 1 ? 's' : '');
} else {
unset($string[$key]);
}
} elseif ($elapsed->$key) {
$value = $elapsed->$key . ' ' . $value . ($elapsed->$key > 1 ? 's' : '');
} else {
unset($string[$key]);
}
}
if (!$complete) {
$string = array_slice($string, 0, 1);
}
return $string ? implode(', ', $string) . ' ago' : 'Just now';
}
2019-09-02 18:24:34 +02:00
// Returns the value from the field, false if the fields doesn't exists
// If you set the $option as TRUE, the function returns an array with all the values of the field
2024-06-23 21:46:43 +02:00
public function custom($field, $options = false)
2019-09-02 18:24:34 +02:00
{
if (isset($this->vars['custom'][$field])) {
if ($options) {
return $this->vars['custom'][$field];
}
return $this->vars['custom'][$field]['value'];
}
return false;
}
2020-07-23 12:14:43 +02:00
// Returns an array with all pages key related to the page
// The relation is based on the tags
2024-06-23 21:46:43 +02:00
public function related()
{
2020-07-23 12:14:43 +02:00
global $tags;
$pageTags = $this->tags(true);
$list = array();
// For each tag get the list of related pages
2024-06-23 21:46:43 +02:00
foreach ($pageTags as $tagKey => $tagName) {
2020-07-23 12:14:43 +02:00
$pagesRelated = $tags->getList($tagKey, 1, -1);
2024-06-23 21:46:43 +02:00
if (is_array($pagesRelated)) {
$list = array_merge($list, $pagesRelated);
}
2020-07-23 12:14:43 +02:00
}
// Remove duplicates
$list = array_unique($list);
// Remove himself from the list
if (($key = array_search($this->key(), $list)) !== false) {
unset($list[$key]);
}
return $list;
}
}