674 lines
		
	
	
		
			20 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			674 lines
		
	
	
		
			20 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php declare(strict_types=1);
 | |
| require_once "core/event.php";
 | |
| 
 | |
| abstract class PageMode
 | |
| {
 | |
|     const REDIRECT = 'redirect';
 | |
|     const DATA = 'data';
 | |
|     const PAGE = 'page';
 | |
|     const FILE = 'file';
 | |
|     const MANUAL = 'manual';
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Class Page
 | |
|  *
 | |
|  * A data structure for holding all the bits of data that make up a page.
 | |
|  *
 | |
|  * The various extensions all add whatever they want to this structure,
 | |
|  * then Layout turns it into HTML.
 | |
|  */
 | |
| class BasePage
 | |
| {
 | |
|     /** @var string */
 | |
|     public $mode = PageMode::PAGE;
 | |
|     /** @var string */
 | |
|     private $type = "text/html; charset=utf-8";
 | |
| 
 | |
|     /**
 | |
|      * Set what this page should do; "page", "data", or "redirect".
 | |
|      */
 | |
|     public function set_mode(string $mode): void
 | |
|     {
 | |
|         $this->mode = $mode;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Set the page's MIME type.
 | |
|      */
 | |
|     public function set_type(string $type): void
 | |
|     {
 | |
|         $this->type = $type;
 | |
|     }
 | |
| 
 | |
|     public function __construct()
 | |
|     {
 | |
|         if (@$_GET["flash"]) {
 | |
|             $this->flash[] = $_GET['flash'];
 | |
|             unset($_GET["flash"]);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     // ==============================================
 | |
| 
 | |
|     /** @var string; public only for unit test */
 | |
|     public $data = "";
 | |
| 
 | |
|     /** @var string */
 | |
|     private $file = null;
 | |
| 
 | |
|     /** @var bool */
 | |
|     private $file_delete = false;
 | |
| 
 | |
|     /** @var string */
 | |
|     private $filename = null;
 | |
| 
 | |
|     private $disposition = null;
 | |
| 
 | |
|     /**
 | |
|      * Set the raw data to be sent.
 | |
|      */
 | |
|     public function set_data(string $data): void
 | |
|     {
 | |
|         $this->data = $data;
 | |
|     }
 | |
| 
 | |
|     public function set_file(string $file, bool $delete = false): void
 | |
|     {
 | |
|         $this->file = $file;
 | |
|         $this->file_delete = $delete;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Set the recommended download filename.
 | |
|      */
 | |
|     public function set_filename(string $filename, string $disposition = "attachment"): void
 | |
|     {
 | |
|         $this->filename = $filename;
 | |
|         $this->disposition = $disposition;
 | |
|     }
 | |
| 
 | |
|     // ==============================================
 | |
| 
 | |
|     /** @var string */
 | |
|     public $redirect = "";
 | |
| 
 | |
|     /**
 | |
|      * Set the URL to redirect to (remember to use make_link() if linking
 | |
|      * to a page in the same site).
 | |
|      */
 | |
|     public function set_redirect(string $redirect): void
 | |
|     {
 | |
|         $this->redirect = $redirect;
 | |
|     }
 | |
| 
 | |
|     // ==============================================
 | |
| 
 | |
|     /** @var int */
 | |
|     public $code = 200;
 | |
| 
 | |
|     /** @var string */
 | |
|     public $title = "";
 | |
| 
 | |
|     /** @var string */
 | |
|     public $heading = "";
 | |
| 
 | |
|     /** @var string */
 | |
|     public $subheading = "";
 | |
| 
 | |
|     /** @var string[] */
 | |
|     public $html_headers = [];
 | |
| 
 | |
|     /** @var string[] */
 | |
|     public $http_headers = [];
 | |
| 
 | |
|     /** @var string[][] */
 | |
|     public $cookies = [];
 | |
| 
 | |
|     /** @var Block[] */
 | |
|     public $blocks = [];
 | |
| 
 | |
|     /** @var string[] */
 | |
|     public $flash = [];
 | |
| 
 | |
|     /**
 | |
|      * Set the HTTP status code
 | |
|      */
 | |
|     public function set_code(int $code): void
 | |
|     {
 | |
|         $this->code = $code;
 | |
|     }
 | |
| 
 | |
|     public function set_title(string $title): void
 | |
|     {
 | |
|         $this->title = $title;
 | |
|     }
 | |
| 
 | |
|     public function set_heading(string $heading): void
 | |
|     {
 | |
|         $this->heading = $heading;
 | |
|     }
 | |
| 
 | |
|     public function set_subheading(string $subheading): void
 | |
|     {
 | |
|         $this->subheading = $subheading;
 | |
|     }
 | |
| 
 | |
|     public function flash(string $message): void
 | |
|     {
 | |
|         $this->flash[] = $message;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Add a line to the HTML head section.
 | |
|      */
 | |
|     public function add_html_header(string $line, int $position = 50): void
 | |
|     {
 | |
|         while (isset($this->html_headers[$position])) {
 | |
|             $position++;
 | |
|         }
 | |
|         $this->html_headers[$position] = $line;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Add a http header to be sent to the client.
 | |
|      */
 | |
|     public function add_http_header(string $line, int $position = 50): void
 | |
|     {
 | |
|         while (isset($this->http_headers[$position])) {
 | |
|             $position++;
 | |
|         }
 | |
|         $this->http_headers[$position] = $line;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * The counterpart for get_cookie, this works like php's
 | |
|      * setcookie method, but prepends the site-wide cookie prefix to
 | |
|      * the $name argument before doing anything.
 | |
|      */
 | |
|     public function add_cookie(string $name, string $value, int $time, string $path): void
 | |
|     {
 | |
|         $full_name = COOKIE_PREFIX . "_" . $name;
 | |
|         $this->cookies[] = [$full_name, $value, $time, $path];
 | |
|     }
 | |
| 
 | |
|     public function get_cookie(string $name): ?string
 | |
|     {
 | |
|         $full_name = COOKIE_PREFIX . "_" . $name;
 | |
|         if (isset($_COOKIE[$full_name])) {
 | |
|             return $_COOKIE[$full_name];
 | |
|         } else {
 | |
|             return null;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get all the HTML headers that are currently set and return as a string.
 | |
|      */
 | |
|     public function get_all_html_headers(): string
 | |
|     {
 | |
|         $data = '';
 | |
|         ksort($this->html_headers);
 | |
|         foreach ($this->html_headers as $line) {
 | |
|             $data .= "\t\t" . $line . "\n";
 | |
|         }
 | |
|         return $data;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Add a Block of data to the page.
 | |
|      */
 | |
|     public function add_block(Block $block): void
 | |
|     {
 | |
|         $this->blocks[] = $block;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Find a block which contains the given text
 | |
|      * (Useful for unit tests)
 | |
|      */
 | |
|     public function find_block(string $text): ?Block
 | |
|     {
 | |
|         foreach ($this->blocks as $block) {
 | |
|             if ($block->header == $text) {
 | |
|                 return $block;
 | |
|             }
 | |
|         }
 | |
|         return null;
 | |
|     }
 | |
| 
 | |
|     // ==============================================
 | |
| 
 | |
|     public function send_headers(): void
 | |
|     {
 | |
|         if (!headers_sent()) {
 | |
|             header("HTTP/1.0 {$this->code} Shimmie");
 | |
|             header("Content-type: " . $this->type);
 | |
|             header("X-Powered-By: Shimmie-" . VERSION);
 | |
| 
 | |
|             foreach ($this->http_headers as $head) {
 | |
|                 header($head);
 | |
|             }
 | |
|             foreach ($this->cookies as $c) {
 | |
|                 setcookie($c[0], $c[1], $c[2], $c[3]);
 | |
|             }
 | |
|         } else {
 | |
|             print "Error: Headers have already been sent to the client.";
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     public function flush_output(): void
 | |
|     {
 | |
| 
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Display the page according to the mode and data given.
 | |
|      */
 | |
|     public function display(): void
 | |
|     {
 | |
|         if($this->mode!=PageMode::MANUAL) {
 | |
|             $this->send_headers();
 | |
|         }
 | |
| 
 | |
|         switch ($this->mode) {
 | |
|             case PageMode::MANUAL:
 | |
|                 break;
 | |
|             case PageMode::PAGE:
 | |
|                 usort($this->blocks, "blockcmp");
 | |
|                 $this->add_auto_html_headers();
 | |
|                 $this->render();
 | |
|                 break;
 | |
|             case PageMode::DATA:
 | |
|                 header("Content-Length: " . strlen($this->data));
 | |
|                 if (!is_null($this->filename)) {
 | |
|                     header('Content-Disposition: ' . $this->disposition . '; filename=' . $this->filename);
 | |
|                 }
 | |
|                 print $this->data;
 | |
|                 break;
 | |
|             case PageMode::FILE:
 | |
|                 if (!is_null($this->filename)) {
 | |
|                     header('Content-Disposition: ' . $this->disposition . '; filename=' . $this->filename);
 | |
|                 }
 | |
| 
 | |
|                 // https://gist.github.com/codler/3906826
 | |
|                 $size = filesize($this->file); // File size
 | |
|                 $length = $size;           // Content length
 | |
|                 $start = 0;               // Start byte
 | |
|                 $end = $size - 1;       // End byte
 | |
| 
 | |
|                 header("Content-Length: " . $size);
 | |
|                 header('Accept-Ranges: bytes');
 | |
| 
 | |
|                 if (isset($_SERVER['HTTP_RANGE'])) {
 | |
|                     list(, $range) = explode('=', $_SERVER['HTTP_RANGE'], 2);
 | |
|                     if (strpos($range, ',') !== false) {
 | |
|                         header('HTTP/1.1 416 Requested Range Not Satisfiable');
 | |
|                         header("Content-Range: bytes $start-$end/$size");
 | |
|                         break;
 | |
|                     }
 | |
|                     if ($range == '-') {
 | |
|                         $c_start = $size - (int)substr($range, 1);
 | |
|                         $c_end = $end;
 | |
|                     } else {
 | |
|                         $range = explode('-', $range);
 | |
|                         $c_start = (int)$range[0];
 | |
|                         $c_end = (isset($range[1]) && is_numeric($range[1])) ? (int)$range[1] : $size;
 | |
|                     }
 | |
|                     $c_end = ($c_end > $end) ? $end : $c_end;
 | |
|                     if ($c_start > $c_end || $c_start > $size - 1 || $c_end >= $size) {
 | |
|                         header('HTTP/1.1 416 Requested Range Not Satisfiable');
 | |
|                         header("Content-Range: bytes $start-$end/$size");
 | |
|                         break;
 | |
|                     }
 | |
|                     $start = $c_start;
 | |
|                     $end = $c_end;
 | |
|                     $length = $end - $start + 1;
 | |
|                     header('HTTP/1.1 206 Partial Content');
 | |
|                 }
 | |
|                 header("Content-Range: bytes $start-$end/$size");
 | |
|                 header("Content-Length: " . $length);
 | |
| 
 | |
|                 try {
 | |
|                     stream_file($this->file, $start, $end);
 | |
|                 } finally {
 | |
|                     if ($this->file_delete === true) {
 | |
|                         unlink($this->file);
 | |
|                     }
 | |
|                 }
 | |
|                 break;
 | |
|             case PageMode::REDIRECT:
 | |
|                 if ($this->flash) {
 | |
|                     $this->redirect .= (strpos($this->redirect, "?") === false) ? "?" : "&";
 | |
|                     $this->redirect .= "flash=" . url_escape(implode("\n", $this->flash));
 | |
|                 }
 | |
|                 header('Location: ' . $this->redirect);
 | |
|                 print 'You should be redirected to <a href="' . $this->redirect . '">' . $this->redirect . '</a>';
 | |
|                 break;
 | |
|             default:
 | |
|                 print "Invalid page mode";
 | |
|                 break;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * This function grabs all the CSS and JavaScript files sprinkled throughout Shimmie's folders,
 | |
|      * concatenates them together into two large files (one for CSS and one for JS) and then stores
 | |
|      * them in the /cache/ directory for serving to the user.
 | |
|      *
 | |
|      * Why do this? Two reasons:
 | |
|      *  1. Reduces the number of files the user's browser needs to download.
 | |
|      *  2. Allows these cached files to be compressed/minified by the admin.
 | |
|      *
 | |
|      * TODO: This should really be configurable somehow...
 | |
|      */
 | |
|     public function add_auto_html_headers(): void
 | |
|     {
 | |
|         global $config;
 | |
| 
 | |
|         $data_href = get_base_href();
 | |
|         $theme_name = $config->get_string(SetupConfig::THEME, 'default');
 | |
| 
 | |
|         $this->add_html_header("<script type='text/javascript'>base_href = '$data_href';</script>", 40);
 | |
| 
 | |
|         # static handler will map these to themes/foo/static/bar.ico or ext/static_files/static/bar.ico
 | |
|         $this->add_html_header("<link rel='icon' type='image/x-icon' href='$data_href/favicon.ico'>", 41);
 | |
|         $this->add_html_header("<link rel='apple-touch-icon' href='$data_href/apple-touch-icon.png'>", 42);
 | |
| 
 | |
|         //We use $config_latest to make sure cache is reset if config is ever updated.
 | |
|         $config_latest = 0;
 | |
|         foreach (zglob("data/config/*") as $conf) {
 | |
|             $config_latest = max($config_latest, filemtime($conf));
 | |
|         }
 | |
| 
 | |
|         /*** Generate CSS cache files ***/
 | |
|         $css_latest = $config_latest;
 | |
|         $css_files = array_merge(
 | |
|             zglob("ext/{" . Extension::get_enabled_extensions_as_string() . "}/style.css"),
 | |
|             zglob("themes/$theme_name/style.css")
 | |
|         );
 | |
|         foreach ($css_files as $css) {
 | |
|             $css_latest = max($css_latest, filemtime($css));
 | |
|         }
 | |
|         $css_md5 = md5(serialize($css_files));
 | |
|         $css_cache_file = data_path("cache/style/{$theme_name}.{$css_latest}.{$css_md5}.css");
 | |
|         if (!file_exists($css_cache_file)) {
 | |
|             $css_data = "";
 | |
|             foreach ($css_files as $file) {
 | |
|                 $file_data = file_get_contents($file);
 | |
|                 $pattern = '/url[\s]*\([\s]*["\']?([^"\'\)]+)["\']?[\s]*\)/';
 | |
|                 $replace = 'url("../../../' . dirname($file) . '/$1")';
 | |
|                 $file_data = preg_replace($pattern, $replace, $file_data);
 | |
|                 $css_data .= $file_data . "\n";
 | |
|             }
 | |
|             file_put_contents($css_cache_file, $css_data);
 | |
|         }
 | |
|         $this->add_html_header("<link rel='stylesheet' href='$data_href/$css_cache_file' type='text/css'>", 43);
 | |
| 
 | |
|         /*** Generate JS cache files ***/
 | |
|         $js_latest = $config_latest;
 | |
|         $js_files = array_merge(
 | |
|             [
 | |
|                 "vendor/bower-asset/jquery/dist/jquery.min.js",
 | |
|                 "vendor/bower-asset/jquery-timeago/jquery.timeago.js",
 | |
|                 "vendor/bower-asset/js-cookie/src/js.cookie.js",
 | |
|                 "ext/static_files/modernizr-3.3.1.custom.js",
 | |
|             ],
 | |
|             zglob("ext/{" . Extension::get_enabled_extensions_as_string() . "}/script.js"),
 | |
|             zglob("themes/$theme_name/script.js")
 | |
|         );
 | |
|         foreach ($js_files as $js) {
 | |
|             $js_latest = max($js_latest, filemtime($js));
 | |
|         }
 | |
|         $js_md5 = md5(serialize($js_files));
 | |
|         $js_cache_file = data_path("cache/script/{$theme_name}.{$js_latest}.{$js_md5}.js");
 | |
|         if (!file_exists($js_cache_file)) {
 | |
|             $js_data = "";
 | |
|             foreach ($js_files as $file) {
 | |
|                 $js_data .= file_get_contents($file) . "\n";
 | |
|             }
 | |
|             file_put_contents($js_cache_file, $js_data);
 | |
|         }
 | |
|         $this->add_html_header("<script defer src='$data_href/$js_cache_file' type='text/javascript'></script>", 44);
 | |
|     }
 | |
| 
 | |
|     protected function get_nav_links()
 | |
|     {
 | |
|         $pnbe = send_event(new PageNavBuildingEvent());
 | |
| 
 | |
|         $nav_links = $pnbe->links;
 | |
| 
 | |
|         $active_link = null;
 | |
|         // To save on event calls, we check if one of the top-level links has already been marked as active
 | |
|         foreach ($nav_links as $link) {
 | |
|             if ($link->active===true) {
 | |
|                 $active_link = $link;
 | |
|                 break;
 | |
|             }
 | |
|         }
 | |
|         $sub_links = null;
 | |
|         // If one is, we just query for sub-menu options under that one tab
 | |
|         if ($active_link!==null) {
 | |
|             $psnbe = send_event(new PageSubNavBuildingEvent($active_link->name));
 | |
|             $sub_links = $psnbe->links;
 | |
|         } else {
 | |
|             // Otherwise we query for the sub-items under each of the tabs
 | |
|             foreach ($nav_links as $link) {
 | |
|                 $psnbe = send_event(new PageSubNavBuildingEvent($link->name));
 | |
| 
 | |
|                 // Now we check for a current link so we can identify the sub-links to show
 | |
|                 foreach ($psnbe->links as $sub_link) {
 | |
|                     if ($sub_link->active===true) {
 | |
|                         $sub_links = $psnbe->links;
 | |
|                         break;
 | |
|                     }
 | |
|                 }
 | |
|                 // If the active link has been detected, we break out
 | |
|                 if ($sub_links!==null) {
 | |
|                     $link->active = true;
 | |
|                     break;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         $sub_links = $sub_links??[];
 | |
|         usort($nav_links, "sort_nav_links");
 | |
|         usort($sub_links, "sort_nav_links");
 | |
| 
 | |
|         return [$nav_links, $sub_links];
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * turns the Page into HTML
 | |
|      */
 | |
|     public function render()
 | |
|     {
 | |
|         $head_html = $this->head_html();
 | |
|         $body_html = $this->body_html();
 | |
| 
 | |
|         print <<<EOD
 | |
| <!doctype html>
 | |
| <html class="no-js" lang="en">
 | |
|     $head_html
 | |
|     $body_html
 | |
| </html>
 | |
| EOD;
 | |
|     }
 | |
| 
 | |
|     protected function head_html(): string
 | |
|     {
 | |
|         $html_header_html = $this->get_all_html_headers();
 | |
| 
 | |
|         return "
 | |
|         <head>
 | |
| 		    <title>{$this->title}</title>
 | |
|             $html_header_html
 | |
| 	    </head>
 | |
|         ";
 | |
|     }
 | |
| 
 | |
|     protected function body_html(): string
 | |
|     {
 | |
|         $left_block_html = "";
 | |
|         $main_block_html = "";
 | |
|         $sub_block_html  = "";
 | |
| 
 | |
|         foreach ($this->blocks as $block) {
 | |
|             switch ($block->section) {
 | |
|                 case "left":
 | |
|                     $left_block_html .= $block->get_html(true);
 | |
|                     break;
 | |
|                 case "main":
 | |
|                     $main_block_html .= $block->get_html(false);
 | |
|                     break;
 | |
|                 case "subheading":
 | |
|                     $sub_block_html .= $block->get_html(false);
 | |
|                     break;
 | |
|                 default:
 | |
|                     print "<p>error: {$block->header} using an unknown section ({$block->section})";
 | |
|                     break;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         $wrapper = "";
 | |
|         if (strlen($this->heading) > 100) {
 | |
|             $wrapper = ' style="height: 3em; overflow: auto;"';
 | |
|         }
 | |
| 
 | |
|         $footer_html = $this->footer_html();
 | |
|         $flash_html = $this->flash ? "<b id='flash'>".nl2br(html_escape(implode("\n", $this->flash)))."</b>" : "";
 | |
|         return "
 | |
|             <body>
 | |
|                 <header>
 | |
|                     <h1$wrapper>{$this->heading}</h1>
 | |
|                     $sub_block_html
 | |
|                 </header>
 | |
|                 <nav>
 | |
|                     $left_block_html
 | |
|                 </nav>
 | |
|                 <article>
 | |
|                     $flash_html
 | |
|                     $main_block_html
 | |
|                 </article>
 | |
|                 <footer>
 | |
|                     $footer_html
 | |
|                 </footer>
 | |
|             </body>
 | |
|         ";
 | |
|     }
 | |
| 
 | |
|     protected function footer_html(): string
 | |
|     {
 | |
|         $debug = get_debug_info();
 | |
|         $contact_link = contact_link();
 | |
|         $contact = empty($contact_link) ? "" : "<br><a href='$contact_link'>Contact</a>";
 | |
| 
 | |
|         return "
 | |
| 			Images © their respective owners,
 | |
| 			<a href=\"https://code.shishnet.org/shimmie2/\">Shimmie</a> ©
 | |
| 			<a href=\"https://www.shishnet.org/\">Shish</a> &
 | |
| 			<a href=\"https://github.com/shish/shimmie2/graphs/contributors\">The Team</a>
 | |
| 			2007-2020,
 | |
| 			based on the Danbooru concept.
 | |
| 			$debug
 | |
| 			$contact
 | |
|         ";
 | |
|     }
 | |
| }
 | |
| 
 | |
| class PageNavBuildingEvent extends Event
 | |
| {
 | |
|     public $links = [];
 | |
| 
 | |
|     public function add_nav_link(string $name, Link $link, string $desc, ?bool $active = null, int $order = 50)
 | |
|     {
 | |
|         $this->links[]  = new NavLink($name, $link, $desc, $active, $order);
 | |
|     }
 | |
| }
 | |
| 
 | |
| class PageSubNavBuildingEvent extends Event
 | |
| {
 | |
|     public $parent;
 | |
| 
 | |
|     public $links = [];
 | |
| 
 | |
|     public function __construct(string $parent)
 | |
|     {
 | |
|         parent::__construct();
 | |
|         $this->parent= $parent;
 | |
|     }
 | |
| 
 | |
|     public function add_nav_link(string $name, Link $link, string $desc, ?bool $active = null, int $order = 50)
 | |
|     {
 | |
|         $this->links[]  = new NavLink($name, $link, $desc, $active, $order);
 | |
|     }
 | |
| }
 | |
| 
 | |
| class NavLink
 | |
| {
 | |
|     public $name;
 | |
|     public $link;
 | |
|     public $description;
 | |
|     public $order;
 | |
|     public $active = false;
 | |
| 
 | |
|     public function __construct(String $name, Link $link, String $description, ?bool $active = null, int $order = 50)
 | |
|     {
 | |
|         global $config;
 | |
| 
 | |
|         $this->name = $name;
 | |
|         $this->link = $link;
 | |
|         $this->description = $description;
 | |
|         $this->order = $order;
 | |
|         if ($active==null) {
 | |
|             $query = ltrim(_get_query(), "/");
 | |
|             if ($query === "") {
 | |
|                 // This indicates the front page, so we check what's set as the front page
 | |
|                 $front_page = trim($config->get_string(SetupConfig::FRONT_PAGE), "/");
 | |
| 
 | |
|                 if ($front_page === $link->page) {
 | |
|                     $this->active = true;
 | |
|                 } else {
 | |
|                     $this->active = self::is_active([$link->page], $front_page);
 | |
|                 }
 | |
|             } elseif ($query===$link->page) {
 | |
|                 $this->active = true;
 | |
|             } else {
 | |
|                 $this->active = self::is_active([$link->page]);
 | |
|             }
 | |
|         } else {
 | |
|             $this->active = $active;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     public static function is_active(array $pages_matched, string $url = null): bool
 | |
|     {
 | |
|         /**
 | |
|          * Woo! We can actually SEE THE CURRENT PAGE!! (well... see it highlighted in the menu.)
 | |
|          */
 | |
|         $url = $url??ltrim(_get_query(), "/");
 | |
| 
 | |
|         $re1='.*?';
 | |
|         $re2='((?:[a-z][a-z_]+))';
 | |
| 
 | |
|         if (preg_match_all("/".$re1.$re2."/is", $url, $matches)) {
 | |
|             $url=$matches[1][0];
 | |
|         }
 | |
| 
 | |
|         $count_pages_matched = count($pages_matched);
 | |
| 
 | |
|         for ($i=0; $i < $count_pages_matched; $i++) {
 | |
|             if ($url == $pages_matched[$i]) {
 | |
|                 return true;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return false;
 | |
|     }
 | |
| }
 | |
| 
 | |
| function sort_nav_links(NavLink $a, NavLink $b)
 | |
| {
 | |
|     return $a->order - $b->order;
 | |
| }
 |