You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
182 lines
4.7 KiB
182 lines
4.7 KiB
1 year ago
|
<?php
|
||
|
/*
|
||
|
** Zabbix
|
||
|
** Copyright (C) 2001-2023 Zabbix SIA
|
||
|
**
|
||
|
** This program is free software; you can redistribute it and/or modify
|
||
|
** it under the terms of the GNU General Public License as published by
|
||
|
** the Free Software Foundation; either version 2 of the License, or
|
||
|
** (at your option) any later version.
|
||
|
**
|
||
|
** This program is distributed in the hope that it will be useful,
|
||
|
** but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
|
** MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
|
** GNU General Public License for more details.
|
||
|
**
|
||
|
** You should have received a copy of the GNU General Public License
|
||
|
** along with this program; if not, write to the Free Software
|
||
|
** Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||
|
**/
|
||
|
|
||
|
|
||
|
/**
|
||
|
* Class for rendering partial-views, which are useful for reusing the same partial-views or updating specific parts of
|
||
|
* the page asynchronously, i.e., partials can be included into a view initially and updated in AJAX manner, later.
|
||
|
*/
|
||
|
class CPartial {
|
||
|
|
||
|
/**
|
||
|
* Directory list of MVC partials ordered by search priority.
|
||
|
*
|
||
|
* @static
|
||
|
*
|
||
|
* @var array
|
||
|
*/
|
||
|
private static $directories = ['local/app/partials', 'app/partials'];
|
||
|
|
||
|
/**
|
||
|
* Partial name.
|
||
|
*
|
||
|
* @var string
|
||
|
*/
|
||
|
private $name;
|
||
|
|
||
|
/**
|
||
|
* Data provided for partial.
|
||
|
*
|
||
|
* @var array
|
||
|
*/
|
||
|
private $data;
|
||
|
|
||
|
/**
|
||
|
* Directory where the partial file was found.
|
||
|
*
|
||
|
* @var string
|
||
|
*/
|
||
|
private $directory;
|
||
|
|
||
|
/**
|
||
|
* Create a partial based on partial name and data.
|
||
|
*
|
||
|
* @param string $name Partial name to search for.
|
||
|
* @param array $data Accessible data within the partial.
|
||
|
*
|
||
|
* @throws InvalidArgumentException if partial name not valid.
|
||
|
* @throws RuntimeException if partial not found or not readable.
|
||
|
*/
|
||
|
public function __construct($name, array $data = []) {
|
||
|
if (!preg_match('/^[a-z]+(\/[a-z]+)*(\.[a-z]+)*$/', $name)) {
|
||
|
throw new InvalidArgumentException(sprintf('Invalid partial name: "%s".', $name));
|
||
|
}
|
||
|
|
||
|
$file_path = null;
|
||
|
|
||
|
foreach (self::$directories as $directory) {
|
||
|
$file_path = $directory.'/'.$name.'.php';
|
||
|
if (is_file($file_path)) {
|
||
|
$this->directory = $directory;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($this->directory === null) {
|
||
|
throw new RuntimeException(sprintf('Partial not found: "%s".', $name));
|
||
|
}
|
||
|
|
||
|
if (!is_readable($file_path)) {
|
||
|
throw new RuntimeException(sprintf('Partial not readable: "%s".', $file_path));
|
||
|
}
|
||
|
|
||
|
$this->name = $name;
|
||
|
$this->data = $data;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Render partial and return the output.
|
||
|
* Note: partial should only output textual content like HTML, JSON, scripts or similar.
|
||
|
*
|
||
|
* @throws RuntimeException if partial not found, not readable or returned false.
|
||
|
*
|
||
|
* @return string
|
||
|
*/
|
||
|
public function getOutput() {
|
||
|
$data = $this->data;
|
||
|
|
||
|
$file_path = $this->directory.'/'.$this->name.'.php';
|
||
|
|
||
|
ob_start();
|
||
|
|
||
|
if ((include $file_path) === false) {
|
||
|
ob_end_clean();
|
||
|
|
||
|
throw new RuntimeException(sprintf('Cannot render partial: "%s".', $file_path));
|
||
|
}
|
||
|
|
||
|
return ob_get_clean();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get the contents of a PHP-preprocessed JavaScript file.
|
||
|
* Notes:
|
||
|
* - JavaScript file will be searched in the "js" subdirectory of the partial file.
|
||
|
* - A copy of $data variable will be available for using within the file.
|
||
|
*
|
||
|
* @param string $file_name
|
||
|
* @param array $data
|
||
|
*
|
||
|
* @throws RuntimeException if the file not found, not readable or returned false.
|
||
|
*
|
||
|
* @return string
|
||
|
*/
|
||
|
public function readJsFile(string $file_name, array $data = null): string {
|
||
|
$data = ($data === null) ? $this->data : $data;
|
||
|
|
||
|
$file_path = $this->directory.'/js/'.$file_name;
|
||
|
|
||
|
ob_start();
|
||
|
|
||
|
if ((include $file_path) === false) {
|
||
|
ob_end_clean();
|
||
|
|
||
|
throw new RuntimeException(sprintf('Cannot read file: "%s".', $file_path));
|
||
|
}
|
||
|
|
||
|
return ob_get_clean();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Include a PHP-preprocessed JavaScript file inline.
|
||
|
* Notes:
|
||
|
* - JavaScript file will be searched in the "js" subdirectory of the partial file.
|
||
|
* - A copy of $data variable will be available for using within the file.
|
||
|
*
|
||
|
* @param string $file_name
|
||
|
* @param array $data
|
||
|
*
|
||
|
* @throws RuntimeException if the file not found, not readable or returned false.
|
||
|
*/
|
||
|
public function includeJsFile(string $file_name, array $data = null): void {
|
||
|
echo $this->readJsFile($file_name, $data);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Register custom directory of MVC partials. The last registered will have the first priority.
|
||
|
*
|
||
|
* @param string $directory
|
||
|
*/
|
||
|
public static function registerDirectory($directory) {
|
||
|
if (!in_array($directory, self::$directories)) {
|
||
|
array_unshift(self::$directories, $directory);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get partial file name.
|
||
|
*
|
||
|
* @return string
|
||
|
*/
|
||
|
public function getName(): string {
|
||
|
return $this->name;
|
||
|
}
|
||
|
}
|